]>
git.saurik.com Git - apple/xnu.git/blob - libkern/c++/OSSet.cpp
2 * Copyright (c) 2000, 2014 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@
28 /* IOSet.m created by rsulack on Thu 11-Jun-1998 */
30 #define IOKIT_ENABLE_SHARED_PTR
32 #include <libkern/c++/OSArray.h>
33 #include <libkern/c++/OSDictionary.h>
34 #include <libkern/c++/OSSerialize.h>
35 #include <libkern/c++/OSSet.h>
36 #include <libkern/c++/OSSharedPtr.h>
37 #include <os/cpp_util.h>
38 #include <kern/zalloc.h>
40 #define super OSCollection
42 OSDefineMetaClassAndStructorsWithZone(OSSet
, OSCollection
,
44 OSMetaClassDefineReservedUnused(OSSet
, 0);
45 OSMetaClassDefineReservedUnused(OSSet
, 1);
46 OSMetaClassDefineReservedUnused(OSSet
, 2);
47 OSMetaClassDefineReservedUnused(OSSet
, 3);
48 OSMetaClassDefineReservedUnused(OSSet
, 4);
49 OSMetaClassDefineReservedUnused(OSSet
, 5);
50 OSMetaClassDefineReservedUnused(OSSet
, 6);
51 OSMetaClassDefineReservedUnused(OSSet
, 7);
53 #define EXT_CAST(obj) \
54 reinterpret_cast<OSObject *>(const_cast<OSMetaClassBase *>(obj))
57 OSSet::initWithCapacity(unsigned int inCapacity
)
63 members
= OSArray::withCapacity(inCapacity
);
72 OSSet::initWithObjects(const OSObject
*inObjects
[],
74 unsigned int inCapacity
)
76 unsigned int capacity
= inCount
;
79 if (inCount
> inCapacity
) {
83 capacity
= inCapacity
;
86 if (!inObjects
|| !initWithCapacity(capacity
)) {
90 for (unsigned int i
= 0; i
< inCount
; i
++) {
91 // xx-review: no test here for failure of setObject()
92 if (members
->getCount() < capacity
) {
93 setObject(inObjects
[i
]);
103 OSSet::initWithArray(const OSArray
*inArray
,
104 unsigned int inCapacity
)
110 return initWithObjects((const OSObject
**) inArray
->array
,
111 inArray
->count
, inCapacity
);
115 OSSet::initWithSet(const OSSet
*inSet
,
116 unsigned int inCapacity
)
118 return initWithArray(inSet
->members
.get(), inCapacity
);
122 OSSet::withCapacity(unsigned int capacity
)
124 OSSharedPtr
<OSSet
> me
= OSMakeShared
<OSSet
>();
126 if (me
&& !me
->initWithCapacity(capacity
)) {
134 OSSet::withObjects(const OSObject
*objects
[],
136 unsigned int capacity
)
138 OSSharedPtr
<OSSet
> me
= OSMakeShared
<OSSet
>();
140 if (me
&& !me
->initWithObjects(objects
, count
, capacity
)) {
148 OSSet::withArray(const OSArray
*array
,
149 unsigned int capacity
)
151 OSSharedPtr
<OSSet
> me
= OSMakeShared
<OSSet
>();
153 if (me
&& !me
->initWithArray(array
, capacity
)) {
161 OSSet::withSet(const OSSet
*set
,
162 unsigned int capacity
)
164 OSSharedPtr
<OSSet
> me
= OSMakeShared
<OSSet
>();;
166 if (me
&& !me
->initWithSet(set
, capacity
)) {
177 (void) members
->super::setOptions(0, kImmutable
);
184 OSSet::getCount() const
186 return members
->count
;
190 OSSet::getCapacity() const
192 return members
->capacity
;
196 OSSet::getCapacityIncrement() const
198 return members
->capacityIncrement
;
202 OSSet::setCapacityIncrement(unsigned int increment
)
204 return members
->setCapacityIncrement(increment
);
208 OSSet::ensureCapacity(unsigned int newCapacity
)
210 return members
->ensureCapacity(newCapacity
);
214 OSSet::flushCollection()
217 members
->flushCollection();
221 OSSet::setObject(const OSMetaClassBase
*anObject
)
223 if (containsObject(anObject
)) {
227 return members
->setObject(anObject
);
232 OSSet::setObject(OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
234 return setObject(anObject
.get());
238 OSSet::merge(const OSArray
* array
)
240 const OSMetaClassBase
* anObject
= NULL
;
243 for (int i
= 0; (anObject
= array
->getObject(i
)); i
++) {
244 /* setObject() returns false if the object is already in the set,
245 * so we have to check beforehand here with containsObject().
247 if (containsObject(anObject
)) {
250 if (!setObject(anObject
)) {
259 OSSet::merge(const OSSet
* set
)
261 return merge(set
->members
.get());
265 OSSet::removeObject(const OSMetaClassBase
*anObject
)
267 const OSMetaClassBase
*probeObject
;
269 for (int i
= 0; (probeObject
= members
->getObject(i
)); i
++) {
270 if (probeObject
== anObject
) {
272 members
->removeObject(i
);
279 OSSet::removeObject(OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
281 removeObject(anObject
.get());
286 OSSet::containsObject(const OSMetaClassBase
*anObject
) const
288 return anObject
&& member(anObject
);
292 OSSet::member(const OSMetaClassBase
*anObject
) const
294 OSMetaClassBase
*probeObject
;
296 for (int i
= 0; (probeObject
= members
->getObject(i
)); i
++) {
297 if (probeObject
== anObject
) {
306 OSSet::getAnyObject() const
308 return members
->getObject(0);
312 OSSet::isEqualTo(const OSSet
*aSet
) const
316 const OSMetaClassBase
*obj1
;
317 const OSMetaClassBase
*obj2
;
323 count
= members
->count
;
324 if (count
!= aSet
->getCount()) {
328 for (i
= 0; i
< count
; i
++) {
329 obj1
= aSet
->members
->getObject(i
);
330 if (containsObject(obj1
)) {
333 obj2
= members
->getObject(i
);
334 if (!obj1
|| !obj2
) {
338 if (!obj1
->isEqualTo(obj2
)) {
347 OSSet::isEqualTo(const OSMetaClassBase
*anObject
) const
351 otherSet
= OSDynamicCast(OSSet
, anObject
);
353 return isEqualTo(otherSet
);
360 OSSet::iteratorSize() const
362 return sizeof(unsigned int);
366 OSSet::initIterator(void *inIterator
) const
368 unsigned int *iteratorP
= (unsigned int *) inIterator
;
375 OSSet::getNextObjectForIterator(void *inIterator
, OSObject
**ret
) const
377 unsigned int *iteratorP
= (unsigned int *) inIterator
;
378 unsigned int index
= (*iteratorP
)++;
380 if (index
< members
->count
) {
381 *ret
= members
->getObject(index
);
390 OSSet::serialize(OSSerialize
*s
) const
392 const OSMetaClassBase
*o
;
394 if (s
->previouslySerialized(this)) {
398 if (!s
->addXMLStartTag(this, "set")) {
402 for (int i
= 0; (o
= members
->getObject(i
)); i
++) {
403 if (!o
->serialize(s
)) {
408 return s
->addXMLEndTag("set");
412 OSSet::setOptions(unsigned options
, unsigned mask
, void *)
414 unsigned old
= super::setOptions(options
, mask
);
415 if ((old
^ options
) & mask
) {
416 members
->setOptions(options
, mask
);
422 OSSharedPtr
<OSCollection
>
423 OSSet::copyCollection(OSDictionary
*cycleDict
)
425 OSSharedPtr
<OSDictionary
> ourCycleDict
;
426 OSSharedPtr
<OSCollection
> ret
;
427 OSSharedPtr
<OSSet
> newSet
;
430 ourCycleDict
= OSDictionary::withCapacity(16);
434 cycleDict
= ourCycleDict
.get();
439 ret
= super::copyCollection(cycleDict
);
441 continue; // Found it
443 newSet
= OSSet::withCapacity(members
->capacity
);
445 continue; // Couldn't create new set abort
447 // Insert object into cycle Dictionary
448 cycleDict
->setObject((const OSSymbol
*) this, newSet
.get());
450 OSArray
*newMembers
= newSet
->members
.get();
451 newMembers
->capacityIncrement
= members
->capacityIncrement
;
453 // Now copy over the contents into the new duplicate
454 for (unsigned int i
= 0; i
< members
->count
; i
++) {
455 OSObject
*obj
= EXT_CAST(members
->array
[i
].get());
456 OSCollection
*coll
= OSDynamicCast(OSCollection
, obj
);
458 OSSharedPtr
<OSCollection
> newColl
= coll
->copyCollection(cycleDict
);
460 obj
= newColl
.get(); // Rely on cycleDict ref for a bit
465 newMembers
->setObject(obj
);
468 ret
= os::move(newSet
);