]>
Commit | Line | Data |
---|---|---|
1 | /* | |
2 | * Copyright (c) 1998-2000 Apple Computer, Inc. All rights reserved. | |
3 | * | |
4 | * @APPLE_LICENSE_HEADER_START@ | |
5 | * | |
6 | * The contents of this file constitute Original Code as defined in and | |
7 | * are subject to the Apple Public Source License Version 1.1 (the | |
8 | * "License"). You may not use this file except in compliance with the | |
9 | * License. Please obtain a copy of the License at | |
10 | * http://www.apple.com/publicsource and read it before using this file. | |
11 | * | |
12 | * This Original Code and all software distributed under the License are | |
13 | * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER | |
14 | * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, | |
15 | * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, | |
16 | * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the | |
17 | * License for the specific language governing rights and limitations | |
18 | * under the License. | |
19 | * | |
20 | * @APPLE_LICENSE_HEADER_END@ | |
21 | */ | |
22 | /* IOMemoryCursor.cpp created by wgulland on 1999-3-02 */ | |
23 | ||
24 | #include <IOKit/assert.h> | |
25 | #include <IOKit/IOLib.h> | |
26 | #include <IOKit/IOMemoryCursor.h> | |
27 | #include <IOKit/IOMemoryDescriptor.h> | |
28 | #include <libkern/OSByteOrder.h> | |
29 | ||
30 | /**************************** class IOMemoryCursor ***************************/ | |
31 | ||
32 | #undef super | |
33 | #define super OSObject | |
34 | OSDefineMetaClassAndStructors(IOMemoryCursor, OSObject) | |
35 | ||
36 | IOMemoryCursor * | |
37 | IOMemoryCursor::withSpecification(SegmentFunction inSegFunc, | |
38 | IOPhysicalLength inMaxSegmentSize, | |
39 | IOPhysicalLength inMaxTransferSize, | |
40 | IOPhysicalLength inAlignment) | |
41 | { | |
42 | IOMemoryCursor * me = new IOMemoryCursor; | |
43 | ||
44 | if (me && !me->initWithSpecification(inSegFunc, | |
45 | inMaxSegmentSize, | |
46 | inMaxTransferSize, | |
47 | inAlignment)) | |
48 | { | |
49 | me->release(); | |
50 | return 0; | |
51 | } | |
52 | ||
53 | return me; | |
54 | } | |
55 | ||
56 | /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ | |
57 | ||
58 | bool | |
59 | IOMemoryCursor::initWithSpecification(SegmentFunction inSegFunc, | |
60 | IOPhysicalLength inMaxSegmentSize, | |
61 | IOPhysicalLength inMaxTransferSize, | |
62 | IOPhysicalLength inAlignment) | |
63 | { | |
64 | // @@@ gvdl: Remove me | |
65 | #if 1 | |
66 | static UInt sMaxDBDMASegment; | |
67 | if (!sMaxDBDMASegment) { | |
68 | sMaxDBDMASegment = (UInt) -1; | |
69 | if (PE_parse_boot_arg("mseg", &sMaxDBDMASegment)) | |
70 | IOLog("Setting MaxDBDMASegment to %d\n", sMaxDBDMASegment); | |
71 | } | |
72 | ||
73 | if (inMaxSegmentSize > sMaxDBDMASegment) inMaxSegmentSize = sMaxDBDMASegment; | |
74 | #endif | |
75 | ||
76 | if (!super::init()) | |
77 | return false; | |
78 | ||
79 | if (!inSegFunc) | |
80 | return false; | |
81 | ||
82 | outSeg = inSegFunc; | |
83 | maxSegmentSize = inMaxSegmentSize; | |
84 | if (inMaxTransferSize) | |
85 | maxTransferSize = inMaxTransferSize; | |
86 | else | |
87 | maxTransferSize = (IOPhysicalLength) -1; | |
88 | alignMask = inAlignment - 1; | |
89 | assert(alignMask == 0); // No alignment code yet! | |
90 | ||
91 | return true; | |
92 | } | |
93 | ||
94 | /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ | |
95 | ||
96 | UInt32 | |
97 | IOMemoryCursor::genPhysicalSegments(IOMemoryDescriptor *inDescriptor, | |
98 | IOPhysicalLength fromPosition, | |
99 | void * inSegments, | |
100 | UInt32 inMaxSegments, | |
101 | UInt32 inMaxTransferSize, | |
102 | IOByteCount *outTransferSize) | |
103 | { | |
104 | if (!inDescriptor) | |
105 | return 0; | |
106 | ||
107 | if (!inMaxSegments) | |
108 | return 0; | |
109 | ||
110 | if (!inMaxTransferSize) | |
111 | inMaxTransferSize = maxTransferSize; | |
112 | ||
113 | /* | |
114 | * Iterate over the packet, translating segments where allowed | |
115 | * | |
116 | * If we finished cleanly return number of segments found | |
117 | * and update the position in the descriptor. | |
118 | */ | |
119 | PhysicalSegment curSeg = { 0, 0 }; | |
120 | UInt curSegIndex = 0; | |
121 | UInt curTransferSize = 0; | |
122 | IOByteCount inDescriptorLength = inDescriptor->getLength(); | |
123 | PhysicalSegment seg = { 0, 0 }; | |
124 | ||
125 | while ((seg.location) || (fromPosition < inDescriptorLength)) | |
126 | { | |
127 | if (!seg.location) | |
128 | { | |
129 | seg.location = inDescriptor->getPhysicalSegment( | |
130 | fromPosition, &seg.length); | |
131 | assert(seg.location); | |
132 | assert(seg.length); | |
133 | fromPosition += seg.length; | |
134 | } | |
135 | ||
136 | if (!curSeg.location) | |
137 | { | |
138 | curTransferSize += seg.length; | |
139 | curSeg = seg; | |
140 | seg.location = 0; | |
141 | } | |
142 | else if ((curSeg.location + curSeg.length == seg.location)) | |
143 | { | |
144 | curTransferSize += seg.length; | |
145 | curSeg.length += seg.length; | |
146 | seg.location = 0; | |
147 | } | |
148 | ||
149 | if (!seg.location) | |
150 | { | |
151 | if ((curSeg.length > maxSegmentSize)) | |
152 | { | |
153 | seg.location = curSeg.location + maxSegmentSize; | |
154 | seg.length = curSeg.length - maxSegmentSize; | |
155 | curTransferSize -= seg.length; | |
156 | curSeg.length -= seg.length; | |
157 | } | |
158 | ||
159 | if ((curTransferSize >= inMaxTransferSize)) | |
160 | { | |
161 | curSeg.length -= curTransferSize - inMaxTransferSize; | |
162 | curTransferSize = inMaxTransferSize; | |
163 | break; | |
164 | } | |
165 | } | |
166 | ||
167 | if (seg.location) | |
168 | { | |
169 | if ((curSegIndex + 1 == inMaxSegments)) | |
170 | break; | |
171 | (*outSeg)(curSeg, inSegments, curSegIndex++); | |
172 | curSeg.location = 0; | |
173 | } | |
174 | } | |
175 | ||
176 | if (curSeg.location) | |
177 | (*outSeg)(curSeg, inSegments, curSegIndex++); | |
178 | ||
179 | if (outTransferSize) | |
180 | *outTransferSize = curTransferSize; | |
181 | ||
182 | return curSegIndex; | |
183 | } | |
184 | ||
185 | /************************ class IONaturalMemoryCursor ************************/ | |
186 | ||
187 | #undef super | |
188 | #define super IOMemoryCursor | |
189 | OSDefineMetaClassAndStructors(IONaturalMemoryCursor, IOMemoryCursor) | |
190 | ||
191 | void IONaturalMemoryCursor::outputSegment(PhysicalSegment segment, | |
192 | void * outSegments, | |
193 | UInt32 outSegmentIndex) | |
194 | { | |
195 | ((PhysicalSegment *) outSegments)[outSegmentIndex] = segment; | |
196 | } | |
197 | ||
198 | IONaturalMemoryCursor * | |
199 | IONaturalMemoryCursor::withSpecification(IOPhysicalLength inMaxSegmentSize, | |
200 | IOPhysicalLength inMaxTransferSize, | |
201 | IOPhysicalLength inAlignment) | |
202 | { | |
203 | IONaturalMemoryCursor *me = new IONaturalMemoryCursor; | |
204 | ||
205 | if (me && !me->initWithSpecification(inMaxSegmentSize, | |
206 | inMaxTransferSize, | |
207 | inAlignment)) | |
208 | { | |
209 | me->release(); | |
210 | return 0; | |
211 | } | |
212 | ||
213 | return me; | |
214 | } | |
215 | ||
216 | bool | |
217 | IONaturalMemoryCursor::initWithSpecification(IOPhysicalLength inMaxSegmentSize, | |
218 | IOPhysicalLength inMaxTransferSize, | |
219 | IOPhysicalLength inAlignment) | |
220 | { | |
221 | return super::initWithSpecification(&IONaturalMemoryCursor::outputSegment, | |
222 | inMaxSegmentSize, | |
223 | inMaxTransferSize, | |
224 | inAlignment); | |
225 | } | |
226 | ||
227 | /************************** class IOBigMemoryCursor **************************/ | |
228 | ||
229 | #undef super | |
230 | #define super IOMemoryCursor | |
231 | OSDefineMetaClassAndStructors(IOBigMemoryCursor, IOMemoryCursor) | |
232 | ||
233 | void | |
234 | IOBigMemoryCursor::outputSegment(PhysicalSegment inSegment, | |
235 | void * inSegments, | |
236 | UInt32 inSegmentIndex) | |
237 | { | |
238 | IOPhysicalAddress * segment; | |
239 | ||
240 | segment = &((PhysicalSegment *) inSegments)[inSegmentIndex].location; | |
241 | OSWriteBigInt(segment, 0, inSegment.location); | |
242 | OSWriteBigInt(segment, sizeof(IOPhysicalAddress), inSegment.length); | |
243 | } | |
244 | ||
245 | IOBigMemoryCursor * | |
246 | IOBigMemoryCursor::withSpecification(IOPhysicalLength inMaxSegmentSize, | |
247 | IOPhysicalLength inMaxTransferSize, | |
248 | IOPhysicalLength inAlignment) | |
249 | { | |
250 | IOBigMemoryCursor * me = new IOBigMemoryCursor; | |
251 | ||
252 | if (me && !me->initWithSpecification(inMaxSegmentSize, | |
253 | inMaxTransferSize, | |
254 | inAlignment)) | |
255 | { | |
256 | me->release(); | |
257 | return 0; | |
258 | } | |
259 | ||
260 | return me; | |
261 | } | |
262 | ||
263 | bool | |
264 | IOBigMemoryCursor::initWithSpecification(IOPhysicalLength inMaxSegmentSize, | |
265 | IOPhysicalLength inMaxTransferSize, | |
266 | IOPhysicalLength inAlignment) | |
267 | { | |
268 | return super::initWithSpecification(&IOBigMemoryCursor::outputSegment, | |
269 | inMaxSegmentSize, | |
270 | inMaxTransferSize, | |
271 | inAlignment); | |
272 | } | |
273 | ||
274 | /************************* class IOLittleMemoryCursor ************************/ | |
275 | ||
276 | #undef super | |
277 | #define super IOMemoryCursor | |
278 | OSDefineMetaClassAndStructors(IOLittleMemoryCursor, IOMemoryCursor) | |
279 | ||
280 | void | |
281 | IOLittleMemoryCursor::outputSegment(PhysicalSegment inSegment, | |
282 | void * inSegments, | |
283 | UInt32 inSegmentIndex) | |
284 | { | |
285 | IOPhysicalAddress * segment; | |
286 | ||
287 | segment = &((PhysicalSegment *) inSegments)[inSegmentIndex].location; | |
288 | OSWriteLittleInt(segment, 0, inSegment.location); | |
289 | OSWriteLittleInt(segment, sizeof(IOPhysicalAddress), inSegment.length); | |
290 | } | |
291 | ||
292 | IOLittleMemoryCursor * | |
293 | IOLittleMemoryCursor::withSpecification(IOPhysicalLength inMaxSegmentSize, | |
294 | IOPhysicalLength inMaxTransferSize, | |
295 | IOPhysicalLength inAlignment) | |
296 | { | |
297 | IOLittleMemoryCursor * me = new IOLittleMemoryCursor; | |
298 | ||
299 | if (me && !me->initWithSpecification(inMaxSegmentSize, | |
300 | inMaxTransferSize, | |
301 | inAlignment)) | |
302 | { | |
303 | me->release(); | |
304 | return 0; | |
305 | } | |
306 | ||
307 | return me; | |
308 | } | |
309 | ||
310 | /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ | |
311 | ||
312 | bool | |
313 | IOLittleMemoryCursor::initWithSpecification(IOPhysicalLength inMaxSegmentSize, | |
314 | IOPhysicalLength inMaxTransferSize, | |
315 | IOPhysicalLength inAlignment) | |
316 | { | |
317 | return super::initWithSpecification(&IOLittleMemoryCursor::outputSegment, | |
318 | inMaxSegmentSize, | |
319 | inMaxTransferSize, | |
320 | inAlignment); | |
321 | } | |
322 | ||
323 | /************************* class IODBDMAMemoryCursor *************************/ | |
324 | ||
325 | #if defined(__ppc__) | |
326 | ||
327 | #include <IOKit/ppc/IODBDMA.h> | |
328 | ||
329 | #undef super | |
330 | #define super IOMemoryCursor | |
331 | OSDefineMetaClassAndStructors(IODBDMAMemoryCursor, IOMemoryCursor) | |
332 | ||
333 | void | |
334 | IODBDMAMemoryCursor::outputSegment(PhysicalSegment inSegment, | |
335 | void * inSegments, | |
336 | UInt32 inSegmentIndex) | |
337 | { | |
338 | IODBDMADescriptor *segment; | |
339 | ||
340 | segment = &((IODBDMADescriptor *) inSegments)[inSegmentIndex]; | |
341 | ||
342 | // Write location into address field | |
343 | OSWriteSwapInt32((UInt32 *) segment, 4, inSegment.location); | |
344 | ||
345 | // Write count into 1st two bytes of operation field. | |
346 | // DO NOT touch rest of operation field as it should contain a STOP command. | |
347 | OSWriteSwapInt16((UInt16 *) segment, 0, inSegment.length); | |
348 | } | |
349 | ||
350 | /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ | |
351 | ||
352 | IODBDMAMemoryCursor * | |
353 | IODBDMAMemoryCursor::withSpecification(IOPhysicalLength inMaxSegmentSize, | |
354 | IOPhysicalLength inMaxTransferSize, | |
355 | IOPhysicalLength inAlignment) | |
356 | { | |
357 | IODBDMAMemoryCursor *me = new IODBDMAMemoryCursor; | |
358 | ||
359 | if (me && !me->initWithSpecification(inMaxSegmentSize, | |
360 | inMaxTransferSize, | |
361 | inAlignment)) | |
362 | { | |
363 | me->release(); | |
364 | return 0; | |
365 | } | |
366 | ||
367 | return me; | |
368 | } | |
369 | ||
370 | /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ | |
371 | ||
372 | bool | |
373 | IODBDMAMemoryCursor::initWithSpecification(IOPhysicalLength inMaxSegmentSize, | |
374 | IOPhysicalLength inMaxTransferSize, | |
375 | IOPhysicalLength inAlignment) | |
376 | { | |
377 | return super::initWithSpecification(&IODBDMAMemoryCursor::outputSegment, | |
378 | inMaxSegmentSize, | |
379 | inMaxTransferSize, | |
380 | inAlignment); | |
381 | } | |
382 | ||
383 | #endif /* defined(__ppc__) */ | |
384 |