3 * Copyright (C) 1998, 1999, 2000, 2001, 2002 ReactOS Team
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19 /* $Id: queue.c,v 1.10 2003/11/02 01:15:15 ekohl Exp $
21 * PROJECT: ReactOS kernel
22 * FILE: ntoskrnl/ke/queue.c
23 * PURPOSE: Implements kernel queues
24 * PROGRAMMER: Eric Kohl (ekohl@rz-online.de)
29 /* INCLUDES *****************************************************************/
31 #include <ddk/ntddk.h>
33 #include <internal/ke.h>
34 #include <internal/id.h>
35 #include <internal/ps.h>
38 #include <internal/debug.h>
40 /* FUNCTIONS *****************************************************************/
46 KeInitializeQueue(IN PKQUEUE Queue
,
47 IN ULONG Count OPTIONAL
)
49 KeInitializeDispatcherHeader(&Queue
->Header
,
51 sizeof(KQUEUE
)/sizeof(ULONG
),
53 InitializeListHead(&Queue
->EntryListHead
);
54 InitializeListHead(&Queue
->ThreadListHead
);
55 Queue
->CurrentCount
= 0;
56 Queue
->MaximumCount
= (Count
== 0) ? (ULONG
) KeNumberProcessors
: Count
;
64 KeReadStateQueue(IN PKQUEUE Queue
)
66 return(Queue
->Header
.SignalState
);
80 DPRINT("KiInsertQueue(Queue %x, Entry %x)\n", Queue
, Entry
);
82 OldIrql
= KeAcquireDispatcherDatabaseLock ();
84 InitialState
= Queue
->Header
.SignalState
;
85 Queue
->Header
.SignalState
++;
89 InsertHeadList(&Queue
->EntryListHead
, Entry
);
93 InsertTailList(&Queue
->EntryListHead
, Entry
);
96 if (Queue
->CurrentCount
< Queue
->MaximumCount
&& InitialState
== 0)
98 KeDispatcherObjectWake(&Queue
->Header
);
101 KeReleaseDispatcherDatabaseLock(OldIrql
);
111 KeInsertHeadQueue(IN PKQUEUE Queue
,
112 IN PLIST_ENTRY Entry
)
114 return KiInsertQueue(Queue
,Entry
,TRUE
);
122 KeInsertQueue(IN PKQUEUE Queue
,
123 IN PLIST_ENTRY Entry
)
125 return KiInsertQueue(Queue
,Entry
,FALSE
);
133 KeRemoveQueue(IN PKQUEUE Queue
,
134 IN KPROCESSOR_MODE WaitMode
,
135 IN PLARGE_INTEGER Timeout OPTIONAL
)
137 PLIST_ENTRY ListEntry
;
139 PKTHREAD Thread
= KeGetCurrentThread();
142 OldIrql
= KeAcquireDispatcherDatabaseLock ();
144 //assiciate new thread with queue?
145 if (Thread
->Queue
!= Queue
)
147 //remove association from other queue
148 if (!IsListEmpty(&Thread
->QueueListEntry
))
150 RemoveEntryList(&Thread
->QueueListEntry
);
153 //associate with this queue
154 InsertHeadList(&Queue
->ThreadListHead
, &Thread
->QueueListEntry
);
155 Queue
->CurrentCount
++;
156 Thread
->Queue
= Queue
;
159 if (Queue
->CurrentCount
<= Queue
->MaximumCount
&& !IsListEmpty(&Queue
->EntryListHead
))
161 ListEntry
= RemoveHeadList(&Queue
->EntryListHead
);
162 Queue
->Header
.SignalState
--;
163 KeReleaseDispatcherDatabaseLock (OldIrql
);
167 //need to wait for it...
168 KeReleaseDispatcherDatabaseLock (OldIrql
);
170 Status
= KeWaitForSingleObject(Queue
,
176 if (Status
== STATUS_TIMEOUT
|| Status
== STATUS_USER_APC
)
178 return (PVOID
)Status
;
182 OldIrql
= KeAcquireDispatcherDatabaseLock ();
183 ListEntry
= RemoveHeadList(&Queue
->EntryListHead
);
184 KeReleaseDispatcherDatabaseLock (OldIrql
);
195 KeRundownQueue(IN PKQUEUE Queue
)
197 PLIST_ENTRY EnumEntry
;
201 DPRINT("KeRundownQueue(Queue %x)\n", Queue
);
203 //FIXME: should we wake thread waiting on a queue?
205 OldIrql
= KeAcquireDispatcherDatabaseLock ();
207 // Clear Queue and QueueListEntry members of all threads associated with this queue
208 while (!IsListEmpty(&Queue
->ThreadListHead
))
210 EnumEntry
= RemoveHeadList(&Queue
->ThreadListHead
);
211 InitializeListHead(EnumEntry
);
212 Thread
= CONTAINING_RECORD(EnumEntry
, KTHREAD
, QueueListEntry
);
213 Thread
->Queue
= NULL
;
216 if (!IsListEmpty(&Queue
->EntryListHead
))
217 EnumEntry
= Queue
->EntryListHead
.Flink
;
221 KeReleaseDispatcherDatabaseLock (OldIrql
);