]>
Commit | Line | Data |
---|---|---|
1c79356b A |
1 | /* |
2 | * Copyright (c) 2000 Apple Computer, Inc. All rights reserved. | |
3 | * | |
2d21ac55 | 4 | * @APPLE_OSREFERENCE_LICENSE_HEADER_START@ |
0a7de745 | 5 | * |
2d21ac55 A |
6 | * This file contains Original Code and/or Modifications of Original Code |
7 | * as defined in and that are subject to the Apple Public Source License | |
8 | * Version 2.0 (the 'License'). You may not use this file except in | |
9 | * compliance with the License. The rights granted to you under the License | |
10 | * may not be used to create, or enable the creation or redistribution of, | |
11 | * unlawful or unlicensed copies of an Apple operating system, or to | |
12 | * circumvent, violate, or enable the circumvention or violation of, any | |
13 | * terms of an Apple operating system software license agreement. | |
0a7de745 | 14 | * |
2d21ac55 A |
15 | * Please obtain a copy of the License at |
16 | * http://www.opensource.apple.com/apsl/ and read it before using this file. | |
0a7de745 | 17 | * |
2d21ac55 A |
18 | * The Original Code and all software distributed under the License are |
19 | * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER | |
8f6c56a5 A |
20 | * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, |
21 | * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, | |
2d21ac55 A |
22 | * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. |
23 | * Please see the License for the specific language governing rights and | |
24 | * limitations under the License. | |
0a7de745 | 25 | * |
2d21ac55 | 26 | * @APPLE_OSREFERENCE_LICENSE_HEADER_END@ |
1c79356b A |
27 | */ |
28 | ||
29 | /* | |
30 | * | |
31 | * Copyright (c) 2000 Apple Computer, Inc. All rights reserved. | |
32 | * | |
33 | * HISTORY | |
34 | * | |
35 | * 2001-01-17 gvdl Re-implement on IOCommandGate::commandSleep | |
36 | * 10/9/2000 CJS Created IOCommandPool class and implementation | |
37 | * | |
38 | */ | |
39 | ||
f427ee49 A |
40 | #define IOKIT_ENABLE_SHARED_PTR |
41 | ||
1c79356b | 42 | #include <IOKit/IOCommandPool.h> |
f427ee49 | 43 | #include <libkern/c++/OSSharedPtr.h> |
1c79356b A |
44 | |
45 | #define super OSObject | |
46 | OSDefineMetaClassAndStructors(IOCommandPool, OSObject); | |
47 | OSMetaClassDefineReservedUnused(IOCommandPool, 0); | |
48 | OSMetaClassDefineReservedUnused(IOCommandPool, 1); | |
49 | OSMetaClassDefineReservedUnused(IOCommandPool, 2); | |
50 | OSMetaClassDefineReservedUnused(IOCommandPool, 3); | |
51 | OSMetaClassDefineReservedUnused(IOCommandPool, 4); | |
52 | OSMetaClassDefineReservedUnused(IOCommandPool, 5); | |
53 | OSMetaClassDefineReservedUnused(IOCommandPool, 6); | |
54 | OSMetaClassDefineReservedUnused(IOCommandPool, 7); | |
55 | ||
56 | //-------------------------------------------------------------------------- | |
57 | // withWorkLoop - primary initializer and factory method | |
58 | //-------------------------------------------------------------------------- | |
59 | ||
f427ee49 | 60 | OSSharedPtr<IOCommandPool> |
0a7de745 | 61 | IOCommandPool:: |
1c79356b A |
62 | withWorkLoop(IOWorkLoop *inWorkLoop) |
63 | { | |
f427ee49 | 64 | OSSharedPtr<IOCommandPool> me = OSMakeShared<IOCommandPool>(); |
0a7de745 A |
65 | |
66 | if (me && !me->initWithWorkLoop(inWorkLoop)) { | |
f427ee49 | 67 | return nullptr; |
0a7de745 A |
68 | } |
69 | ||
70 | return me; | |
1c79356b A |
71 | } |
72 | ||
73 | ||
0a7de745 A |
74 | bool |
75 | IOCommandPool:: | |
1c79356b A |
76 | initWithWorkLoop(IOWorkLoop *inWorkLoop) |
77 | { | |
0a7de745 A |
78 | assert(inWorkLoop); |
79 | ||
80 | if (!super::init()) { | |
81 | return false; | |
82 | } | |
83 | ||
84 | queue_init(&fQueueHead); | |
85 | ||
86 | fSerializer = IOCommandGate::commandGate(this); | |
87 | assert(fSerializer); | |
88 | if (!fSerializer) { | |
89 | return false; | |
90 | } | |
91 | ||
f427ee49 | 92 | if (kIOReturnSuccess != inWorkLoop->addEventSource(fSerializer.get())) { |
0a7de745 A |
93 | return false; |
94 | } | |
95 | ||
96 | return true; | |
1c79356b A |
97 | } |
98 | ||
99 | //-------------------------------------------------------------------------- | |
100 | // commandPool & init - obsolete initializer and factory method | |
101 | //-------------------------------------------------------------------------- | |
102 | ||
f427ee49 | 103 | OSSharedPtr<IOCommandPool> |
0a7de745 | 104 | IOCommandPool:: |
1c79356b A |
105 | commandPool(IOService * inOwner, IOWorkLoop *inWorkLoop, UInt32 inSize) |
106 | { | |
f427ee49 | 107 | OSSharedPtr<IOCommandPool> me = OSMakeShared<IOCommandPool>(); |
0a7de745 A |
108 | |
109 | if (me && !me->init(inOwner, inWorkLoop, inSize)) { | |
f427ee49 | 110 | return nullptr; |
0a7de745 A |
111 | } |
112 | ||
113 | return me; | |
1c79356b A |
114 | } |
115 | ||
0a7de745 A |
116 | bool |
117 | IOCommandPool:: | |
1c79356b A |
118 | init(IOService */* inOwner */, IOWorkLoop *inWorkLoop, UInt32 /* inSize */) |
119 | { | |
0a7de745 | 120 | return initWithWorkLoop(inWorkLoop); |
1c79356b A |
121 | } |
122 | ||
123 | ||
124 | //-------------------------------------------------------------------------- | |
125 | // free - free all allocated resources | |
126 | //-------------------------------------------------------------------------- | |
127 | ||
128 | void | |
129 | IOCommandPool::free(void) | |
130 | { | |
0a7de745 A |
131 | if (fSerializer) { |
132 | // remove our event source from owner's workloop | |
133 | IOWorkLoop *wl = fSerializer->getWorkLoop(); | |
134 | if (wl) { | |
f427ee49 | 135 | wl->removeEventSource(fSerializer.get()); |
0a7de745 A |
136 | } |
137 | ||
f427ee49 | 138 | fSerializer.reset(); |
0a7de745 A |
139 | } |
140 | ||
141 | // Tell our superclass to cleanup too | |
142 | super::free(); | |
1c79356b A |
143 | } |
144 | ||
145 | ||
146 | //-------------------------------------------------------------------------- | |
147 | // getCommand - Gets a command from the pool. Pass true in | |
148 | // blockForCommand if you want your thread to sleep | |
149 | // waiting for resources | |
150 | //-------------------------------------------------------------------------- | |
151 | ||
f427ee49 | 152 | OSSharedPtr<IOCommand> |
1c79356b A |
153 | IOCommandPool::getCommand(bool blockForCommand) |
154 | { | |
0a7de745 | 155 | IOReturn result = kIOReturnSuccess; |
cb323159 | 156 | IOCommand *command = NULL; |
0a7de745 A |
157 | |
158 | IOCommandGate::Action func = OSMemberFunctionCast( | |
159 | IOCommandGate::Action, this, &IOCommandPool::gatedGetCommand); | |
160 | result = fSerializer-> | |
161 | runAction(func, (void *) &command, (void *) blockForCommand); | |
162 | if (kIOReturnSuccess == result) { | |
f427ee49 | 163 | return OSSharedPtr<IOCommand>(command, OSNoRetain); |
0a7de745 | 164 | } else { |
cb323159 | 165 | return NULL; |
0a7de745 | 166 | } |
1c79356b A |
167 | } |
168 | ||
169 | ||
170 | //-------------------------------------------------------------------------- | |
171 | // gatedGetCommand - Static callthrough function | |
172 | // (on safe side of command gate) | |
173 | //-------------------------------------------------------------------------- | |
174 | ||
0a7de745 A |
175 | IOReturn |
176 | IOCommandPool:: | |
1c79356b A |
177 | gatedGetCommand(IOCommand **command, bool blockForCommand) |
178 | { | |
0a7de745 A |
179 | while (queue_empty(&fQueueHead)) { |
180 | if (!blockForCommand) { | |
181 | return kIOReturnNoResources; | |
182 | } | |
183 | ||
184 | fSleepers++; | |
185 | fSerializer->commandSleep(&fSleepers, THREAD_UNINT); | |
186 | } | |
187 | ||
188 | queue_remove_first(&fQueueHead, | |
189 | *command, IOCommand *, fCommandChain); | |
190 | return kIOReturnSuccess; | |
1c79356b A |
191 | } |
192 | ||
193 | ||
194 | //-------------------------------------------------------------------------- | |
195 | // returnCommand - Returns command to the pool. | |
196 | //-------------------------------------------------------------------------- | |
197 | ||
0a7de745 A |
198 | void |
199 | IOCommandPool:: | |
1c79356b A |
200 | returnCommand(IOCommand *command) |
201 | { | |
0a7de745 A |
202 | IOCommandGate::Action func = OSMemberFunctionCast( |
203 | IOCommandGate::Action, this, &IOCommandPool::gatedReturnCommand); | |
204 | (void) fSerializer->runAction(func, (void *) command); | |
1c79356b A |
205 | } |
206 | ||
207 | ||
208 | //-------------------------------------------------------------------------- | |
209 | // gatedReturnCommand - Callthrough function | |
0a7de745 | 210 | // (on safe side of command gate) |
1c79356b A |
211 | //-------------------------------------------------------------------------- |
212 | ||
0a7de745 A |
213 | IOReturn |
214 | IOCommandPool:: | |
1c79356b A |
215 | gatedReturnCommand(IOCommand *command) |
216 | { | |
0a7de745 A |
217 | queue_enter_first(&fQueueHead, command, IOCommand *, fCommandChain); |
218 | if (fSleepers) { | |
219 | fSerializer->commandWakeup(&fSleepers, /* oneThread */ true); | |
220 | fSleepers--; | |
221 | } | |
222 | return kIOReturnSuccess; | |
1c79356b | 223 | } |