/*++ Copyright (c) 2000 Microsoft Corporation Module Name: nbqueue.c Abstract: This module implements non-blocking fifo queue. Author: David N. Cutler (davec) 24-Apr-2000 Environment: Kernel mode only. Revision History: --*/ #include "windows.h" #include "malloc.h" VOID DbgBreakPoint ( VOID ); extern ULONG StopSignal; // // Define non-blocking interlocked queue functions. // // A non-blocking queue is a singly link list of queue entries with a // head pointer and a tail pointer. The head and tail pointers use // sequenced pointers as do next links in the entries themselves. The // queueing discipline is FIFO. New entries are inserted at the tail // of the list and current entries are removed from the front of the // list. // // Non-blocking queues require a descriptor for each entry in the queue. // A descriptor consists of a sequenced next pointer and a PVOID data // value. Descriptors for a queue must be preallocated and inserted in // an SLIST before calling the function to initialize a non-blocking // queue header. The SLIST should have as many entries as required for // the respective queue. // typedef struct _NBQUEUE_BLOCK { ULONG64 Next; ULONG64 Data; } NBQUEUE_BLOCK, *PNBQUEUE_BLOCK; PVOID ExInitializeNBQueueHead ( IN PSLIST_HEADER SlistHead ); BOOLEAN ExInsertTailNBQueue ( IN PVOID Header, IN ULONG64 Value ); BOOLEAN ExRemoveHeadNBQueue ( IN PVOID Header, OUT PULONG64 Value ); #if defined(_X86_) #define InterlockedCompareExchange64(Destination, Exchange, Comperand) \ xInterlockedCompareExchange64(Destination, &(Exchange), &(Comperand)) LONG64 __fastcall xInterlockedCompareExchange64 ( IN OUT LONG64 volatile * Destination, IN PLONG64 Exchange, IN PLONG64 Comparand ); #elif defined(_IA64_) #define InterlockedCompareExchange64 _InterlockedCompareExchange64 LONGLONG __cdecl InterlockedCompareExchange64 ( IN OUT LONGLONG volatile *Destination, IN LONGLONG ExChange, IN LONGLONG Comperand ); #pragma intrinsic(_InterlockedCompareExchange64) #endif // // Define queue pointer structure - this is platform target specific. // #if defined(_AMD64_) typedef union _NBQUEUE_POINTER { struct { LONG64 Node : 48; LONG64 Count : 16; }; LONG64 Data; } NBQUEUE_POINTER, *PNBQUEUE_POINTER; #elif defined(_X86_) typedef union _NBQUEUE_POINTER { struct { LONG Count; LONG Node; }; LONG64 Data; } NBQUEUE_POINTER, *PNBQUEUE_POINTER; #elif defined(_IA64_) typedef union _NBQUEUE_POINTER { struct { LONG64 Node : 45; LONG64 Region : 3; LONG64 Count : 16; }; LONG64 Data; } NBQUEUE_POINTER, *PNBQUEUE_POINTER; #else #error "no target architecture" #endif // // Define queue node struture. // typedef struct _NBQUEUE_NODE { NBQUEUE_POINTER Next; ULONG64 Value; } NBQUEUE_NODE, *PNBQUEUE_NODE; // // Define inline functions to pack and unpack pointers in the platform // specific non-blocking queue pointer structure. // #if defined(_AMD64_) __inline VOID PackNBQPointer ( IN PNBQUEUE_POINTER Entry, IN PNBQUEUE_NODE Node ) { Entry->Node = (LONG64)Node; return; } __inline PNBQUEUE_NODE UnpackNBQPointer ( IN PNBQUEUE_POINTER Entry ) { return (PVOID)((LONG64)(Entry->Node)); } #elif defined(_X86_) __inline VOID PackNBQPointer ( IN PNBQUEUE_POINTER Entry, IN PNBQUEUE_NODE Node ) { Entry->Node = (LONG)Node; return; } __inline PNBQUEUE_NODE UnpackNBQPointer ( IN PNBQUEUE_POINTER Entry ) { return (PVOID)(Entry->Node); } #elif defined(_IA64_) __inline VOID PackNBQPointer ( IN PNBQUEUE_POINTER Entry, IN PNBQUEUE_NODE Node ) { Entry->Node = (LONG64)Node; Entry->Region = (LONG64)Node >> 61; return; } __inline PNBQUEUE_NODE UnpackNBQPointer ( IN PNBQUEUE_POINTER Entry ) { LONG64 Value; Value = Entry->Node & 0x1fffffffffffffff; Value |= Entry->Region << 61; return (PVOID)(Value); } #else #error "no target architecture" #endif // // Define queue descriptor structure. // typedef struct _NBQUEUE_HEADER { NBQUEUE_POINTER Head; NBQUEUE_POINTER Tail; PSLIST_HEADER SlistHead; } NBQUEUE_HEADER, *PNBQUEUE_HEADER; typedef struct _NBQUEUE_LOG { ULONG_PTR Type; PNBQUEUE_HEADER Queue; NBQUEUE_POINTER Head; NBQUEUE_POINTER Tail; NBQUEUE_POINTER Next; ULONG_PTR Value; PNBQUEUE_NODE Node; PVOID *Address; ULONG_PTR Fill; } NBQUEUE_LOG, *PNBQUEUE_LOG; #define NBQUEUE_LOG_SIZE 64 NBQUEUE_LOG NbLog[NBQUEUE_LOG_SIZE + 1]; ULONG xLogIndex = -1; #define LogInsertData(_queue_, _head_, _tail_, _next_) { \ if (StopSignal != 0) { \ LogIndex = NBQUEUE_LOG_SIZE; \ } else { \ LogIndex = InterlockedIncrement(&xLogIndex) & (NBQUEUE_LOG_SIZE - 1); \ } \ NbLog[LogIndex].Type = 0; \ NbLog[LogIndex].Queue = _queue_; \ NbLog[LogIndex].Head.Data = (_head_); \ NbLog[LogIndex].Tail.Data = (_tail_); \ NbLog[LogIndex].Next.Data = (_next_); \ } #define LogRemoveData(_queue_, _head_, _tail_, _next_) { \ if (StopSignal != 0) { \ LogIndex = NBQUEUE_LOG_SIZE; \ } else { \ LogIndex = InterlockedIncrement(&xLogIndex) & (NBQUEUE_LOG_SIZE - 1); \ } \ NbLog[LogIndex].Type = 1; \ NbLog[LogIndex].Queue = _queue_; \ NbLog[LogIndex].Head.Data = (_head_); \ NbLog[LogIndex].Tail.Data = (_tail_); \ NbLog[LogIndex].Next.Data = (_next_); \ } #pragma alloc_text(PAGE, ExInitializeNBQueueHead) PVOID ExInitializeNBQueueHead ( IN PSLIST_HEADER SlistHead ) /*++ Routine Description: This function initializes a non-blocking queue header. N.B. It is assumed that the specified SLIST has been populated with non-blocking queue nodes prior to calling this routine. Arguments: SlistHead - Supplies a pointer to an SLIST header. Return Value: If the non-blocking queue is successfully initialized, then the address of the queue header is returned as the function value. Otherwise, NULL is returned as the function value. --*/ { PNBQUEUE_HEADER QueueHead; PNBQUEUE_NODE QueueNode; // // Attempt to allocate the queue header. If the allocation fails, then // return NULL. // QueueHead = (PNBQUEUE_HEADER)malloc(sizeof(NBQUEUE_HEADER)); if (QueueHead == NULL) { return NULL; } // // Attempt to allocate a queue node from the specified SLIST. If a node // can be allocated, then initialize the non-blocking queue header and // return the address of the queue header. Otherwise, free the queue // header and return NULL. // QueueHead->SlistHead = SlistHead; QueueNode = (PNBQUEUE_NODE)InterlockedPopEntrySList(QueueHead->SlistHead); if (QueueNode != NULL) { // // Initialize the queue node next pointer and value. // QueueNode->Next.Data = 0; QueueNode->Value = 0; // // Initialize the head and tail pointers in the queue header. // PackNBQPointer(&QueueHead->Head, QueueNode); QueueHead->Head.Count = 0; PackNBQPointer(&QueueHead->Tail, QueueNode); QueueHead->Tail.Count = 0; return QueueHead; } else { free(QueueHead); return NULL; } } BOOLEAN ExInsertTailNBQueue ( IN PVOID Header, IN ULONG64 Value ) /*++ Routine Description: This function inserts the specific data value at the tail of the specified non-blocking queue. Arguments: Header - Supplies an opaque pointer to a non-blocking queue header. Value - Supplies a pointer to an opaque data value. Return Value: If the specified opaque data value is successfully inserted at the tail of the specified non-blocking queue, then a value of TRUE is returned as the function value. Otherwise, a value of FALSE is returned. N.B. FALSE is returned if a queue node cannot be allocated from the associated SLIST. --*/ { NBQUEUE_POINTER Head; NBQUEUE_POINTER Insert; ULONG LogIndex; NBQUEUE_POINTER Next; PNBQUEUE_NODE NextNode; PNBQUEUE_HEADER QueueHead; PNBQUEUE_NODE QueueNode; NBQUEUE_POINTER Tail; PNBQUEUE_NODE TailNode; // // Attempt to allocate a queue node from the SLIST associated with // the specified non-blocking queue. If a node can be allocated, then // the node is inserted at the tail of the specified non-blocking // queue, and TRUE is returned as the function value. Otherwise, FALSE // is returned. // QueueHead = (PNBQUEUE_HEADER)Header; QueueNode = (PNBQUEUE_NODE)InterlockedPopEntrySList(QueueHead->SlistHead); if (QueueNode != NULL) { // // Initialize the queue node next pointer and value. // QueueNode->Next.Data = 0; QueueNode->Value = Value; // // The following loop is executed until the specified entry can // be safely inserted at the tail of the specified non-blocking // queue. // do { // // Read the tail queue pointer and the next queue pointer of // the tail queue pointer making sure the two pointers are // coherent. // Head.Data = *((volatile LONG64 *)(&QueueHead->Head.Data)); Tail.Data = *((volatile LONG64 *)(&QueueHead->Tail.Data)); TailNode = UnpackNBQPointer(&Tail); Next.Data = *((volatile LONG64 *)(&TailNode->Next.Data)); LogInsertData(QueueHead, Head.Data, Tail.Data, Next.Data); NbLog[LogIndex].Address = &Header; QueueNode->Next.Count = Tail.Count + 1; if (Tail.Data == *((volatile LONG64 *)(&QueueHead->Tail.Data))) { // // If the tail is pointing to the last node in the list, // then attempt to insert the new node at the end of the // list. Otherwise, the tail is not pointing to the last // node in the list and an attempt is made to move the // tail pointer to the next node. // NextNode = UnpackNBQPointer(&Next); if (NextNode == NULL) { PackNBQPointer(&Insert, QueueNode); Insert.Count = Next.Count + 1; if (InterlockedCompareExchange64(&TailNode->Next.Data, Insert.Data, Next.Data) == Next.Data) { NbLog[LogIndex].Value = (ULONG)Value; NbLog[LogIndex].Node = QueueNode; break; } else { NbLog[LogIndex].Value = 0xffffffff; NbLog[LogIndex].Node = QueueNode; } } else { PackNBQPointer(&Insert, NextNode); Insert.Count = Tail.Count + 1; if (InterlockedCompareExchange64(&QueueHead->Tail.Data, Insert.Data, Tail.Data) == Tail.Data) { NbLog[LogIndex].Value = 0xffffff00; NbLog[LogIndex].Node = QueueNode; } else { NbLog[LogIndex].Value = 0xffff0000; NbLog[LogIndex].Node = QueueNode; } } } else { NbLog[LogIndex].Value = 0x000000ff; NbLog[LogIndex].Node = QueueNode; } } while (TRUE); // // Attempt to move the tail to the new tail node. // LogInsertData(QueueHead, Head.Data, Tail.Data, Next.Data); NbLog[LogIndex].Address = &Header; PackNBQPointer(&Insert, QueueNode); Insert.Count = Tail.Count + 1; if (InterlockedCompareExchange64(&QueueHead->Tail.Data, Insert.Data, Tail.Data) == Tail.Data) { NbLog[LogIndex].Value = 0x0000ffff; NbLog[LogIndex].Node = QueueNode; } else { NbLog[LogIndex].Value = 0x00ffffff; NbLog[LogIndex].Node = QueueNode; } return TRUE; } else { return FALSE; } } BOOLEAN ExRemoveHeadNBQueue ( IN PVOID Header, OUT PULONG64 Value ) /*++ Routine Description: This function removes a queue entry from the head of the specified non-blocking queue and returns the associated data value. Arguments: Header - Supplies an opaque pointer to a non-blocking queue header. Value - Supplies a pointer to a variable that receives the queue element value. Return Value: If an entry is removed from the specified non-blocking queue, then TRUE is returned as the function value. Otherwise, FALSE is returned. --*/ { NBQUEUE_POINTER Head; PNBQUEUE_NODE HeadNode; NBQUEUE_POINTER Insert; ULONG LogIndex; NBQUEUE_POINTER Next; PNBQUEUE_NODE NextNode; PNBQUEUE_HEADER QueueHead; NBQUEUE_POINTER Tail; PNBQUEUE_NODE TailNode; // // The following loop is executed until an entry can be removed from // the specified non-blocking queue or until it can be determined that // the queue is empty. // QueueHead = (PNBQUEUE_HEADER)Header; do { // // Read the head queue pointer, the tail queue pointer, and the // next queue pointer of the head queue pointer making sure the // three pointers are coherent. // Head.Data = *((volatile LONG64 *)(&QueueHead->Head.Data)); Tail.Data = *((volatile LONG64 *)(&QueueHead->Tail.Data)); HeadNode = UnpackNBQPointer(&Head); Next.Data = *((volatile LONG64 *)(&HeadNode->Next.Data)); LogRemoveData(QueueHead, Head.Data, Tail.Data, Next.Data); NbLog[LogIndex].Address = &Header; if (Head.Data == *((volatile LONG64 *)(&QueueHead->Head.Data))) { // // If the queue header node is equal to the queue tail node, // then either the queue is empty or the tail pointer is falling // behind. Otherwise, there is an entry in the queue that can // be removed. // NextNode = UnpackNBQPointer(&Next); TailNode = UnpackNBQPointer(&Tail); if (HeadNode == TailNode) { // // If the next node of head pointer is NULL, then the queue // is empty. Otherwise, attempt to move the tail forward. // if (NextNode == NULL) { NbLog[LogIndex].Value = 0xffffffff; NbLog[LogIndex].Node = NULL; *Value = 0xffffffff; return FALSE; } else { PackNBQPointer(&Insert, NextNode); Insert.Count = Tail.Count + 1; if (InterlockedCompareExchange64(&QueueHead->Tail.Data, Insert.Data, Tail.Data) == Tail.Data) { NbLog[LogIndex].Value = 0xffffff00; NbLog[LogIndex].Node = NULL; } else { NbLog[LogIndex].Value = 0xffff0000; NbLog[LogIndex].Node = NULL; } } } else { // // Attempt to remove the first entry at the head of the queue. // *Value = ((ULONG64)LogIndex << 32) | NextNode->Value; PackNBQPointer(&Insert, NextNode); Insert.Count = Head.Count + 1; if (InterlockedCompareExchange64(&QueueHead->Head.Data, Insert.Data, Head.Data) == Head.Data) { NbLog[LogIndex].Value = (ULONG)*Value; NbLog[LogIndex].Node = NextNode; break; } else { NbLog[LogIndex].Value = 0x00ffffff; NbLog[LogIndex].Node = NextNode; } } } else { NbLog[LogIndex].Value = 0x0000ffff; NbLog[LogIndex].Node = NULL; } } while (TRUE); // // Free the node that was removed for the list by inserting the node // in the associated SLIST. // InterlockedPushEntrySList(QueueHead->SlistHead, (PSLIST_ENTRY)HeadNode); return TRUE; }