2 * Copyright (c) 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@
30 #include <libkern/c++/OSSharedPtr.h>
31 #include <libkern/OSSerializeBinary.h>
32 #include <libkern/c++/OSContainers.h>
33 #include <libkern/c++/OSLib.h>
34 #include <libkern/c++/OSDictionary.h>
35 #include <libkern/OSSerializeBinary.h>
36 #include <libkern/c++/OSSharedPtr.h>
38 #include <IOKit/IOLib.h>
40 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
43 #define DEBG(fmt, args ...) { kprintf(fmt, args); }
45 #define DEBG(fmt, args ...) {}
48 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
51 OSSerialize::binaryWithCapacity(unsigned int inCapacity
,
52 Editor editor
, void * reference
)
56 if (inCapacity
< sizeof(uint32_t)) {
59 me
= OSSerialize::withCapacity(inCapacity
);
65 me
->endCollection
= true;
67 me
->editRef
= reference
;
69 bcopy(kOSSerializeBinarySignature
, &me
->data
[0], sizeof(kOSSerializeBinarySignature
));
70 me
->length
= sizeof(kOSSerializeBinarySignature
);
76 OSSerialize::addBinary(const void * bits
, size_t size
)
78 unsigned int newCapacity
;
81 if (os_add_overflow(size
, 3, &alignSize
)) {
85 if (os_add_overflow(length
, alignSize
, &newCapacity
)) {
88 if (newCapacity
>= capacity
) {
89 newCapacity
= (((newCapacity
- 1) / capacityIncrement
) + 1) * capacityIncrement
;
90 if (newCapacity
< capacity
) {
93 if (newCapacity
> ensureCapacity(newCapacity
)) {
98 bcopy(bits
, &data
[length
], size
);
105 OSSerialize::setIndexed(bool index __unused
)
107 assert(index
&& !indexData
);
108 indexData
= OSData::withCapacity(256);
113 OSSerialize::addBinaryObject(const OSMetaClassBase
* o
, uint32_t key
,
114 const void * bits
, uint32_t size
,
115 uint32_t * startCollection
)
117 unsigned int newCapacity
;
124 headerSize
= sizeof(key
);
126 uint32_t offset
= length
;
127 if (startCollection
) {
128 *startCollection
= offset
;
129 headerSize
+= sizeof(uint32_t);
131 offset
/= sizeof(uint32_t);
132 indexData
->appendBytes(&offset
, sizeof(offset
));
135 if (os_add3_overflow(size
, headerSize
, 3, &alignSize
)) {
139 if (os_add_overflow(length
, alignSize
, &newCapacity
)) {
142 if (newCapacity
>= capacity
) {
143 newCapacity
= (((newCapacity
- 1) / capacityIncrement
) + 1) * capacityIncrement
;
144 if (newCapacity
< capacity
) {
147 if (newCapacity
> ensureCapacity(newCapacity
)) {
153 endCollection
= false;
154 key
|= kOSSerializeEndCollecton
;
157 bcopy(&key
, &data
[length
], sizeof(key
));
158 bcopy(bits
, &data
[length
+ headerSize
], size
);
165 OSSerialize::endBinaryCollection(uint32_t startCollection
)
173 assert(length
> startCollection
);
174 if (length
<= startCollection
) {
178 clength
= length
- startCollection
;
179 assert(!(clength
& 3));
180 clength
/= sizeof(uint32_t);
182 memcpy(&data
[startCollection
+ sizeof(uint32_t)], &clength
, sizeof(clength
));
186 OSSerialize::binarySerialize(const OSMetaClassBase
*o
)
191 ok
= binarySerializeInternal(o
);
197 header
= indexData
->getLength() / sizeof(uint32_t);
198 assert(header
<= kOSSerializeDataMask
);
200 header
|= kOSSerializeIndexedBinarySignature
;
202 memcpy(&data
[0], &header
, sizeof(header
));
209 OSSerialize::binarySerializeInternal(const OSMetaClassBase
*o
)
221 uint32_t i
, key
, startCollection
;
225 tagIdx
= tags
->getNextIndexOfObject(o
, 0);
229 assert(indexData
->getLength() > (tagIdx
* sizeof(uint32_t)));
230 tagIdx
= ((const uint32_t *)indexData
->getBytesNoCopy())[tagIdx
];
231 assert(tagIdx
<= kOSSerializeDataMask
);
233 key
= (kOSSerializeObject
| tagIdx
);
235 endCollection
= false;
236 key
|= kOSSerializeEndCollecton
;
238 ok
= addBinary(&key
, sizeof(key
));
242 if ((dict
= OSDynamicCast(OSDictionary
, o
))) {
243 key
= (kOSSerializeDictionary
| dict
->count
);
244 ok
= addBinaryObject(o
, key
, NULL
, 0, &startCollection
);
245 for (i
= 0; ok
&& (i
< dict
->count
);) {
246 const OSSymbol
* dictKey
;
247 const OSMetaClassBase
* dictValue
;
248 const OSMetaClassBase
* nvalue
= NULL
;
250 dictKey
= dict
->dictionary
[i
].key
;
251 dictValue
= dict
->dictionary
[i
].value
;
254 dictValue
= nvalue
= (*editor
)(editRef
, this, dict
, dictKey
, dictValue
);
259 ok
= binarySerialize(dictKey
);
263 endCollection
= (i
== dict
->count
);
264 ok
= binarySerialize(dictValue
);
266 ok
= dictValue
->serialize(this);
271 // if (!ok) ok = binarySerialize(kOSBooleanFalse);
273 endBinaryCollection(startCollection
);
274 } else if ((array
= OSDynamicCast(OSArray
, o
))) {
275 key
= (kOSSerializeArray
| array
->count
);
276 ok
= addBinaryObject(o
, key
, NULL
, 0, &startCollection
);
277 for (i
= 0; ok
&& (i
< array
->count
);) {
279 endCollection
= (i
== array
->count
);
280 ok
= binarySerialize(array
->array
[i
- 1]);
282 ok
= array
->array
[i
- 1]->serialize(this);
284 // if (!ok) ok = binarySerialize(kOSBooleanFalse);
286 endBinaryCollection(startCollection
);
287 } else if ((set
= OSDynamicCast(OSSet
, o
))) {
288 key
= (kOSSerializeSet
| set
->members
->count
);
289 ok
= addBinaryObject(o
, key
, NULL
, 0, &startCollection
);
290 for (i
= 0; ok
&& (i
< set
->members
->count
);) {
292 endCollection
= (i
== set
->members
->count
);
293 ok
= binarySerialize(set
->members
->array
[i
- 1]);
295 ok
= set
->members
->array
[i
- 1]->serialize(this);
297 // if (!ok) ok = binarySerialize(kOSBooleanFalse);
299 endBinaryCollection(startCollection
);
300 } else if ((num
= OSDynamicCast(OSNumber
, o
))) {
301 key
= (kOSSerializeNumber
| num
->size
);
302 ok
= addBinaryObject(o
, key
, &num
->value
, sizeof(num
->value
), NULL
);
303 } else if ((boo
= OSDynamicCast(OSBoolean
, o
))) {
304 key
= (kOSSerializeBoolean
| (kOSBooleanTrue
== boo
));
305 ok
= addBinaryObject(o
, key
, NULL
, 0, NULL
);
306 } else if ((sym
= OSDynamicCast(OSSymbol
, o
))) {
307 len
= (sym
->getLength() + 1);
308 key
= (kOSSerializeSymbol
| len
);
309 ok
= addBinaryObject(o
, key
, sym
->getCStringNoCopy(), len
, NULL
);
310 } else if ((str
= OSDynamicCast(OSString
, o
))) {
311 len
= (str
->getLength() + ((indexData
!= NULL
) ? 1 : 0));
312 key
= (kOSSerializeString
| len
);
313 ok
= addBinaryObject(o
, key
, str
->getCStringNoCopy(), len
, NULL
);
314 } else if ((ldata
= OSDynamicCast(OSData
, o
))) {
315 len
= ldata
->getLength();
316 if (ldata
->reserved
&& ldata
->reserved
->disableSerialization
) {
319 key
= (kOSSerializeData
| len
);
320 ok
= addBinaryObject(o
, key
, ldata
->getBytesNoCopy(), len
, NULL
);
328 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
330 #define setAtIndex(v, idx, o) \
331 if (idx >= v##Capacity) \
333 if (v##Capacity >= v##CapacityMax) ok = false; \
336 uint32_t ncap = v##Capacity + 64; \
337 typeof(v##Array) nbuf = (typeof(v##Array)) kalloc_container(ncap * sizeof(o)); \
338 if (!nbuf) ok = false; \
343 bcopy(v##Array, nbuf, v##Capacity * sizeof(o)); \
344 kfree(v##Array, v##Capacity * sizeof(o)); \
347 v##Capacity = ncap; \
351 if (ok) v##Array[idx] = o;
353 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
356 OSUnserializeBinary(const char *buffer
, size_t bufferSize
, OSString
**errorString
)
358 OSObject
** objsArray
;
359 uint32_t objsCapacity
;
360 enum { objsCapacityMax
= 16 * 1024 * 1024 };
363 OSObject
** stackArray
;
364 uint32_t stackCapacity
;
365 enum { stackCapacityMax
= 64 };
373 OSDictionary
* newDict
;
381 const uint32_t * next
;
382 uint32_t key
, len
, wordLen
, length
;
383 bool end
, newCollect
, isRef
;
384 unsigned long long value
;
385 bool ok
, indexed
, hasLength
;
392 if (bufferSize
< sizeof(kOSSerializeBinarySignature
)) {
395 if (kOSSerializeIndexedBinarySignature
== (((const uint8_t *) buffer
)[0])) {
397 } else if (0 != strcmp(kOSSerializeBinarySignature
, buffer
)) {
400 if (3 & ((uintptr_t) buffer
)) {
404 bufferPos
= sizeof(kOSSerializeBinarySignature
);
405 next
= (typeof(next
))(((uintptr_t) buffer
) + bufferPos
);
407 DEBG("---------OSUnserializeBinary(%p)\n", buffer
);
409 objsArray
= stackArray
= NULL
;
410 objsIdx
= objsCapacity
= 0;
411 stackIdx
= stackCapacity
= 0;
422 bufferPos
+= sizeof(*next
);
423 if (!(ok
= (bufferPos
<= bufferSize
))) {
429 len
= (key
& kOSSerializeDataMask
);
430 wordLen
= (len
+ 3) >> 2;
431 end
= (0 != (kOSSerializeEndCollecton
& key
));
432 DEBG("key 0x%08x: 0x%04x, %d\n", key
, len
, end
);
434 newCollect
= isRef
= hasLength
= false;
435 o
= NULL
; newDict
= NULL
; newArray
= NULL
; newSet
= NULL
;
437 switch (kOSSerializeTypeMask
& key
) {
438 case kOSSerializeDictionary
:
439 o
= newDict
= OSDictionary::withCapacity(len
);
440 newCollect
= (len
!= 0);
443 case kOSSerializeArray
:
444 o
= newArray
= OSArray::withCapacity(len
);
445 newCollect
= (len
!= 0);
448 case kOSSerializeSet
:
449 o
= newSet
= OSSet::withCapacity(len
);
450 newCollect
= (len
!= 0);
454 case kOSSerializeObject
:
455 if (len
>= objsIdx
) {
462 case kOSSerializeNumber
:
463 bufferPos
+= sizeof(long long);
464 if (bufferPos
> bufferSize
) {
467 if ((len
!= 32) && (len
!= 64) && (len
!= 16) && (len
!= 8)) {
473 o
= OSNumber::withNumber(value
, len
);
477 case kOSSerializeSymbol
:
478 bufferPos
+= (wordLen
* sizeof(uint32_t));
479 if (bufferPos
> bufferSize
) {
485 if (0 != ((const char *)next
)[len
- 1]) {
488 o
= (OSObject
*) OSSymbol::withCString((const char *) next
);
492 case kOSSerializeString
:
493 bufferPos
+= (wordLen
* sizeof(uint32_t));
494 if (bufferPos
> bufferSize
) {
497 o
= OSString::withStringOfLength((const char *) next
, len
);
501 case kOSSerializeData
:
502 bufferPos
+= (wordLen
* sizeof(uint32_t));
503 if (bufferPos
> bufferSize
) {
506 o
= OSData::withBytes(next
, len
);
510 case kOSSerializeBoolean
:
511 o
= (len
? kOSBooleanTrue
: kOSBooleanFalse
);
518 if (!(ok
= (o
!= NULL
))) {
523 bufferPos
+= sizeof(*next
);
524 if (!(ok
= (bufferPos
<= bufferSize
))) {
532 setAtIndex(objs
, objsIdx
, o
);
542 sym
= (OSSymbol
*) o
;
545 sym
= OSDynamicCast(OSSymbol
, sym
);
546 if (!sym
&& (str
= OSDynamicCast(OSString
, str
))) {
547 sym
= const_cast<OSSymbol
*>(OSSymbol::withString(str
));
553 DEBG("%s = %s\n", sym
->getCStringNoCopy(), o
->getMetaClass()->getClassName());
555 ok
= dict
->setObject(sym
, o
);
557 if (sym
&& (sym
!= str
)) {
563 ok
= array
->setObject(o
);
565 ok
= set
->setObject(o
);
582 setAtIndex(stack
, stackIdx
, parent
);
586 DEBG("++stack[%d] %p\n", stackIdx
, parent
);
596 parent
= stackArray
[stackIdx
];
597 DEBG("--stack[%d] %p\n", stackIdx
, parent
);
609 if (!(dict
= OSDynamicCast(OSDictionary
, parent
))) {
610 if (!(array
= OSDynamicCast(OSArray
, parent
))) {
611 ok
= (NULL
!= (set
= OSDynamicCast(OSSet
, parent
)));
616 DEBG("ret %p\n", result
);
623 for (len
= (result
!= NULL
); len
< objsIdx
; len
++) {
624 objsArray
[len
]->release();
626 kfree(objsArray
, objsCapacity
* sizeof(*objsArray
));
629 kfree(stackArray
, stackCapacity
* sizeof(*stackArray
));
638 OSSharedPtr
<OSString
>& errorString
)
640 OSString
* errorStringRaw
= NULL
;
641 OSObject
* result
= OSUnserializeXML(buffer
, &errorStringRaw
);
642 errorString
.reset(errorStringRaw
, OSNoRetain
);
650 OSSharedPtr
<OSString
> &errorString
)
652 OSString
* errorStringRaw
= NULL
;
653 OSObject
* result
= OSUnserializeXML(buffer
, bufferSize
, &errorStringRaw
);
654 errorString
.reset(errorStringRaw
, OSNoRetain
);
659 OSUnserializeBinary(const char *buffer
, size_t bufferSize
, OSSharedPtr
<OSString
>& errorString
)
661 OSString
* errorStringRaw
= NULL
;
662 OSObject
* result
= OSUnserializeBinary(buffer
, bufferSize
, &errorStringRaw
);
663 errorString
.reset(errorStringRaw
, OSNoRetain
);
668 OSUnserialize(const char *buffer
, OSSharedPtr
<OSString
>& errorString
)
670 OSString
* errorStringRaw
= NULL
;
671 OSObject
* result
= OSUnserialize(buffer
, &errorStringRaw
);
672 errorString
.reset(errorStringRaw
, OSNoRetain
);