xref: /xnu-8020.101.4/iokit/Kernel/IORangeAllocator.cpp (revision e7776783b89a353188416a9a346c6cdb4928faad)
1*e7776783SApple OSS Distributions /*
2*e7776783SApple OSS Distributions  * Copyright (c) 1998-2000 Apple Computer, Inc. All rights reserved.
3*e7776783SApple OSS Distributions  *
4*e7776783SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*e7776783SApple OSS Distributions  *
6*e7776783SApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*e7776783SApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*e7776783SApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*e7776783SApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*e7776783SApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*e7776783SApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*e7776783SApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*e7776783SApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*e7776783SApple OSS Distributions  *
15*e7776783SApple OSS Distributions  * Please obtain a copy of the License at
16*e7776783SApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*e7776783SApple OSS Distributions  *
18*e7776783SApple OSS Distributions  * The Original Code and all software distributed under the License are
19*e7776783SApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*e7776783SApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*e7776783SApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*e7776783SApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*e7776783SApple OSS Distributions  * Please see the License for the specific language governing rights and
24*e7776783SApple OSS Distributions  * limitations under the License.
25*e7776783SApple OSS Distributions  *
26*e7776783SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*e7776783SApple OSS Distributions  */
28*e7776783SApple OSS Distributions /*
29*e7776783SApple OSS Distributions  * Copyright (c) 1999 Apple Computer, Inc.
30*e7776783SApple OSS Distributions  *
31*e7776783SApple OSS Distributions  *
32*e7776783SApple OSS Distributions  * HISTORY
33*e7776783SApple OSS Distributions  *
34*e7776783SApple OSS Distributions  * sdouglas 05 Nov 99 - created.
35*e7776783SApple OSS Distributions  */
36*e7776783SApple OSS Distributions 
37*e7776783SApple OSS Distributions #include <libkern/c++/OSArray.h>
38*e7776783SApple OSS Distributions #include <libkern/c++/OSNumber.h>
39*e7776783SApple OSS Distributions #include <IOKit/IORangeAllocator.h>
40*e7776783SApple OSS Distributions #include <IOKit/IOLib.h>
41*e7776783SApple OSS Distributions #include <IOKit/IOLocks.h>
42*e7776783SApple OSS Distributions #include <IOKit/assert.h>
43*e7776783SApple OSS Distributions 
44*e7776783SApple OSS Distributions /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
45*e7776783SApple OSS Distributions 
46*e7776783SApple OSS Distributions #undef super
47*e7776783SApple OSS Distributions #define super OSObject
48*e7776783SApple OSS Distributions 
49*e7776783SApple OSS Distributions OSDefineMetaClassAndStructors( IORangeAllocator, OSObject )
50*e7776783SApple OSS Distributions 
51*e7776783SApple OSS Distributions struct IORangeAllocatorElement {
52*e7776783SApple OSS Distributions 	// closed range
53*e7776783SApple OSS Distributions 	IORangeScalar       start;
54*e7776783SApple OSS Distributions 	IORangeScalar       end;
55*e7776783SApple OSS Distributions };
56*e7776783SApple OSS Distributions 
57*e7776783SApple OSS Distributions LCK_GRP_DECLARE(range_allocator_grp, "range_allocator_grp");
58*e7776783SApple OSS Distributions LCK_MTX_DECLARE(gIORangeAllocatorLock, &range_allocator_grp);
59*e7776783SApple OSS Distributions 
60*e7776783SApple OSS Distributions #define LOCK()          \
61*e7776783SApple OSS Distributions 	if( options & kLocking)	lck_mtx_lock( &gIORangeAllocatorLock )
62*e7776783SApple OSS Distributions #define UNLOCK()        \
63*e7776783SApple OSS Distributions 	if( options & kLocking)	lck_mtx_unlock( &gIORangeAllocatorLock )
64*e7776783SApple OSS Distributions 
65*e7776783SApple OSS Distributions /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
66*e7776783SApple OSS Distributions 
67*e7776783SApple OSS Distributions bool
init(IORangeScalar endOfRange,IORangeScalar _defaultAlignment,UInt32 _capacity,IOOptionBits _options)68*e7776783SApple OSS Distributions IORangeAllocator::init( IORangeScalar endOfRange,
69*e7776783SApple OSS Distributions     IORangeScalar _defaultAlignment,
70*e7776783SApple OSS Distributions     UInt32 _capacity,
71*e7776783SApple OSS Distributions     IOOptionBits _options )
72*e7776783SApple OSS Distributions {
73*e7776783SApple OSS Distributions 	if (!super::init()) {
74*e7776783SApple OSS Distributions 		return false;
75*e7776783SApple OSS Distributions 	}
76*e7776783SApple OSS Distributions 
77*e7776783SApple OSS Distributions 	if (!_capacity) {
78*e7776783SApple OSS Distributions 		_capacity = 1;
79*e7776783SApple OSS Distributions 	}
80*e7776783SApple OSS Distributions 	if (!_defaultAlignment) {
81*e7776783SApple OSS Distributions 		_defaultAlignment = 1;
82*e7776783SApple OSS Distributions 	}
83*e7776783SApple OSS Distributions 	capacity            = 0;
84*e7776783SApple OSS Distributions 	capacityIncrement   = _capacity;
85*e7776783SApple OSS Distributions 	numElements         = 0;
86*e7776783SApple OSS Distributions 	elements            = NULL;
87*e7776783SApple OSS Distributions 	defaultAlignmentMask = _defaultAlignment - 1;
88*e7776783SApple OSS Distributions 	options             = _options;
89*e7776783SApple OSS Distributions 
90*e7776783SApple OSS Distributions 	if (endOfRange) {
91*e7776783SApple OSS Distributions 		deallocate( 0, endOfRange + 1 );
92*e7776783SApple OSS Distributions 	}
93*e7776783SApple OSS Distributions 
94*e7776783SApple OSS Distributions 	return true;
95*e7776783SApple OSS Distributions }
96*e7776783SApple OSS Distributions 
97*e7776783SApple OSS Distributions IORangeAllocator *
withRange(IORangeScalar endOfRange,IORangeScalar defaultAlignment,UInt32 capacity,IOOptionBits options)98*e7776783SApple OSS Distributions IORangeAllocator::withRange(
99*e7776783SApple OSS Distributions 	IORangeScalar endOfRange,
100*e7776783SApple OSS Distributions 	IORangeScalar defaultAlignment,
101*e7776783SApple OSS Distributions 	UInt32 capacity,
102*e7776783SApple OSS Distributions 	IOOptionBits options )
103*e7776783SApple OSS Distributions {
104*e7776783SApple OSS Distributions 	IORangeAllocator * thingy;
105*e7776783SApple OSS Distributions 
106*e7776783SApple OSS Distributions 	thingy = new IORangeAllocator;
107*e7776783SApple OSS Distributions 	if (thingy && !thingy->init( endOfRange, defaultAlignment,
108*e7776783SApple OSS Distributions 	    capacity, options )) {
109*e7776783SApple OSS Distributions 		thingy->release();
110*e7776783SApple OSS Distributions 		thingy = NULL;
111*e7776783SApple OSS Distributions 	}
112*e7776783SApple OSS Distributions 
113*e7776783SApple OSS Distributions 	return thingy;
114*e7776783SApple OSS Distributions }
115*e7776783SApple OSS Distributions 
116*e7776783SApple OSS Distributions void
free()117*e7776783SApple OSS Distributions IORangeAllocator::free()
118*e7776783SApple OSS Distributions {
119*e7776783SApple OSS Distributions 	if (elements) {
120*e7776783SApple OSS Distributions 		IODeleteData( elements, IORangeAllocatorElement, capacity );
121*e7776783SApple OSS Distributions 	}
122*e7776783SApple OSS Distributions 
123*e7776783SApple OSS Distributions 	super::free();
124*e7776783SApple OSS Distributions }
125*e7776783SApple OSS Distributions 
126*e7776783SApple OSS Distributions UInt32
getFragmentCount(void)127*e7776783SApple OSS Distributions IORangeAllocator::getFragmentCount( void )
128*e7776783SApple OSS Distributions {
129*e7776783SApple OSS Distributions 	return numElements;
130*e7776783SApple OSS Distributions }
131*e7776783SApple OSS Distributions 
132*e7776783SApple OSS Distributions UInt32
getFragmentCapacity(void)133*e7776783SApple OSS Distributions IORangeAllocator::getFragmentCapacity( void )
134*e7776783SApple OSS Distributions {
135*e7776783SApple OSS Distributions 	return capacity;
136*e7776783SApple OSS Distributions }
137*e7776783SApple OSS Distributions 
138*e7776783SApple OSS Distributions void
setFragmentCapacityIncrement(UInt32 count)139*e7776783SApple OSS Distributions IORangeAllocator::setFragmentCapacityIncrement( UInt32 count )
140*e7776783SApple OSS Distributions {
141*e7776783SApple OSS Distributions 	capacityIncrement = count;
142*e7776783SApple OSS Distributions }
143*e7776783SApple OSS Distributions 
144*e7776783SApple OSS Distributions 
145*e7776783SApple OSS Distributions // allocate element at index
146*e7776783SApple OSS Distributions bool
allocElement(UInt32 index)147*e7776783SApple OSS Distributions IORangeAllocator::allocElement( UInt32 index )
148*e7776783SApple OSS Distributions {
149*e7776783SApple OSS Distributions 	UInt32                      newCapacity;
150*e7776783SApple OSS Distributions 	IORangeAllocatorElement *   newElements;
151*e7776783SApple OSS Distributions 
152*e7776783SApple OSS Distributions 	if (((numElements == capacity) && capacityIncrement)
153*e7776783SApple OSS Distributions 	    || (!elements)) {
154*e7776783SApple OSS Distributions 		if (os_add_overflow(capacity, capacityIncrement, &newCapacity)) {
155*e7776783SApple OSS Distributions 			return false;
156*e7776783SApple OSS Distributions 		}
157*e7776783SApple OSS Distributions 		newElements = IONewData( IORangeAllocatorElement, newCapacity );
158*e7776783SApple OSS Distributions 		if (!newElements) {
159*e7776783SApple OSS Distributions 			return false;
160*e7776783SApple OSS Distributions 		}
161*e7776783SApple OSS Distributions 
162*e7776783SApple OSS Distributions 		if (elements) {
163*e7776783SApple OSS Distributions 			bcopy( elements,
164*e7776783SApple OSS Distributions 			    newElements,
165*e7776783SApple OSS Distributions 			    index * sizeof(IORangeAllocatorElement));
166*e7776783SApple OSS Distributions 			bcopy( elements + index,
167*e7776783SApple OSS Distributions 			    newElements + index + 1,
168*e7776783SApple OSS Distributions 			    (numElements - index) * sizeof(IORangeAllocatorElement));
169*e7776783SApple OSS Distributions 
170*e7776783SApple OSS Distributions 			IODeleteData( elements, IORangeAllocatorElement, capacity );
171*e7776783SApple OSS Distributions 		}
172*e7776783SApple OSS Distributions 
173*e7776783SApple OSS Distributions 		elements = newElements;
174*e7776783SApple OSS Distributions 		capacity = newCapacity;
175*e7776783SApple OSS Distributions 	} else {
176*e7776783SApple OSS Distributions 		bcopy( elements + index,
177*e7776783SApple OSS Distributions 		    elements + index + 1,
178*e7776783SApple OSS Distributions 		    (numElements - index) * sizeof(IORangeAllocatorElement));
179*e7776783SApple OSS Distributions 	}
180*e7776783SApple OSS Distributions 	numElements++;
181*e7776783SApple OSS Distributions 
182*e7776783SApple OSS Distributions 	return true;
183*e7776783SApple OSS Distributions }
184*e7776783SApple OSS Distributions 
185*e7776783SApple OSS Distributions // destroy element at index
186*e7776783SApple OSS Distributions void
deallocElement(UInt32 index)187*e7776783SApple OSS Distributions IORangeAllocator::deallocElement( UInt32 index )
188*e7776783SApple OSS Distributions {
189*e7776783SApple OSS Distributions 	numElements--;
190*e7776783SApple OSS Distributions 	bcopy( elements + index + 1,
191*e7776783SApple OSS Distributions 	    elements + index,
192*e7776783SApple OSS Distributions 	    (numElements - index) * sizeof(IORangeAllocatorElement));
193*e7776783SApple OSS Distributions }
194*e7776783SApple OSS Distributions 
195*e7776783SApple OSS Distributions bool
allocate(IORangeScalar size,IORangeScalar * result,IORangeScalar alignment)196*e7776783SApple OSS Distributions IORangeAllocator::allocate( IORangeScalar size,
197*e7776783SApple OSS Distributions     IORangeScalar * result,
198*e7776783SApple OSS Distributions     IORangeScalar alignment )
199*e7776783SApple OSS Distributions {
200*e7776783SApple OSS Distributions 	IORangeScalar       data, dataEnd;
201*e7776783SApple OSS Distributions 	IORangeScalar       thisStart, thisEnd;
202*e7776783SApple OSS Distributions 	UInt32              index;
203*e7776783SApple OSS Distributions 	bool                ok = false;
204*e7776783SApple OSS Distributions 
205*e7776783SApple OSS Distributions 	if (!size || !result) {
206*e7776783SApple OSS Distributions 		return false;
207*e7776783SApple OSS Distributions 	}
208*e7776783SApple OSS Distributions 
209*e7776783SApple OSS Distributions 	if (0 == alignment) {
210*e7776783SApple OSS Distributions 		alignment = defaultAlignmentMask;
211*e7776783SApple OSS Distributions 	} else {
212*e7776783SApple OSS Distributions 		alignment--;
213*e7776783SApple OSS Distributions 	}
214*e7776783SApple OSS Distributions 
215*e7776783SApple OSS Distributions 	size = (size + defaultAlignmentMask) & ~defaultAlignmentMask;
216*e7776783SApple OSS Distributions 
217*e7776783SApple OSS Distributions 	LOCK();
218*e7776783SApple OSS Distributions 
219*e7776783SApple OSS Distributions 	for (index = 0; index < numElements; index++) {
220*e7776783SApple OSS Distributions 		thisStart = elements[index].start;
221*e7776783SApple OSS Distributions 		thisEnd = elements[index].end;
222*e7776783SApple OSS Distributions 		data = (thisStart + alignment) & ~alignment;
223*e7776783SApple OSS Distributions 		dataEnd = (data + size - 1);
224*e7776783SApple OSS Distributions 
225*e7776783SApple OSS Distributions 		ok = (dataEnd <= thisEnd);
226*e7776783SApple OSS Distributions 		if (ok) {
227*e7776783SApple OSS Distributions 			if (data != thisStart) {
228*e7776783SApple OSS Distributions 				if (dataEnd != thisEnd) {
229*e7776783SApple OSS Distributions 					if (allocElement( index + 1 )) {
230*e7776783SApple OSS Distributions 						elements[index++].end = data - 1;
231*e7776783SApple OSS Distributions 						elements[index].start = dataEnd + 1;
232*e7776783SApple OSS Distributions 						elements[index].end = thisEnd;
233*e7776783SApple OSS Distributions 					} else {
234*e7776783SApple OSS Distributions 						ok = false;
235*e7776783SApple OSS Distributions 					}
236*e7776783SApple OSS Distributions 				} else {
237*e7776783SApple OSS Distributions 					elements[index].end = data - 1;
238*e7776783SApple OSS Distributions 				}
239*e7776783SApple OSS Distributions 			} else {
240*e7776783SApple OSS Distributions 				if (dataEnd != thisEnd) {
241*e7776783SApple OSS Distributions 					elements[index].start = dataEnd + 1;
242*e7776783SApple OSS Distributions 				} else {
243*e7776783SApple OSS Distributions 					deallocElement( index );
244*e7776783SApple OSS Distributions 				}
245*e7776783SApple OSS Distributions 			}
246*e7776783SApple OSS Distributions 			if (ok) {
247*e7776783SApple OSS Distributions 				*result = data;
248*e7776783SApple OSS Distributions 			}
249*e7776783SApple OSS Distributions 			break;
250*e7776783SApple OSS Distributions 		}
251*e7776783SApple OSS Distributions 	}
252*e7776783SApple OSS Distributions 
253*e7776783SApple OSS Distributions 	UNLOCK();
254*e7776783SApple OSS Distributions 
255*e7776783SApple OSS Distributions 	return ok;
256*e7776783SApple OSS Distributions }
257*e7776783SApple OSS Distributions 
258*e7776783SApple OSS Distributions bool
allocateRange(IORangeScalar data,IORangeScalar size)259*e7776783SApple OSS Distributions IORangeAllocator::allocateRange( IORangeScalar data,
260*e7776783SApple OSS Distributions     IORangeScalar size )
261*e7776783SApple OSS Distributions {
262*e7776783SApple OSS Distributions 	IORangeScalar       thisStart, thisEnd;
263*e7776783SApple OSS Distributions 	IORangeScalar       dataEnd;
264*e7776783SApple OSS Distributions 	UInt32              index;
265*e7776783SApple OSS Distributions 	bool                found = false;
266*e7776783SApple OSS Distributions 
267*e7776783SApple OSS Distributions 	if (!size) {
268*e7776783SApple OSS Distributions 		return 0;
269*e7776783SApple OSS Distributions 	}
270*e7776783SApple OSS Distributions 
271*e7776783SApple OSS Distributions 	size = (size + defaultAlignmentMask) & ~defaultAlignmentMask;
272*e7776783SApple OSS Distributions 	dataEnd = data + size - 1;
273*e7776783SApple OSS Distributions 
274*e7776783SApple OSS Distributions 	LOCK();
275*e7776783SApple OSS Distributions 
276*e7776783SApple OSS Distributions 	for (index = 0;
277*e7776783SApple OSS Distributions 	    (!found) && (index < numElements);
278*e7776783SApple OSS Distributions 	    index++) {
279*e7776783SApple OSS Distributions 		thisStart = elements[index].start;
280*e7776783SApple OSS Distributions 		thisEnd = elements[index].end;
281*e7776783SApple OSS Distributions 
282*e7776783SApple OSS Distributions 		if (thisStart > data) {
283*e7776783SApple OSS Distributions 			break;
284*e7776783SApple OSS Distributions 		}
285*e7776783SApple OSS Distributions 		found = (dataEnd <= thisEnd);
286*e7776783SApple OSS Distributions 
287*e7776783SApple OSS Distributions 		if (found) {
288*e7776783SApple OSS Distributions 			if (data != thisStart) {
289*e7776783SApple OSS Distributions 				if (dataEnd != thisEnd) {
290*e7776783SApple OSS Distributions 					found = allocElement( index + 1 );
291*e7776783SApple OSS Distributions 					if (found) {
292*e7776783SApple OSS Distributions 						elements[index++].end = data - 1;
293*e7776783SApple OSS Distributions 						elements[index].start = dataEnd + 1;
294*e7776783SApple OSS Distributions 						elements[index].end = thisEnd;
295*e7776783SApple OSS Distributions 					}
296*e7776783SApple OSS Distributions 				} else {
297*e7776783SApple OSS Distributions 					elements[index].end = data - 1;
298*e7776783SApple OSS Distributions 				}
299*e7776783SApple OSS Distributions 			} else if (dataEnd != thisEnd) {
300*e7776783SApple OSS Distributions 				elements[index].start = dataEnd + 1;
301*e7776783SApple OSS Distributions 			} else {
302*e7776783SApple OSS Distributions 				deallocElement( index );
303*e7776783SApple OSS Distributions 			}
304*e7776783SApple OSS Distributions 		}
305*e7776783SApple OSS Distributions 	}
306*e7776783SApple OSS Distributions 
307*e7776783SApple OSS Distributions 	UNLOCK();
308*e7776783SApple OSS Distributions 
309*e7776783SApple OSS Distributions 	return found;
310*e7776783SApple OSS Distributions }
311*e7776783SApple OSS Distributions 
312*e7776783SApple OSS Distributions void
deallocate(IORangeScalar data,IORangeScalar size)313*e7776783SApple OSS Distributions IORangeAllocator::deallocate( IORangeScalar data,
314*e7776783SApple OSS Distributions     IORangeScalar size )
315*e7776783SApple OSS Distributions {
316*e7776783SApple OSS Distributions 	IORangeScalar       dataEnd;
317*e7776783SApple OSS Distributions 	UInt32              index;
318*e7776783SApple OSS Distributions 	bool                headContig = false;
319*e7776783SApple OSS Distributions 	bool                tailContig = false;
320*e7776783SApple OSS Distributions 
321*e7776783SApple OSS Distributions 	size = (size + defaultAlignmentMask) & ~defaultAlignmentMask;
322*e7776783SApple OSS Distributions 	dataEnd = data + size - 1;
323*e7776783SApple OSS Distributions 
324*e7776783SApple OSS Distributions 	LOCK();
325*e7776783SApple OSS Distributions 
326*e7776783SApple OSS Distributions 	for (index = 0; index < numElements; index++) {
327*e7776783SApple OSS Distributions 		if (elements[index].start < data) {
328*e7776783SApple OSS Distributions 			headContig = (data <= (elements[index].end + 1));
329*e7776783SApple OSS Distributions 			continue;
330*e7776783SApple OSS Distributions 		}
331*e7776783SApple OSS Distributions 		tailContig = ((data + size) >= elements[index].start);
332*e7776783SApple OSS Distributions 		break;
333*e7776783SApple OSS Distributions 	}
334*e7776783SApple OSS Distributions 
335*e7776783SApple OSS Distributions 	if (headContig) {
336*e7776783SApple OSS Distributions 		if (tailContig) {
337*e7776783SApple OSS Distributions 			elements[index - 1].end = elements[index].end;
338*e7776783SApple OSS Distributions 			deallocElement( index );
339*e7776783SApple OSS Distributions 		} else /*safe*/ if (dataEnd > elements[index - 1].end) {
340*e7776783SApple OSS Distributions 			elements[index - 1].end = dataEnd;
341*e7776783SApple OSS Distributions 		}
342*e7776783SApple OSS Distributions 	} else if (tailContig) {
343*e7776783SApple OSS Distributions 		if (data < elements[index].start) { /*safe*/
344*e7776783SApple OSS Distributions 			elements[index].start = data;
345*e7776783SApple OSS Distributions 		}
346*e7776783SApple OSS Distributions 	} else if (allocElement( index)) {
347*e7776783SApple OSS Distributions 		elements[index].start = data;
348*e7776783SApple OSS Distributions 		elements[index].end = dataEnd;
349*e7776783SApple OSS Distributions 	}
350*e7776783SApple OSS Distributions 
351*e7776783SApple OSS Distributions 	UNLOCK();
352*e7776783SApple OSS Distributions }
353*e7776783SApple OSS Distributions 
354*e7776783SApple OSS Distributions bool
serialize(OSSerialize * s) const355*e7776783SApple OSS Distributions IORangeAllocator::serialize(OSSerialize *s) const
356*e7776783SApple OSS Distributions {
357*e7776783SApple OSS Distributions 	OSArray *   array = OSArray::withCapacity( numElements * 2 );
358*e7776783SApple OSS Distributions 	OSNumber *  num;
359*e7776783SApple OSS Distributions 	UInt32      index;
360*e7776783SApple OSS Distributions 	bool        ret;
361*e7776783SApple OSS Distributions 
362*e7776783SApple OSS Distributions 	if (!array) {
363*e7776783SApple OSS Distributions 		return false;
364*e7776783SApple OSS Distributions 	}
365*e7776783SApple OSS Distributions 
366*e7776783SApple OSS Distributions 	LOCK();
367*e7776783SApple OSS Distributions 
368*e7776783SApple OSS Distributions 	for (index = 0; index < numElements; index++) {
369*e7776783SApple OSS Distributions 		if ((num = OSNumber::withNumber( elements[index].start,
370*e7776783SApple OSS Distributions 		    8 * sizeof(IORangeScalar)))) {
371*e7776783SApple OSS Distributions 			array->setObject(num);
372*e7776783SApple OSS Distributions 			num->release();
373*e7776783SApple OSS Distributions 		}
374*e7776783SApple OSS Distributions 		if ((num = OSNumber::withNumber( elements[index].end,
375*e7776783SApple OSS Distributions 		    8 * sizeof(IORangeScalar)))) {
376*e7776783SApple OSS Distributions 			array->setObject(num);
377*e7776783SApple OSS Distributions 			num->release();
378*e7776783SApple OSS Distributions 		}
379*e7776783SApple OSS Distributions 	}
380*e7776783SApple OSS Distributions 
381*e7776783SApple OSS Distributions 	UNLOCK();
382*e7776783SApple OSS Distributions 
383*e7776783SApple OSS Distributions 	ret = array->serialize(s);
384*e7776783SApple OSS Distributions 	array->release();
385*e7776783SApple OSS Distributions 
386*e7776783SApple OSS Distributions 	return ret;
387*e7776783SApple OSS Distributions }
388*e7776783SApple OSS Distributions 
389*e7776783SApple OSS Distributions IORangeScalar
getFreeCount(void)390*e7776783SApple OSS Distributions IORangeAllocator::getFreeCount( void )
391*e7776783SApple OSS Distributions {
392*e7776783SApple OSS Distributions 	UInt32              index;
393*e7776783SApple OSS Distributions 	IORangeScalar       sum = 0;
394*e7776783SApple OSS Distributions 
395*e7776783SApple OSS Distributions 	for (index = 0; index < numElements; index++) {
396*e7776783SApple OSS Distributions 		sum += elements[index].end - elements[index].start + 1;
397*e7776783SApple OSS Distributions 	}
398*e7776783SApple OSS Distributions 
399*e7776783SApple OSS Distributions 	return sum;
400*e7776783SApple OSS Distributions }
401