PLIST_ENTRY
NdisInterlockedInsertHeadList(
IN PLIST_ENTRY ListHead,
IN PLIST_ENTRY ListEntry,
IN PNDIS_SPIN_LOCK SpinLock
);
NdisInterlockedInsertHeadList inserts an entry, usually a packet, at the head of a doubly linked list so that access to the list is synchronized in a multiprocessor-safe way.
NdisInterlockedInsertHeadList returns a pointer to the entry that was at the head of the queue before the given entry was inserted. If the queue was empty, it returns NULL.
Before calling NdisInterlockedInsertHeadList, a driver must initialize the variable at ListHead with NdisInitializeListHead and the variable at SpinLock with NdisAllocateSpinLock. The driver also must provide resident storage for these variables and for its internal queue.
The caller-supplied spin lock prevents any other function from accessing the driver’s internal queue while NdisInterlockedInsertHeadList is inserting the given entry, even when the driver is running on a multiprocessor machine.
NdisInterlockedInsertHeadList raises IRQL to DISPATCH_LEVEL when it acquires the given spin lock and restores the original IRQL before it returns control. Consequently, any driver function that calls NdisInterlockedInsertHeadList cannot be pageable code.
Most NDIS drivers process packets in FIFO order, so any driver that uses an interlocked queue tends to call NdisInterlockedInsertTailList far more frequently than NdisInterlockedInsertHeadList. Such a driver usually calls NdisInterlockedInsertHeadList only to requeue a packet for a retry operation.
To convert a returned value back to the address of the inserted entry, a driver can use the CONTAINING_RECORD macro (see the Kernel-Mode Driver Reference).
Callers of NdisInterlockedInsertHeadList run at IRQL <= DISPATCH_LEVEL.
NdisAllocateSpinLock, NdisInitializeListHead, NdisInterlockedInsertTailList, NdisInterlockedRemoveHeadList