2 * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
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.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
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
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
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.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
29 #define IOKIT_ENABLE_SHARED_PTR
31 #include <libkern/c++/OSDictionary.h>
32 #include <libkern/c++/OSLib.h>
33 #include <libkern/c++/OSOrderedSet.h>
34 #include <libkern/c++/OSSharedPtr.h>
35 #include <os/cpp_util.h>
37 #define super OSCollection
39 OSDefineMetaClassAndStructors(OSOrderedSet
, OSCollection
)
40 OSMetaClassDefineReservedUnused(OSOrderedSet
, 0);
41 OSMetaClassDefineReservedUnused(OSOrderedSet
, 1);
42 OSMetaClassDefineReservedUnused(OSOrderedSet
, 2);
43 OSMetaClassDefineReservedUnused(OSOrderedSet
, 3);
44 OSMetaClassDefineReservedUnused(OSOrderedSet
, 4);
45 OSMetaClassDefineReservedUnused(OSOrderedSet
, 5);
46 OSMetaClassDefineReservedUnused(OSOrderedSet
, 6);
47 OSMetaClassDefineReservedUnused(OSOrderedSet
, 7);
51 OSTaggedPtr
<const OSMetaClassBase
> obj
;
54 #define EXT_CAST(obj) \
55 reinterpret_cast<OSObject *>(const_cast<OSMetaClassBase *>(obj))
59 initWithCapacity(unsigned int inCapacity
,
60 OSOrderFunction inOrdering
, void *inOrderingRef
)
68 if (inCapacity
> (UINT_MAX
/ sizeof(_Element
))) {
72 size
= sizeof(_Element
) * inCapacity
;
73 array
= (_Element
*) kalloc_container(size
);
79 capacity
= inCapacity
;
80 capacityIncrement
= (inCapacity
)? inCapacity
: 16;
81 ordering
= inOrdering
;
82 orderingRef
= inOrderingRef
;
85 OSCONTAINER_ACCUMSIZE(size
);
90 OSSharedPtr
<OSOrderedSet
>
92 withCapacity(unsigned int capacity
,
93 OSOrderFunction ordering
, void * orderingRef
)
95 auto me
= OSMakeShared
<OSOrderedSet
>();
97 if (me
&& !me
->initWithCapacity(capacity
, ordering
, orderingRef
)) {
107 (void) super::setOptions(0, kImmutable
);
111 kfree(array
, sizeof(_Element
) * capacity
);
112 OSCONTAINER_ACCUMSIZE( -(sizeof(_Element
) * capacity
));
119 OSOrderedSet::getCount() const
124 OSOrderedSet::getCapacity() const
129 OSOrderedSet::getCapacityIncrement() const
131 return capacityIncrement
;
134 OSOrderedSet::setCapacityIncrement(unsigned int increment
)
136 capacityIncrement
= (increment
)? increment
: 16;
137 return capacityIncrement
;
141 OSOrderedSet::ensureCapacity(unsigned int newCapacity
)
144 vm_size_t finalCapacity
;
145 vm_size_t oldSize
, newSize
;
147 if (newCapacity
<= capacity
) {
152 finalCapacity
= (((newCapacity
- 1) / capacityIncrement
) + 1)
154 if (finalCapacity
< newCapacity
) {
157 newSize
= sizeof(_Element
) * finalCapacity
;
159 newArray
= (_Element
*) kallocp_container(&newSize
);
161 // use all of the actual allocation size
162 finalCapacity
= (newSize
/ sizeof(_Element
));
163 if (finalCapacity
> UINT_MAX
) {
164 // failure, too large
165 kfree(newArray
, newSize
);
169 oldSize
= sizeof(_Element
) * capacity
;
171 OSCONTAINER_ACCUMSIZE(((size_t)newSize
) - ((size_t)oldSize
));
173 bcopy(array
, newArray
, oldSize
);
174 bzero(&newArray
[capacity
], newSize
- oldSize
);
175 kfree(array
, oldSize
);
177 capacity
= (unsigned int) finalCapacity
;
184 OSOrderedSet::flushCollection()
190 for (i
= 0; i
< count
; i
++) {
191 array
[i
].obj
.reset();
199 OSOrderedSet::setObject(unsigned int index
, const OSMetaClassBase
*anObject
)
202 unsigned int newCount
= count
+ 1;
204 if ((index
> count
) || !anObject
) {
208 if (containsObject(anObject
)) {
212 // do we need more space?
213 if (newCount
> capacity
&& newCount
> ensureCapacity(newCount
)) {
218 if (index
!= count
) {
219 for (i
= count
; i
> index
; i
--) {
220 array
[i
] = os::move(array
[i
- 1]);
223 array
[index
].obj
.reset(anObject
, OSRetain
);
230 OSOrderedSet::setObject(unsigned int index
, OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
232 return setObject(index
, anObject
.get());
236 OSOrderedSet::setFirstObject(const OSMetaClassBase
*anObject
)
238 return setObject(0, anObject
);
242 OSOrderedSet::setFirstObject(OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
244 return setFirstObject(anObject
.get());
248 OSOrderedSet::setLastObject(const OSMetaClassBase
*anObject
)
250 return setObject( count
, anObject
);
254 OSOrderedSet::setLastObject(OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
256 return setLastObject(anObject
.get());
260 #define ORDER(obj1, obj2) \
261 (ordering ? ((*ordering)( (const OSObject *) obj1, (const OSObject *) obj2, orderingRef)) : 0)
264 OSOrderedSet::setObject(const OSMetaClassBase
*anObject
)
268 // queue it behind those with same priority
270 (i
< count
) && (ORDER(array
[i
].obj
.get(), anObject
) >= 0);
274 return setObject(i
, anObject
);
278 OSOrderedSet::setObject(OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
280 return setObject(anObject
.get());
284 OSOrderedSet::removeObject(const OSMetaClassBase
*anObject
)
286 bool deleted
= false;
289 for (i
= 0; i
< count
; i
++) {
291 array
[i
- 1] = os::move(array
[i
]);
292 } else if (array
[i
].obj
== anObject
) {
294 haveUpdated(); // Pity we can't flush the log
295 array
[i
].obj
.reset();
305 OSOrderedSet::removeObject(OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
307 return removeObject(anObject
.get());
311 OSOrderedSet::containsObject(const OSMetaClassBase
*anObject
) const
313 return anObject
&& member(anObject
);
317 OSOrderedSet::member(const OSMetaClassBase
*anObject
) const
322 (i
< count
) && (array
[i
].obj
!= anObject
);
331 OSOrderedSet::getObject( unsigned int index
) const
333 if (index
>= count
) {
337 return const_cast<OSObject
*>((const OSObject
*) array
[index
].obj
.get());
341 OSOrderedSet::getFirstObject() const
344 return const_cast<OSObject
*>((const OSObject
*) array
[0].obj
.get());
351 OSOrderedSet::getLastObject() const
354 return const_cast<OSObject
*>((const OSObject
*) array
[count
- 1].obj
.get());
361 OSOrderedSet::orderObject( const OSMetaClassBase
* anObject
)
363 return ORDER( anObject
, NULL
);
367 OSOrderedSet::getOrderingRef()
373 OSOrderedSet::isEqualTo(const OSOrderedSet
*anOrderedSet
) const
377 if (this == anOrderedSet
) {
381 if (count
!= anOrderedSet
->getCount()) {
385 for (i
= 0; i
< count
; i
++) {
386 if (!array
[i
].obj
->isEqualTo(anOrderedSet
->getObject(i
))) {
395 OSOrderedSet::isEqualTo(const OSMetaClassBase
*anObject
) const
399 oSet
= OSDynamicCast(OSOrderedSet
, anObject
);
401 return isEqualTo(oSet
);
408 OSOrderedSet::iteratorSize() const
410 return sizeof(unsigned int);
414 OSOrderedSet::initIterator(void *inIterator
) const
416 unsigned int *iteratorP
= (unsigned int *) inIterator
;
424 getNextObjectForIterator(void *inIterator
, OSObject
**ret
) const
426 unsigned int *iteratorP
= (unsigned int *) inIterator
;
427 unsigned int index
= (*iteratorP
)++;
430 *ret
= const_cast<OSObject
*>((const OSObject
*) array
[index
].obj
.get());
440 OSOrderedSet::setOptions(unsigned options
, unsigned mask
, void *)
442 unsigned old
= super::setOptions(options
, mask
);
443 if ((old
^ options
) & mask
) {
444 // Value changed need to recurse over all of the child collections
445 for (unsigned i
= 0; i
< count
; i
++) {
446 OSCollection
*coll
= OSDynamicCast(OSCollection
, array
[i
].obj
.get());
448 coll
->setOptions(options
, mask
);
456 OSSharedPtr
<OSCollection
>
457 OSOrderedSet::copyCollection(OSDictionary
*cycleDict
)
459 OSSharedPtr
<OSDictionary
> ourCycleDict
;
460 OSSharedPtr
<OSCollection
> ret
;
461 OSSharedPtr
<OSOrderedSet
> newSet
;
464 ourCycleDict
= OSDictionary::withCapacity(16);
468 cycleDict
= ourCycleDict
.get();
473 ret
= super::copyCollection(cycleDict
);
478 // Duplicate the set with no contents
479 newSet
= OSOrderedSet::withCapacity(capacity
, ordering
, orderingRef
);
484 // Insert object into cycle Dictionary
485 cycleDict
->setObject((const OSSymbol
*) this, newSet
.get());
487 newSet
->capacityIncrement
= capacityIncrement
;
489 // Now copy over the contents to the new duplicate
490 for (unsigned int i
= 0; i
< count
; i
++) {
491 OSObject
*obj
= EXT_CAST(array
[i
].obj
.get());
492 OSCollection
*coll
= OSDynamicCast(OSCollection
, obj
);
494 OSSharedPtr
<OSCollection
> newColl
= coll
->copyCollection(cycleDict
);
496 obj
= newColl
.get(); // Rely on cycleDict ref for a bit
502 newSet
->setLastObject(obj
);
505 ret
= os::move(newSet
);