Using Framework Locks

Sometimes drivers must provide driver-specific synchronization of I/O request-related callback functions, either in addition to or as a replacement for framework-supplied synchronization. Drivers can use callback synchronization locks, spin locks, wait locks, and interrupt locks to synchronize driver code.

Callback Synchronization Locks

If you have set up your driver to use the framework's automatic synchronization capability, the framework acquires a synchronization lock before calling the driver's I/O request-related event callback functions.

These callback synchronization locks, which are associated with framework device objects and queue objects, can also be acquired by drivers. To acquire a synchronization lock, a driver calls WdfObjectAcquireLock. To release the lock, the driver calls WdfObjectReleaseLock.

You might want your driver to use the callback synchronization locks if the driver uses the framework's device-level or queue-level synchronization of I/O request-related callback functions but must synchronize some code that runs at IRQL = PASSIVE_LEVEL with callback functions that run at IRQL = DISPATCH_LEVEL. This is because drivers can use automatic synchronization only for callback functions that execute at the same IRQL.

For example, a driver can use automatic synchronization for a work-item object only if the execution level of the work-item object's parent is WdfExecutionLevelPassive (because a work item's callback function always executes at IRQL= PASSIVE_LEVEL). Therefore, if a driver specifies WdfExecutionLevelDispatch in the ExecutionLevel member of a device object's WDF_OBJECT_ATTRIBUTES structure, the driver cannot set the AutomaticSerialization member of a child work-item object's configuration structure. Instead, the driver must acquire a callback synchronization lock to synchronize the EvtWorkItem callback functions with the parent device object's callback functions.

Framework Wait Locks

Use framework wait locks to synchronize access to driver data from code that runs at IRQL = PASSIVE_LEVEL. Before a driver can use a framework wait lock, it must call WdfWaitLockCreate to create a wait-lock object. The driver can then call WdfWaitLockAcquire to acquire the lock and WdfWaitLockRelease to release it.

Framework Spin Locks

Use framework spin locks to synchronize access to driver data from code that runs at IRQL <= DISPATCH_LEVEL. When a driver thread acquires a spin lock, the system sets the thread's IRQL to DISPATCH_LEVEL. When the thread releases the lock, the system restores the thread's IRQL to its previous level.

A driver that is not using automatic framework synchronization might use a spin lock to synchronize access to a device object's context space, if the context space is writable and if more than one of the driver's event callback functions access the space.

Before a driver can use a framework spin lock it must call WdfSpinLockCreate to create a spin-lock object. The driver can then call WdfSpinLockAcquire to acquire the lock and WdfSpinLockRelease to release it.

For an example use of spin locks, see Synchronizing Cancellation of Sent Requests.

Framework Interrupt Locks

For interrupt objects that support DIRQL interrupt handling, framework interrupt locks are spin locks. After your driver acquires an interrupt spin lock, the driver executes at the device's DIRQL until it releases the lock. For more information about using interrupt locks, see Synchronizing Interrupt Code.

For interrupt objects that support passive-level handling, framework interrupt locks are wait locks. After your driver acquires an interrupt wait lock, the driver executes at IRQL = PASSIVE_LEVEL until it releases the lock. For more information about passive-level handling, see Supporting Passive Level Interrupts.