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@
28 /* OSDictionary.m created by rsulack on Fri 12-Sep-1997 */
29 /* OSDictionary.cpp converted to C++ by gvdl on Fri 1998-10-30 */
30 /* OSDictionary.cpp rewritten by gvdl on Fri 1998-10-30 */
32 #define IOKIT_ENABLE_SHARED_PTR
34 #include <libkern/c++/OSArray.h>
35 #include <libkern/c++/OSCollectionIterator.h>
36 #include <libkern/c++/OSDictionary.h>
37 #include <libkern/c++/OSLib.h>
38 #include <libkern/c++/OSSerialize.h>
39 #include <libkern/c++/OSSharedPtr.h>
40 #include <libkern/c++/OSSymbol.h>
41 #include <os/cpp_util.h>
43 #define super OSCollection
45 OSDefineMetaClassAndStructorsWithZone(OSDictionary
, OSCollection
,
46 (zone_create_flags_t
) (ZC_CACHING
| ZC_ZFREE_CLEARMEM
))
47 OSMetaClassDefineReservedUnused(OSDictionary
, 0);
48 OSMetaClassDefineReservedUnused(OSDictionary
, 1);
49 OSMetaClassDefineReservedUnused(OSDictionary
, 2);
50 OSMetaClassDefineReservedUnused(OSDictionary
, 3);
51 OSMetaClassDefineReservedUnused(OSDictionary
, 4);
52 OSMetaClassDefineReservedUnused(OSDictionary
, 5);
53 OSMetaClassDefineReservedUnused(OSDictionary
, 6);
54 OSMetaClassDefineReservedUnused(OSDictionary
, 7);
56 #define EXT_CAST(obj) \
57 reinterpret_cast<OSObject *>(const_cast<OSMetaClassBase *>(obj))
60 void qsort(void *, size_t, size_t, int (*)(const void *, const void *));
64 OSDictionary::dictEntry::compare(const void *_e1
, const void *_e2
)
66 const OSDictionary::dictEntry
*e1
= (const OSDictionary::dictEntry
*)_e1
;
67 const OSDictionary::dictEntry
*e2
= (const OSDictionary::dictEntry
*)_e2
;
69 if ((uintptr_t)e1
->key
.get() == (uintptr_t)e2
->key
.get()) {
73 return (uintptr_t)e1
->key
.get() > (uintptr_t)e2
->key
.get() ? 1 : -1;
77 OSDictionary::sortBySymbol(void)
79 qsort(dictionary
, count
, sizeof(OSDictionary::dictEntry
),
80 &OSDictionary::dictEntry::compare
);
84 OSDictionary::initWithCapacity(unsigned int inCapacity
)
90 if (inCapacity
> (UINT_MAX
/ sizeof(dictEntry
))) {
94 unsigned int size
= inCapacity
* sizeof(dictEntry
);
97 dictionary
= (dictEntry
*) kalloc_container(size
);
102 os::uninitialized_value_construct(dictionary
, dictionary
+ inCapacity
);
103 OSCONTAINER_ACCUMSIZE(size
);
106 capacity
= inCapacity
;
107 capacityIncrement
= (inCapacity
)? inCapacity
: 16;
113 OSDictionary::initWithObjects(const OSObject
*objects
[],
114 const OSSymbol
*keys
[],
115 unsigned int theCount
,
116 unsigned int theCapacity
)
118 unsigned int newCapacity
= theCount
;
120 if (!objects
|| !keys
) {
125 if (theCount
> theCapacity
) {
129 newCapacity
= theCapacity
;
132 if (!initWithCapacity(newCapacity
)) {
136 for (unsigned int i
= 0; i
< theCount
; i
++) {
137 const OSMetaClassBase
*newObject
= *objects
++;
139 if (!newObject
|| !keys
[i
] || !setObject(keys
[i
], newObject
)) {
148 OSDictionary::initWithObjects(const OSObject
*objects
[],
149 const OSString
*keys
[],
150 unsigned int theCount
,
151 unsigned int theCapacity
)
153 unsigned int newCapacity
= theCount
;
155 if (!objects
|| !keys
) {
160 if (theCount
> theCapacity
) {
164 newCapacity
= theCapacity
;
167 if (!initWithCapacity(newCapacity
)) {
171 for (unsigned int i
= 0; i
< theCount
; i
++) {
172 OSSharedPtr
<const OSSymbol
> key
= OSSymbol::withString(*keys
++);
173 const OSMetaClassBase
*newObject
= *objects
++;
179 if (!newObject
|| !setObject(key
.get(), newObject
)) {
188 OSDictionary::initWithDictionary(const OSDictionary
*dict
,
189 unsigned int theCapacity
)
191 unsigned int newCapacity
;
197 newCapacity
= dict
->count
;
200 if (dict
->count
> theCapacity
) {
204 newCapacity
= theCapacity
;
207 if (!initWithCapacity(newCapacity
)) {
212 for (unsigned int i
= 0; i
< count
; i
++) {
213 dictionary
[i
].key
= dict
->dictionary
[i
].key
;
214 dictionary
[i
].value
= dict
->dictionary
[i
].value
;
217 if ((kSort
& fOptions
) && !(kSort
& dict
->fOptions
)) {
224 OSSharedPtr
<OSDictionary
>
225 OSDictionary::withCapacity(unsigned int capacity
)
227 OSSharedPtr
<OSDictionary
> me
= OSMakeShared
<OSDictionary
>();
229 if (me
&& !me
->initWithCapacity(capacity
)) {
236 OSSharedPtr
<OSDictionary
>
237 OSDictionary::withObjects(const OSObject
*objects
[],
238 const OSSymbol
*keys
[],
240 unsigned int capacity
)
242 OSSharedPtr
<OSDictionary
> me
= OSMakeShared
<OSDictionary
>();
244 if (me
&& !me
->initWithObjects(objects
, keys
, count
, capacity
)) {
251 OSSharedPtr
<OSDictionary
>
252 OSDictionary::withObjects(const OSObject
*objects
[],
253 const OSString
*keys
[],
255 unsigned int capacity
)
257 OSSharedPtr
<OSDictionary
> me
= OSMakeShared
<OSDictionary
>();
259 if (me
&& !me
->initWithObjects(objects
, keys
, count
, capacity
)) {
266 OSSharedPtr
<OSDictionary
>
267 OSDictionary::withDictionary(const OSDictionary
*dict
,
268 unsigned int capacity
)
270 OSSharedPtr
<OSDictionary
> me
= OSMakeShared
<OSDictionary
>();
272 if (me
&& !me
->initWithDictionary(dict
, capacity
)) {
282 (void) super::setOptions(0, kImmutable
);
285 kfree(dictionary
, capacity
* sizeof(dictEntry
));
286 OSCONTAINER_ACCUMSIZE( -(capacity
* sizeof(dictEntry
)));
293 OSDictionary::getCount() const
298 OSDictionary::getCapacity() const
304 OSDictionary::getCapacityIncrement() const
306 return capacityIncrement
;
310 OSDictionary::setCapacityIncrement(unsigned int increment
)
312 capacityIncrement
= (increment
)? increment
: 16;
314 return capacityIncrement
;
318 OSDictionary::ensureCapacity(unsigned int newCapacity
)
321 vm_size_t finalCapacity
;
322 vm_size_t oldSize
, newSize
;
324 if (newCapacity
<= capacity
) {
329 finalCapacity
= (((newCapacity
- 1) / capacityIncrement
) + 1)
332 // integer overflow check
333 if (finalCapacity
< newCapacity
) {
337 newSize
= sizeof(dictEntry
) * finalCapacity
;
339 newDict
= (dictEntry
*) kallocp_container(&newSize
);
341 // use all of the actual allocation size
342 finalCapacity
= (newSize
/ sizeof(dictEntry
));
343 if (finalCapacity
> UINT_MAX
) {
344 // failure, too large
345 kfree(newDict
, newSize
);
349 oldSize
= sizeof(dictEntry
) * capacity
;
351 os::uninitialized_move(dictionary
, dictionary
+ capacity
, newDict
);
352 os::uninitialized_value_construct(newDict
+ capacity
, newDict
+ finalCapacity
);
353 os::destroy(dictionary
, dictionary
+ capacity
);
355 OSCONTAINER_ACCUMSIZE(((size_t)newSize
) - ((size_t)oldSize
));
356 kfree(dictionary
, oldSize
);
358 dictionary
= newDict
;
359 capacity
= (unsigned int) finalCapacity
;
366 OSDictionary::flushCollection()
370 for (unsigned int i
= 0; i
< count
; i
++) {
371 dictionary
[i
].key
->taggedRelease(OSTypeID(OSCollection
));
372 dictionary
[i
].value
->taggedRelease(OSTypeID(OSCollection
));
379 setObject(const OSSymbol
*aKey
, const OSMetaClassBase
*anObject
, bool onlyAdd
)
384 if (!anObject
|| !aKey
) {
388 // if the key exists, replace the object
390 if (fOptions
& kSort
) {
391 i
= OSSymbol::bsearch(aKey
, &dictionary
[0], count
, sizeof(dictionary
[0]));
392 exists
= (i
< count
) && (aKey
== dictionary
[i
].key
);
394 for (exists
= false, i
= 0; i
< count
; i
++) {
395 if ((exists
= (aKey
== dictionary
[i
].key
))) {
406 OSTaggedSharedPtr
<const OSMetaClassBase
, OSCollection
> oldObject
;
410 dictionary
[i
].value
.reset(anObject
, OSRetain
);
414 // add new key, possibly extending our capacity
415 if (count
>= capacity
&& count
>= ensureCapacity(count
+ 1)) {
421 new (&dictionary
[count
]) dictEntry();
422 os::move_backward(&dictionary
[i
], &dictionary
[count
], &dictionary
[count
+ 1]);
424 dictionary
[i
].key
.reset(aKey
, OSRetain
);
425 dictionary
[i
].value
.reset(anObject
, OSRetain
);
433 setObject(const OSSymbol
*aKey
, const OSMetaClassBase
*anObject
)
435 return setObject(aKey
, anObject
, false);
439 OSDictionary::setObject(OSSharedPtr
<const OSSymbol
> const& aKey
, OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
441 return setObject(aKey
.get(), anObject
.get());
445 OSDictionary::setObject(const OSString
* aKey
, OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
447 return setObject(aKey
, anObject
.get());
451 OSDictionary::setObject(const char* aKey
, OSSharedPtr
<const OSMetaClassBase
> const& anObject
)
453 return setObject(aKey
, anObject
.get());
457 OSDictionary::removeObject(const OSSymbol
*aKey
)
466 // if the key exists, remove the object
468 if (fOptions
& kSort
) {
469 i
= OSSymbol::bsearch(aKey
, &dictionary
[0], count
, sizeof(dictionary
[0]));
470 exists
= (i
< count
) && (aKey
== dictionary
[i
].key
);
472 for (exists
= false, i
= 0; i
< count
; i
++) {
473 if ((exists
= (aKey
== dictionary
[i
].key
))) {
480 dictEntry oldEntry
= dictionary
[i
];
485 bcopy(&dictionary
[i
+ 1], &dictionary
[i
], (count
- i
) * sizeof(dictionary
[0]));
487 oldEntry
.key
->taggedRelease(OSTypeID(OSCollection
));
488 oldEntry
.value
->taggedRelease(OSTypeID(OSCollection
));
494 // Returns true on success, false on an error condition.
496 OSDictionary::merge(const OSDictionary
*srcDict
)
498 const OSSymbol
* sym
;
499 OSSharedPtr
<OSCollectionIterator
> iter
;
501 if (!OSDynamicCast(OSDictionary
, srcDict
)) {
505 iter
= OSCollectionIterator::withCollection(const_cast<OSDictionary
*>(srcDict
));
510 while ((sym
= (const OSSymbol
*)iter
->getNextObject())) {
511 const OSMetaClassBase
* obj
;
513 obj
= srcDict
->getObject(sym
);
514 if (!setObject(sym
, obj
)) {
523 OSDictionary::getObject(const OSSymbol
*aKey
) const
525 unsigned int i
, l
= 0, r
= count
;
531 // if the key exists, return the object
533 // inline OSSymbol::bsearch in this performance critical codepath
534 // for performance, the compiler can't do that due to the genericity
535 // of OSSymbol::bsearch
537 // If we have less than 4 objects, scanning is faster.
538 if (count
> 4 && (fOptions
& kSort
)) {
541 if (aKey
== dictionary
[i
].key
) {
542 return const_cast<OSObject
*> ((const OSObject
*)dictionary
[i
].value
.get());
545 if ((uintptr_t)aKey
< (uintptr_t)dictionary
[i
].key
.get()) {
552 for (i
= l
; i
< r
; i
++) {
553 if (aKey
== dictionary
[i
].key
) {
554 return const_cast<OSObject
*> ((const OSObject
*)dictionary
[i
].value
.get());
563 #define OBJECT_WRAP_1(cmd, k) \
565 OSSharedPtr<const OSSymbol> tmpKey = k; \
566 OSObject *retObj = NULL; \
568 retObj = cmd(tmpKey.get()); \
573 #define OBJECT_WRAP_2(cmd, k, o) \
575 OSSharedPtr<const OSSymbol> tmpKey = k; \
576 bool ret = cmd(tmpKey.get(), o); \
581 #define OBJECT_WRAP_3(cmd, k) \
583 OSSharedPtr<const OSSymbol> tmpKey = k; \
591 OSDictionary::setObject(const OSString
*aKey
, const OSMetaClassBase
*anObject
)
592 OBJECT_WRAP_2(setObject
, OSSymbol::withString(aKey
), anObject
)
594 OSDictionary::setObject(const char *aKey
, const OSMetaClassBase
*anObject
)
595 OBJECT_WRAP_2(setObject
, OSSymbol::withCString(aKey
), anObject
)
597 OSObject
*OSDictionary::getObject(const OSString
* aKey
) const
598 OBJECT_WRAP_1(getObject
, OSSymbol::existingSymbolForString(aKey
))
599 OSObject
*OSDictionary::getObject(const char *aKey
) const
600 OBJECT_WRAP_1(getObject
, OSSymbol::existingSymbolForCString(aKey
))
603 OSDictionary::removeObject(const OSString
*aKey
)
604 OBJECT_WRAP_3(removeObject
, OSSymbol::existingSymbolForString(aKey
))
606 OSDictionary::removeObject(const char *aKey
)
607 OBJECT_WRAP_3(removeObject
, OSSymbol::existingSymbolForCString(aKey
))
610 OSDictionary::isEqualTo(const OSDictionary
*srcDict
, const OSCollection
*keys
) const
612 OSSharedPtr
<OSCollectionIterator
> iter
;
613 unsigned int keysCount
;
614 const OSMetaClassBase
* obj1
;
615 const OSMetaClassBase
* obj2
;
619 if (this == srcDict
) {
623 keysCount
= keys
->getCount();
624 if ((count
< keysCount
) || (srcDict
->getCount() < keysCount
)) {
628 iter
= OSCollectionIterator::withCollection(keys
);
634 while ((aKey
= OSDynamicCast(OSString
, iter
->getNextObject()))) {
635 obj1
= getObject(aKey
);
636 obj2
= srcDict
->getObject(aKey
);
637 if (!obj1
|| !obj2
) {
642 if (!obj1
->isEqualTo(obj2
)) {
652 OSDictionary::isEqualTo(const OSDictionary
*srcDict
) const
655 const OSMetaClassBase
* obj
;
657 if (this == srcDict
) {
661 if (count
!= srcDict
->getCount()) {
665 for (i
= 0; i
< count
; i
++) {
666 obj
= srcDict
->getObject(dictionary
[i
].key
.get());
671 if (!dictionary
[i
].value
->isEqualTo(obj
)) {
680 OSDictionary::isEqualTo(const OSMetaClassBase
*anObject
) const
684 dict
= OSDynamicCast(OSDictionary
, anObject
);
686 return isEqualTo(dict
);
693 OSDictionary::iteratorSize() const
695 return sizeof(unsigned int);
699 OSDictionary::initIterator(void *inIterator
) const
701 unsigned int *iteratorP
= (unsigned int *) inIterator
;
708 OSDictionary::getNextObjectForIterator(void *inIterator
, OSObject
**ret
) const
710 unsigned int *iteratorP
= (unsigned int *) inIterator
;
711 unsigned int index
= (*iteratorP
)++;
714 *ret
= const_cast<OSSymbol
*>(dictionary
[index
].key
.get());
723 OSDictionary::serialize(OSSerialize
*s
) const
725 if (s
->previouslySerialized(this)) {
729 if (!s
->addXMLStartTag(this, "dict")) {
733 for (unsigned i
= 0; i
< count
; i
++) {
734 const OSSymbol
*key
= dictionary
[i
].key
.get();
736 // due the nature of the XML syntax, this must be a symbol
737 if (!key
->metaCast("OSSymbol")) {
740 if (!s
->addString("<key>")) {
743 const char *c
= key
->getCStringNoCopy();
746 if (!s
->addString("<")) {
749 } else if (*c
== '>') {
750 if (!s
->addString(">")) {
753 } else if (*c
== '&') {
754 if (!s
->addString("&")) {
758 if (!s
->addChar(*c
)) {
764 if (!s
->addXMLEndTag("key")) {
768 if (!dictionary
[i
].value
->serialize(s
)) {
773 return s
->addXMLEndTag("dict");
777 OSDictionary::setOptions(unsigned options
, unsigned mask
, void *)
779 unsigned old
= super::setOptions(options
, mask
);
780 if ((old
^ options
) & mask
) {
781 // Value changed need to recurse over all of the child collections
782 for (unsigned i
= 0; i
< count
; i
++) {
783 OSCollection
*v
= OSDynamicCast(OSCollection
, dictionary
[i
].value
.get());
785 v
->setOptions(options
, mask
);
790 if (!(old
& kSort
) && (fOptions
& kSort
)) {
797 OSSharedPtr
<OSCollection
>
798 OSDictionary::copyCollection(OSDictionary
*cycleDict
)
800 OSSharedPtr
<OSDictionary
> ourCycleDict
;
801 OSSharedPtr
<OSCollection
> ret
;
802 OSSharedPtr
<OSDictionary
> newDict
;
805 ourCycleDict
= OSDictionary::withCapacity(16);
809 cycleDict
= ourCycleDict
.get();
814 ret
= super::copyCollection(cycleDict
);
819 newDict
= OSDictionary::withDictionary(this);
824 // Insert object into cycle Dictionary
825 cycleDict
->setObject((const OSSymbol
*) this, newDict
.get());
827 for (unsigned int i
= 0; i
< count
; i
++) {
828 const OSMetaClassBase
*obj
= dictionary
[i
].value
.get();
829 OSTaggedSharedPtr
<OSCollection
, OSCollection
> coll(OSDynamicCast(OSCollection
, EXT_CAST(obj
)), OSNoRetain
);
832 OSSharedPtr
<OSCollection
> newColl
= coll
->copyCollection(cycleDict
);
836 newDict
->dictionary
[i
].value
.detach();
837 newDict
->dictionary
[i
].value
.reset(newColl
.get(), OSRetain
);
841 ret
= os::move(newDict
);
848 OSDictionary::copyKeys(void)
850 OSSharedPtr
<OSArray
> array
;
852 array
= OSArray::withCapacity(count
);
857 for (unsigned int i
= 0; i
< count
; i
++) {
858 if (!array
->setObject(i
, dictionary
[i
].key
.get())) {
866 OSDictionary::iterateObjects(void * refcon
, bool (*callback
)(void * refcon
, const OSSymbol
* key
, OSObject
* object
))
868 unsigned int initialUpdateStamp
;
871 initialUpdateStamp
= updateStamp
;
873 for (unsigned int i
= 0; i
< count
; i
++) {
874 done
= callback(refcon
, dictionary
[i
].key
.get(), EXT_CAST(dictionary
[i
].value
.get()));
878 if (initialUpdateStamp
!= updateStamp
) {
883 return initialUpdateStamp
== updateStamp
;
887 OSDictionaryIterateObjectsBlock(void * refcon
, const OSSymbol
* key
, OSObject
* object
)
889 bool (^block
)(const OSSymbol
* key
, OSObject
* object
) = (typeof(block
))refcon
;
890 return block(key
, object
);
894 OSDictionary::iterateObjects(bool (^block
)(const OSSymbol
* key
, OSObject
* object
))
896 return iterateObjects((void *)block
, &OSDictionaryIterateObjectsBlock
);