[NETAPI32]
[reactos.git] / reactos / dll / win32 / netapi32 / nbcmdqueue.c
1 /* Copyright (c) 2003 Juan Lang
2 *
3 * This library is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU Lesser General Public
5 * License as published by the Free Software Foundation; either
6 * version 2.1 of the License, or (at your option) any later version.
7 *
8 * This library is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * Lesser General Public License for more details.
12 *
13 * You should have received a copy of the GNU Lesser General Public
14 * License along with this library; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
16 */
17 #include <config.h>
18 #include <wine/debug.h>
19 #include "nbcmdqueue.h"
20
21 WINE_DEFAULT_DEBUG_CHANNEL(netbios);
22
23 struct NBCmdQueue
24 {
25 HANDLE heap;
26 CRITICAL_SECTION cs;
27 PNCB head;
28 };
29
30 #define CANCEL_EVENT_PTR(ncb) (PHANDLE)((ncb)->ncb_reserve)
31 #define NEXT_PTR(ncb) (PNCB *)((ncb)->ncb_reserve + sizeof(HANDLE))
32
33 /* The reserved area of an ncb will be used for the following data:
34 * - a cancelled flag (BOOL, 4 bytes??)
35 * - a handle to an event that's set by a cancelled command on completion
36 * (HANDLE, 4 bytes)
37 * These members are used in the following way
38 * - on cancel, set the event member of the reserved field (with create event)
39 * - NBCmdComplete will delete the ncb from the queue of there's no event;
40 * otherwise it will set the event and not delete the ncb
41 * - cancel must lock the queue before finding the ncb in it, and can unlock it
42 * once it's set the event (and the cancelled flag)
43 * - NBCmdComplete must lock the queue before attempting to remove the ncb or
44 * check the event
45 * - NBCmdQueueCancelAll will lock the queue, and cancel all ncb's in the queue.
46 * It'll then unlock the queue, and wait on the event in the head of the queue
47 * until there's no more ncb's in the queue.
48 * Space optimization: use the handle as a boolean. NULL == 0 => not cancelled.
49 * Non-NULL == valid handle => cancelled. This allows storing a next pointer
50 * in the ncb's reserved field as well, avoiding a memory alloc for a new
51 * command (cool).
52 */
53
54 struct NBCmdQueue *NBCmdQueueCreate(HANDLE heap)
55 {
56 struct NBCmdQueue *queue;
57
58 if (heap == NULL)
59 heap = GetProcessHeap();
60 queue = HeapAlloc(heap, 0, sizeof(struct NBCmdQueue));
61 if (queue)
62 {
63 queue->heap = heap;
64 InitializeCriticalSection(&queue->cs);
65 queue->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": NBCmdQueue.cs");
66 queue->head = NULL;
67 }
68 return queue;
69 }
70
71 UCHAR NBCmdQueueAdd(struct NBCmdQueue *queue, PNCB ncb)
72 {
73 UCHAR ret;
74
75 TRACE(": queue %p, ncb %p\n", queue, ncb);
76
77 if (!queue)
78 return NRC_BADDR;
79 if (!ncb)
80 return NRC_INVADDRESS;
81
82 *CANCEL_EVENT_PTR(ncb) = NULL;
83 EnterCriticalSection(&queue->cs);
84 *NEXT_PTR(ncb) = queue->head;
85 queue->head = ncb;
86 ret = NRC_GOODRET;
87 LeaveCriticalSection(&queue->cs);
88 TRACE("returning 0x%02x\n", ret);
89 return ret;
90 }
91
92 static PNCB *NBCmdQueueFindNBC(struct NBCmdQueue *queue, PNCB ncb)
93 {
94 PNCB *ret;
95
96 if (!queue || !ncb)
97 ret = NULL;
98 else
99 {
100 ret = &queue->head;
101 while (ret && *ret != ncb)
102 ret = NEXT_PTR(*ret);
103 }
104 return ret;
105 }
106
107 UCHAR NBCmdQueueCancel(struct NBCmdQueue *queue, PNCB ncb)
108 {
109 UCHAR ret;
110 PNCB *spot;
111
112 TRACE(": queue %p, ncb %p\n", queue, ncb);
113
114 if (!queue)
115 return NRC_BADDR;
116 if (!ncb)
117 return NRC_INVADDRESS;
118
119 EnterCriticalSection(&queue->cs);
120 spot = NBCmdQueueFindNBC(queue, ncb);
121 if (spot)
122 {
123 *CANCEL_EVENT_PTR(*spot) = CreateEventW(NULL, FALSE, FALSE, NULL);
124 WaitForSingleObject(*CANCEL_EVENT_PTR(*spot), INFINITE);
125 CloseHandle(*CANCEL_EVENT_PTR(*spot));
126 *spot = *NEXT_PTR(*spot);
127 if (ncb->ncb_retcode == NRC_CMDCAN)
128 ret = NRC_CMDCAN;
129 else
130 ret = NRC_CANOCCR;
131 }
132 else
133 ret = NRC_INVADDRESS;
134 LeaveCriticalSection(&queue->cs);
135 TRACE("returning 0x%02x\n", ret);
136 return ret;
137 }
138
139 UCHAR NBCmdQueueComplete(struct NBCmdQueue *queue, PNCB ncb, UCHAR retcode)
140 {
141 UCHAR ret;
142 PNCB *spot;
143
144 TRACE(": queue %p, ncb %p\n", queue, ncb);
145
146 if (!queue)
147 return NRC_BADDR;
148 if (!ncb)
149 return NRC_INVADDRESS;
150
151 EnterCriticalSection(&queue->cs);
152 spot = NBCmdQueueFindNBC(queue, ncb);
153 if (spot)
154 {
155 if (*CANCEL_EVENT_PTR(*spot))
156 SetEvent(*CANCEL_EVENT_PTR(*spot));
157 else
158 *spot = *NEXT_PTR(*spot);
159 ret = NRC_GOODRET;
160 }
161 else
162 ret = NRC_INVADDRESS;
163 LeaveCriticalSection(&queue->cs);
164 TRACE("returning 0x%02x\n", ret);
165 return ret;
166 }
167
168 UCHAR NBCmdQueueCancelAll(struct NBCmdQueue *queue)
169 {
170 UCHAR ret;
171
172 TRACE(": queue %p\n", queue);
173
174 if (!queue)
175 return NRC_BADDR;
176
177 EnterCriticalSection(&queue->cs);
178 while (queue->head)
179 {
180 TRACE(": waiting for ncb %p (command 0x%02x)\n", queue->head,
181 queue->head->ncb_command);
182 NBCmdQueueCancel(queue, queue->head);
183 }
184 LeaveCriticalSection(&queue->cs);
185 ret = NRC_GOODRET;
186 TRACE("returning 0x%02x\n", ret);
187 return ret;
188 }
189
190 void NBCmdQueueDestroy(struct NBCmdQueue *queue)
191 {
192 TRACE(": queue %p\n", queue);
193
194 if (queue)
195 {
196 NBCmdQueueCancelAll(queue);
197 queue->cs.DebugInfo->Spare[0] = 0;
198 DeleteCriticalSection(&queue->cs);
199 HeapFree(queue->heap, 0, queue);
200 }
201 }