1*4f1223e8SApple OSS Distributions/* 2*4f1223e8SApple OSS Distributions * Copyright (c) 2015 Apple Inc. All rights reserved. 3*4f1223e8SApple OSS Distributions * 4*4f1223e8SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_START@ 5*4f1223e8SApple OSS Distributions * 6*4f1223e8SApple OSS Distributions * This file contains Original Code and/or Modifications of Original Code 7*4f1223e8SApple OSS Distributions * as defined in and that are subject to the Apple Public Source License 8*4f1223e8SApple OSS Distributions * Version 2.0 (the 'License'). You may not use this file except in 9*4f1223e8SApple OSS Distributions * compliance with the License. The rights granted to you under the License 10*4f1223e8SApple OSS Distributions * may not be used to create, or enable the creation or redistribution of, 11*4f1223e8SApple OSS Distributions * unlawful or unlicensed copies of an Apple operating system, or to 12*4f1223e8SApple OSS Distributions * circumvent, violate, or enable the circumvention or violation of, any 13*4f1223e8SApple OSS Distributions * terms of an Apple operating system software license agreement. 14*4f1223e8SApple OSS Distributions * 15*4f1223e8SApple OSS Distributions * Please obtain a copy of the License at 16*4f1223e8SApple OSS Distributions * http://www.opensource.apple.com/apsl/ and read it before using this file. 17*4f1223e8SApple OSS Distributions * 18*4f1223e8SApple OSS Distributions * The Original Code and all software distributed under the License are 19*4f1223e8SApple OSS Distributions * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 20*4f1223e8SApple OSS Distributions * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, 21*4f1223e8SApple OSS Distributions * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, 22*4f1223e8SApple OSS Distributions * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 23*4f1223e8SApple OSS Distributions * Please see the License for the specific language governing rights and 24*4f1223e8SApple OSS Distributions * limitations under the License. 25*4f1223e8SApple OSS Distributions * 26*4f1223e8SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_END@ 27*4f1223e8SApple OSS Distributions */ 28*4f1223e8SApple OSS Distributions 29*4f1223e8SApple OSS Distributions#include <kcdata.h> 30*4f1223e8SApple OSS Distributions#import <Foundation/Foundation.h> 31*4f1223e8SApple OSS Distributions#import "kdd.h" 32*4f1223e8SApple OSS Distributions#import "KCDBasicTypeDescription.h" 33*4f1223e8SApple OSS Distributions#import "KCDStructTypeDescription.h" 34*4f1223e8SApple OSS Distributions#import "KCDEmbeddedBufferDescription.h" 35*4f1223e8SApple OSS Distributions 36*4f1223e8SApple OSS Distributions#define LIB_KCD_ERR_DOMAIN @"KCDataError" 37*4f1223e8SApple OSS Distributions 38*4f1223e8SApple OSS Distributions#define GEN_ERROR(code, msg) gen_error(__LINE__, code, @msg) 39*4f1223e8SApple OSS Distributions#define GEN_ERRORF(code, msg, ...) gen_error(__LINE__, code, [NSString stringWithFormat:@msg, __VA_ARGS__]) 40*4f1223e8SApple OSS Distributions 41*4f1223e8SApple OSS Distributions#define MAX_KCDATATYPE_BUFFER_SIZE 2048 42*4f1223e8SApple OSS Distributionsextern struct kcdata_type_definition * kcdata_get_typedescription(unsigned type_id, uint8_t * buffer, uint32_t buffer_size); 43*4f1223e8SApple OSS Distributions 44*4f1223e8SApple OSS DistributionsBOOL setKCDataTypeForID(uint32_t newTypeID, KCDataType *newTypeObj); 45*4f1223e8SApple OSS Distributions 46*4f1223e8SApple OSS Distributionsstatic NSError * 47*4f1223e8SApple OSS Distributionsgen_error(int line, NSInteger code, NSString *message) 48*4f1223e8SApple OSS Distributions{ 49*4f1223e8SApple OSS Distributions return [NSError errorWithDomain:LIB_KCD_ERR_DOMAIN 50*4f1223e8SApple OSS Distributions code:code 51*4f1223e8SApple OSS Distributions userInfo:@{ @"line": @(line), @"message": message }]; 52*4f1223e8SApple OSS Distributions} 53*4f1223e8SApple OSS Distributions 54*4f1223e8SApple OSS Distributionsstatic BOOL 55*4f1223e8SApple OSS Distributionsmergedict(NSMutableDictionary * container, NSDictionary * object, NSError ** error) 56*4f1223e8SApple OSS Distributions{ 57*4f1223e8SApple OSS Distributions for (id key in object) { 58*4f1223e8SApple OSS Distributions id existing = container[key]; 59*4f1223e8SApple OSS Distributions id new = object[key]; 60*4f1223e8SApple OSS Distributions if (existing) { 61*4f1223e8SApple OSS Distributions if ([existing isKindOfClass:[NSMutableArray class]] && [new isKindOfClass:[ NSArray class ]]) { 62*4f1223e8SApple OSS Distributions [existing addObjectsFromArray:new]; 63*4f1223e8SApple OSS Distributions } else { 64*4f1223e8SApple OSS Distributions if (error) { 65*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "repeated key: %@", key); 66*4f1223e8SApple OSS Distributions } 67*4f1223e8SApple OSS Distributions return FALSE; 68*4f1223e8SApple OSS Distributions } 69*4f1223e8SApple OSS Distributions } else { 70*4f1223e8SApple OSS Distributions [container setValue:new forKey:key]; 71*4f1223e8SApple OSS Distributions } 72*4f1223e8SApple OSS Distributions } 73*4f1223e8SApple OSS Distributions return TRUE; 74*4f1223e8SApple OSS Distributions} 75*4f1223e8SApple OSS Distributions 76*4f1223e8SApple OSS Distributions/*! 77*4f1223e8SApple OSS Distributions * @function getTypeFromTypeDef 78*4f1223e8SApple OSS Distributions * 79*4f1223e8SApple OSS Distributions * @abstract 80*4f1223e8SApple OSS Distributions * Build a KCDataType from a type definition. 81*4f1223e8SApple OSS Distributions * 82*4f1223e8SApple OSS Distributions * @param typeDef 83*4f1223e8SApple OSS Distributions * A pointer to kcdata_type_definition_t that specifies the type fields and has subtype definitions 84*4f1223e8SApple OSS Distributions * in the memory immediately following the type_definition. 85*4f1223e8SApple OSS Distributions * 86*4f1223e8SApple OSS Distributions * @return KCDataType * type object which can be used to parse data into dictionaries. 87*4f1223e8SApple OSS Distributions * This may return nil if it finds the data to be invalid. 88*4f1223e8SApple OSS Distributions * 89*4f1223e8SApple OSS Distributions * @discussion 90*4f1223e8SApple OSS Distributions * This routine tries to decode the typeDef structure and create either a basic type (KCDBasicTypeDescription) 91*4f1223e8SApple OSS Distributions * or a struct type. 92*4f1223e8SApple OSS Distributions */ 93*4f1223e8SApple OSS Distributionsstatic KCDataType * getTypeFromTypeDef(struct kcdata_type_definition * typeDef); 94*4f1223e8SApple OSS Distributions 95*4f1223e8SApple OSS Distributionsstatic KCDataType * 96*4f1223e8SApple OSS DistributionsgetTypeFromTypeDef(struct kcdata_type_definition * typeDef) 97*4f1223e8SApple OSS Distributions{ 98*4f1223e8SApple OSS Distributions if (typeDef == NULL) { 99*4f1223e8SApple OSS Distributions return nil; 100*4f1223e8SApple OSS Distributions } 101*4f1223e8SApple OSS Distributions NSString * kct_name = [NSString stringWithFormat:@"%s", typeDef->kct_name]; 102*4f1223e8SApple OSS Distributions if (typeDef->kct_num_elements == 1 && !(typeDef->kct_elements[0].kcs_flags & KCS_SUBTYPE_FLAGS_STRUCT)) { 103*4f1223e8SApple OSS Distributions KCDBasicTypeDescription * retval = [[KCDBasicTypeDescription alloc] initWithKCTypeDesc:&typeDef->kct_elements[0]]; 104*4f1223e8SApple OSS Distributions return retval; 105*4f1223e8SApple OSS Distributions } else { 106*4f1223e8SApple OSS Distributions KCDStructTypeDescription * retval = 107*4f1223e8SApple OSS Distributions [[KCDStructTypeDescription alloc] initWithType:typeDef->kct_type_identifier withName:kct_name]; 108*4f1223e8SApple OSS Distributions /* need to do work here to get the array of elements setup here */ 109*4f1223e8SApple OSS Distributions KCDBasicTypeDescription * curField = nil; 110*4f1223e8SApple OSS Distributions for (unsigned int i = 0; i < typeDef->kct_num_elements; i++) { 111*4f1223e8SApple OSS Distributions curField = [[KCDBasicTypeDescription alloc] initWithKCTypeDesc:&typeDef->kct_elements[i]]; 112*4f1223e8SApple OSS Distributions [retval addFieldBasicType:curField]; 113*4f1223e8SApple OSS Distributions if (typeDef->kct_elements[i].kcs_flags & KCS_SUBTYPE_FLAGS_MERGE) { 114*4f1223e8SApple OSS Distributions [retval setFlagsRequestedMerge]; 115*4f1223e8SApple OSS Distributions } 116*4f1223e8SApple OSS Distributions } 117*4f1223e8SApple OSS Distributions return retval; 118*4f1223e8SApple OSS Distributions } 119*4f1223e8SApple OSS Distributions return nil; 120*4f1223e8SApple OSS Distributions} 121*4f1223e8SApple OSS Distributions 122*4f1223e8SApple OSS Distributionsstatic dispatch_once_t onceToken; 123*4f1223e8SApple OSS Distributionsstatic NSMutableDictionary * knownTypes = nil; 124*4f1223e8SApple OSS Distributions 125*4f1223e8SApple OSS DistributionsKCDataType * 126*4f1223e8SApple OSS DistributionsgetKCDataTypeForID(uint32_t typeID) 127*4f1223e8SApple OSS Distributions{ 128*4f1223e8SApple OSS Distributions dispatch_once(&onceToken, ^{ 129*4f1223e8SApple OSS Distributions if (!knownTypes) { 130*4f1223e8SApple OSS Distributions knownTypes = [[NSMutableDictionary alloc] init]; 131*4f1223e8SApple OSS Distributions } 132*4f1223e8SApple OSS Distributions }); 133*4f1223e8SApple OSS Distributions 134*4f1223e8SApple OSS Distributions NSNumber * type = [NSNumber numberWithUnsignedInt:typeID]; 135*4f1223e8SApple OSS Distributions if (!knownTypes[type]) { 136*4f1223e8SApple OSS Distributions if (typeID == KCDATA_TYPE_NESTED_KCDATA) { 137*4f1223e8SApple OSS Distributions knownTypes[type] = [[KCDEmbeddedBufferDescription alloc] init]; 138*4f1223e8SApple OSS Distributions return knownTypes[type]; 139*4f1223e8SApple OSS Distributions } 140*4f1223e8SApple OSS Distributions /* code to query system for type information */ 141*4f1223e8SApple OSS Distributions uint8_t buffer[MAX_KCDATATYPE_BUFFER_SIZE]; 142*4f1223e8SApple OSS Distributions struct kcdata_type_definition * sys_def = kcdata_get_typedescription(typeID, buffer, MAX_KCDATATYPE_BUFFER_SIZE); 143*4f1223e8SApple OSS Distributions if (sys_def == NULL) { 144*4f1223e8SApple OSS Distributions knownTypes[type] = [[KCDBasicTypeDescription alloc] createDefaultForType:typeID]; 145*4f1223e8SApple OSS Distributions } else { 146*4f1223e8SApple OSS Distributions knownTypes[type] = getTypeFromTypeDef(sys_def); 147*4f1223e8SApple OSS Distributions } 148*4f1223e8SApple OSS Distributions } 149*4f1223e8SApple OSS Distributions assert(knownTypes[type] != nil); 150*4f1223e8SApple OSS Distributions return knownTypes[type]; 151*4f1223e8SApple OSS Distributions} 152*4f1223e8SApple OSS Distributions 153*4f1223e8SApple OSS DistributionsBOOL 154*4f1223e8SApple OSS DistributionssetKCDataTypeForID(uint32_t newTypeID, KCDataType *newTypeObj) { 155*4f1223e8SApple OSS Distributions if (newTypeObj == NULL || newTypeID == 0) { 156*4f1223e8SApple OSS Distributions return FALSE; 157*4f1223e8SApple OSS Distributions } 158*4f1223e8SApple OSS Distributions 159*4f1223e8SApple OSS Distributions dispatch_once(&onceToken, ^{ 160*4f1223e8SApple OSS Distributions if (!knownTypes) { 161*4f1223e8SApple OSS Distributions knownTypes = [[NSMutableDictionary alloc] init]; 162*4f1223e8SApple OSS Distributions } 163*4f1223e8SApple OSS Distributions }); 164*4f1223e8SApple OSS Distributions 165*4f1223e8SApple OSS Distributions NSNumber * type = [NSNumber numberWithUnsignedInt:newTypeID]; 166*4f1223e8SApple OSS Distributions 167*4f1223e8SApple OSS Distributions if (!knownTypes[type]) { 168*4f1223e8SApple OSS Distributions knownTypes[type] = newTypeObj; 169*4f1223e8SApple OSS Distributions return TRUE; 170*4f1223e8SApple OSS Distributions } 171*4f1223e8SApple OSS Distributions 172*4f1223e8SApple OSS Distributions return FALSE; 173*4f1223e8SApple OSS Distributions} 174*4f1223e8SApple OSS Distributions 175*4f1223e8SApple OSS Distributions 176*4f1223e8SApple OSS DistributionsNSString * 177*4f1223e8SApple OSS DistributionsKCDataTypeNameForID(uint32_t typeID) 178*4f1223e8SApple OSS Distributions{ 179*4f1223e8SApple OSS Distributions NSString * retval = [NSString stringWithFormat:@"%u", typeID]; 180*4f1223e8SApple OSS Distributions KCDataType * t = getKCDataTypeForID(typeID); 181*4f1223e8SApple OSS Distributions 182*4f1223e8SApple OSS Distributions if (![[t name] containsString:@"Type_"]) { 183*4f1223e8SApple OSS Distributions retval = [t name]; 184*4f1223e8SApple OSS Distributions } 185*4f1223e8SApple OSS Distributions return retval; 186*4f1223e8SApple OSS Distributions} 187*4f1223e8SApple OSS Distributions 188*4f1223e8SApple OSS DistributionsNSMutableDictionary * 189*4f1223e8SApple OSS DistributionsparseKCDataArray(kcdata_iter_t iter, NSError **error) 190*4f1223e8SApple OSS Distributions{ 191*4f1223e8SApple OSS Distributions if (!kcdata_iter_array_valid(iter)) { 192*4f1223e8SApple OSS Distributions if (error) 193*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_OBJECT, "invalid array"); 194*4f1223e8SApple OSS Distributions return NULL; 195*4f1223e8SApple OSS Distributions } 196*4f1223e8SApple OSS Distributions 197*4f1223e8SApple OSS Distributions uint32_t typeID = kcdata_iter_array_elem_type(iter); 198*4f1223e8SApple OSS Distributions uint32_t count = kcdata_iter_array_elem_count(iter); 199*4f1223e8SApple OSS Distributions uint32_t size = kcdata_iter_array_elem_size(iter); 200*4f1223e8SApple OSS Distributions uint8_t * buffer = (uint8_t *)kcdata_iter_payload(iter); 201*4f1223e8SApple OSS Distributions KCDataType * datatype = getKCDataTypeForID(typeID); 202*4f1223e8SApple OSS Distributions NSMutableDictionary * retval = [[NSMutableDictionary alloc] initWithCapacity:1]; 203*4f1223e8SApple OSS Distributions NSMutableArray * arr = [[NSMutableArray alloc] initWithCapacity:count]; 204*4f1223e8SApple OSS Distributions retval[[datatype name]] = arr; 205*4f1223e8SApple OSS Distributions NSDictionary * tmpdict = NULL; 206*4f1223e8SApple OSS Distributions for (uint32_t i = 0; i < count; i++) { 207*4f1223e8SApple OSS Distributions tmpdict = [datatype parseData:(void *)&buffer[i * size] ofLength:size]; 208*4f1223e8SApple OSS Distributions if (!tmpdict) { 209*4f1223e8SApple OSS Distributions if (error) 210*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "failed to parse array element. type=0x%x", (int)typeID); 211*4f1223e8SApple OSS Distributions return NULL; 212*4f1223e8SApple OSS Distributions } 213*4f1223e8SApple OSS Distributions if ([datatype shouldMergeData]) { 214*4f1223e8SApple OSS Distributions assert([tmpdict count] == 1); 215*4f1223e8SApple OSS Distributions [arr addObject: [tmpdict allValues][0]]; 216*4f1223e8SApple OSS Distributions } else { 217*4f1223e8SApple OSS Distributions [arr addObject:tmpdict]; 218*4f1223e8SApple OSS Distributions } 219*4f1223e8SApple OSS Distributions } 220*4f1223e8SApple OSS Distributions return retval; 221*4f1223e8SApple OSS Distributions} 222*4f1223e8SApple OSS Distributions 223*4f1223e8SApple OSS DistributionsNSMutableDictionary * 224*4f1223e8SApple OSS DistributionsparseKCDataContainer(kcdata_iter_t *iter_p, NSError **error) 225*4f1223e8SApple OSS Distributions{ 226*4f1223e8SApple OSS Distributions kcdata_iter_t iter = *iter_p; 227*4f1223e8SApple OSS Distributions 228*4f1223e8SApple OSS Distributions if (!kcdata_iter_container_valid(iter)) { 229*4f1223e8SApple OSS Distributions if (error) 230*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_OBJECT, "invalid container"); 231*4f1223e8SApple OSS Distributions return NULL; 232*4f1223e8SApple OSS Distributions } 233*4f1223e8SApple OSS Distributions uint64_t containerID = kcdata_iter_container_id(iter); 234*4f1223e8SApple OSS Distributions 235*4f1223e8SApple OSS Distributions /* setup collection object for sub containers */ 236*4f1223e8SApple OSS Distributions NSMutableDictionary * sub_containers = [[NSMutableDictionary alloc] init]; 237*4f1223e8SApple OSS Distributions NSMutableDictionary * retval = [[NSMutableDictionary alloc] init]; 238*4f1223e8SApple OSS Distributions NSMutableDictionary * container = [[NSMutableDictionary alloc] init]; 239*4f1223e8SApple OSS Distributions 240*4f1223e8SApple OSS Distributions KCDataType * tmptype; 241*4f1223e8SApple OSS Distributions uint32_t _t; 242*4f1223e8SApple OSS Distributions void * _d; 243*4f1223e8SApple OSS Distributions BOOL ok; 244*4f1223e8SApple OSS Distributions NSDictionary * tmpdict; 245*4f1223e8SApple OSS Distributions BOOL found_end = FALSE; 246*4f1223e8SApple OSS Distributions retval[KCDataTypeNameForID(kcdata_iter_container_type(iter))] = container; 247*4f1223e8SApple OSS Distributions 248*4f1223e8SApple OSS Distributions iter = kcdata_iter_next(iter); 249*4f1223e8SApple OSS Distributions 250*4f1223e8SApple OSS Distributions KCDATA_ITER_FOREACH(iter) 251*4f1223e8SApple OSS Distributions { 252*4f1223e8SApple OSS Distributions _t = kcdata_iter_type(iter); 253*4f1223e8SApple OSS Distributions _d = kcdata_iter_payload(iter); 254*4f1223e8SApple OSS Distributions if (_t == KCDATA_TYPE_CONTAINER_END) { 255*4f1223e8SApple OSS Distributions if (kcdata_iter_container_id(iter) != containerID) { 256*4f1223e8SApple OSS Distributions if (error) 257*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_ARGUMENT, "container marker mismatch"); 258*4f1223e8SApple OSS Distributions return NULL; 259*4f1223e8SApple OSS Distributions } 260*4f1223e8SApple OSS Distributions found_end = TRUE; 261*4f1223e8SApple OSS Distributions break; 262*4f1223e8SApple OSS Distributions } 263*4f1223e8SApple OSS Distributions 264*4f1223e8SApple OSS Distributions if (_t == KCDATA_TYPE_ARRAY) { 265*4f1223e8SApple OSS Distributions tmpdict = parseKCDataArray(iter, error); 266*4f1223e8SApple OSS Distributions if (!tmpdict) 267*4f1223e8SApple OSS Distributions return NULL; 268*4f1223e8SApple OSS Distributions 269*4f1223e8SApple OSS Distributions ok = mergedict(container, tmpdict, error); 270*4f1223e8SApple OSS Distributions if (!ok) 271*4f1223e8SApple OSS Distributions return NULL; 272*4f1223e8SApple OSS Distributions 273*4f1223e8SApple OSS Distributions continue; 274*4f1223e8SApple OSS Distributions } 275*4f1223e8SApple OSS Distributions 276*4f1223e8SApple OSS Distributions if (_t == KCDATA_TYPE_CONTAINER_BEGIN) { 277*4f1223e8SApple OSS Distributions NSString * subcontainerID = [NSString stringWithFormat:@"%llu", kcdata_iter_container_id(iter)]; 278*4f1223e8SApple OSS Distributions tmpdict = parseKCDataContainer(&iter, error); 279*4f1223e8SApple OSS Distributions if (!tmpdict) 280*4f1223e8SApple OSS Distributions return NULL; 281*4f1223e8SApple OSS Distributions assert([tmpdict count] == 1); 282*4f1223e8SApple OSS Distributions for (NSString * k in [tmpdict keyEnumerator]) { 283*4f1223e8SApple OSS Distributions if (sub_containers[k] == nil) { 284*4f1223e8SApple OSS Distributions sub_containers[k] = [[NSMutableDictionary alloc] init]; 285*4f1223e8SApple OSS Distributions } 286*4f1223e8SApple OSS Distributions if (sub_containers[k][subcontainerID] != nil) { 287*4f1223e8SApple OSS Distributions if (error) 288*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "repeated container id: %@", subcontainerID); 289*4f1223e8SApple OSS Distributions return NULL; 290*4f1223e8SApple OSS Distributions } 291*4f1223e8SApple OSS Distributions sub_containers[k][subcontainerID] = tmpdict[k]; 292*4f1223e8SApple OSS Distributions } 293*4f1223e8SApple OSS Distributions continue; 294*4f1223e8SApple OSS Distributions } 295*4f1223e8SApple OSS Distributions 296*4f1223e8SApple OSS Distributions tmptype = getKCDataTypeForID(_t); 297*4f1223e8SApple OSS Distributions tmpdict = [tmptype parseData:_d ofLength:kcdata_iter_size(iter)]; 298*4f1223e8SApple OSS Distributions if (!tmpdict) { 299*4f1223e8SApple OSS Distributions if (error) 300*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "failed to parse. type=0x%x", (int)_t); 301*4f1223e8SApple OSS Distributions return NULL; 302*4f1223e8SApple OSS Distributions } 303*4f1223e8SApple OSS Distributions if (![tmptype shouldMergeData]) { 304*4f1223e8SApple OSS Distributions tmpdict = @{[tmptype name] : tmpdict}; 305*4f1223e8SApple OSS Distributions } 306*4f1223e8SApple OSS Distributions ok = mergedict(container, tmpdict, error); 307*4f1223e8SApple OSS Distributions if (!ok) 308*4f1223e8SApple OSS Distributions return NULL; 309*4f1223e8SApple OSS Distributions } 310*4f1223e8SApple OSS Distributions 311*4f1223e8SApple OSS Distributions if (!found_end) { 312*4f1223e8SApple OSS Distributions if (error) 313*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_ARGUMENT, "missing container end"); 314*4f1223e8SApple OSS Distributions return NULL; 315*4f1223e8SApple OSS Distributions } 316*4f1223e8SApple OSS Distributions 317*4f1223e8SApple OSS Distributions ok = mergedict(container, sub_containers, error); 318*4f1223e8SApple OSS Distributions if (!ok) 319*4f1223e8SApple OSS Distributions return NULL; 320*4f1223e8SApple OSS Distributions 321*4f1223e8SApple OSS Distributions *iter_p = iter; 322*4f1223e8SApple OSS Distributions return retval; 323*4f1223e8SApple OSS Distributions} 324*4f1223e8SApple OSS Distributions 325*4f1223e8SApple OSS DistributionsNSDictionary * 326*4f1223e8SApple OSS DistributionsparseKCDataBuffer(void * dataBuffer, uint32_t size, NSError ** error) 327*4f1223e8SApple OSS Distributions{ 328*4f1223e8SApple OSS Distributions if (dataBuffer == NULL) { 329*4f1223e8SApple OSS Distributions if (error) 330*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_ARGUMENT, "buffer is null"); 331*4f1223e8SApple OSS Distributions return NULL; 332*4f1223e8SApple OSS Distributions } 333*4f1223e8SApple OSS Distributions 334*4f1223e8SApple OSS Distributions uint32_t _type = (size >= sizeof(uint32_t)) ? *(uint32_t*)dataBuffer : 0; 335*4f1223e8SApple OSS Distributions uint32_t _size = 0; 336*4f1223e8SApple OSS Distributions uint64_t _flags = 0; 337*4f1223e8SApple OSS Distributions void * _datap = NULL; 338*4f1223e8SApple OSS Distributions KCDataType * kcd_type = NULL; 339*4f1223e8SApple OSS Distributions NSString * rootKey = NULL; 340*4f1223e8SApple OSS Distributions uint32_t rootType = _type; 341*4f1223e8SApple OSS Distributions BOOL ok; 342*4f1223e8SApple OSS Distributions 343*4f1223e8SApple OSS Distributions /* validate begin tag and get root key */ 344*4f1223e8SApple OSS Distributions switch (_type) { 345*4f1223e8SApple OSS Distributions case KCDATA_BUFFER_BEGIN_CRASHINFO: 346*4f1223e8SApple OSS Distributions rootKey = @"kcdata_crashinfo"; 347*4f1223e8SApple OSS Distributions break; 348*4f1223e8SApple OSS Distributions case KCDATA_BUFFER_BEGIN_STACKSHOT: 349*4f1223e8SApple OSS Distributions rootKey = @"kcdata_stackshot"; 350*4f1223e8SApple OSS Distributions break; 351*4f1223e8SApple OSS Distributions case KCDATA_BUFFER_BEGIN_DELTA_STACKSHOT: 352*4f1223e8SApple OSS Distributions rootKey = @"kcdata_delta_stackshot"; 353*4f1223e8SApple OSS Distributions break; 354*4f1223e8SApple OSS Distributions case KCDATA_BUFFER_BEGIN_OS_REASON: 355*4f1223e8SApple OSS Distributions rootKey = @"kcdata_reason"; 356*4f1223e8SApple OSS Distributions break; 357*4f1223e8SApple OSS Distributions case KCDATA_BUFFER_BEGIN_XNUPOST_CONFIG: 358*4f1223e8SApple OSS Distributions rootKey = @"xnupost_testconfig"; 359*4f1223e8SApple OSS Distributions break; 360*4f1223e8SApple OSS Distributions default: { 361*4f1223e8SApple OSS Distributions if (error) 362*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_VALUE, "invalid magic number"); 363*4f1223e8SApple OSS Distributions return NULL; 364*4f1223e8SApple OSS Distributions break; 365*4f1223e8SApple OSS Distributions } 366*4f1223e8SApple OSS Distributions } 367*4f1223e8SApple OSS Distributions assert(rootKey != NULL); 368*4f1223e8SApple OSS Distributions 369*4f1223e8SApple OSS Distributions kcdata_iter_t iter = kcdata_iter(dataBuffer, size); 370*4f1223e8SApple OSS Distributions 371*4f1223e8SApple OSS Distributions if (!kcdata_iter_valid(iter)) { 372*4f1223e8SApple OSS Distributions if (error) { 373*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_OBJECT, "initial item is invalid"); 374*4f1223e8SApple OSS Distributions } 375*4f1223e8SApple OSS Distributions return NULL; 376*4f1223e8SApple OSS Distributions } 377*4f1223e8SApple OSS Distributions 378*4f1223e8SApple OSS Distributions NSMutableDictionary * rootObject = [NSMutableDictionary dictionary]; 379*4f1223e8SApple OSS Distributions NSDictionary * retval = [NSMutableDictionary dictionaryWithObject:rootObject forKey:rootKey]; 380*4f1223e8SApple OSS Distributions 381*4f1223e8SApple OSS Distributions /* iterate over each kcdata item */ 382*4f1223e8SApple OSS Distributions KCDATA_ITER_FOREACH(iter) 383*4f1223e8SApple OSS Distributions { 384*4f1223e8SApple OSS Distributions _type = kcdata_iter_type(iter); 385*4f1223e8SApple OSS Distributions _size = kcdata_iter_size(iter); 386*4f1223e8SApple OSS Distributions _flags = kcdata_iter_flags(iter); 387*4f1223e8SApple OSS Distributions _datap = kcdata_iter_payload(iter); 388*4f1223e8SApple OSS Distributions 389*4f1223e8SApple OSS Distributions if (_type == rootType) 390*4f1223e8SApple OSS Distributions continue; 391*4f1223e8SApple OSS Distributions 392*4f1223e8SApple OSS Distributions if (_type == KCDATA_TYPE_ARRAY) { 393*4f1223e8SApple OSS Distributions NSDictionary * dict = parseKCDataArray(iter, error); 394*4f1223e8SApple OSS Distributions if (!dict) 395*4f1223e8SApple OSS Distributions return nil; 396*4f1223e8SApple OSS Distributions 397*4f1223e8SApple OSS Distributions ok = mergedict(rootObject, dict, error); 398*4f1223e8SApple OSS Distributions if (!ok) 399*4f1223e8SApple OSS Distributions return NULL; 400*4f1223e8SApple OSS Distributions 401*4f1223e8SApple OSS Distributions continue; 402*4f1223e8SApple OSS Distributions } 403*4f1223e8SApple OSS Distributions 404*4f1223e8SApple OSS Distributions if (_type == KCDATA_TYPE_CONTAINER_BEGIN) { 405*4f1223e8SApple OSS Distributions NSString * containerID = [NSString stringWithFormat:@"%llu", kcdata_iter_container_id(iter)]; 406*4f1223e8SApple OSS Distributions NSMutableDictionary *container = parseKCDataContainer(&iter, error); 407*4f1223e8SApple OSS Distributions if (!container) 408*4f1223e8SApple OSS Distributions return nil; 409*4f1223e8SApple OSS Distributions assert([container count] == 1); 410*4f1223e8SApple OSS Distributions for (NSString * k in [container keyEnumerator]) { 411*4f1223e8SApple OSS Distributions if (rootObject[k] == nil) { 412*4f1223e8SApple OSS Distributions rootObject[k] = [[NSMutableDictionary alloc] init]; 413*4f1223e8SApple OSS Distributions } 414*4f1223e8SApple OSS Distributions if (rootObject[k][containerID] != nil) { 415*4f1223e8SApple OSS Distributions if (error) 416*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "repeated container id: %@", containerID); 417*4f1223e8SApple OSS Distributions return NULL; 418*4f1223e8SApple OSS Distributions } 419*4f1223e8SApple OSS Distributions rootObject[k][containerID] = container[k]; 420*4f1223e8SApple OSS Distributions } 421*4f1223e8SApple OSS Distributions continue; 422*4f1223e8SApple OSS Distributions } 423*4f1223e8SApple OSS Distributions 424*4f1223e8SApple OSS Distributions if (_type == KCDATA_TYPE_TYPEDEFINTION) { 425*4f1223e8SApple OSS Distributions KCDataType *new_type = getTypeFromTypeDef((struct kcdata_type_definition *)_datap); 426*4f1223e8SApple OSS Distributions if (new_type != NULL) { 427*4f1223e8SApple OSS Distributions setKCDataTypeForID([new_type typeID], new_type); 428*4f1223e8SApple OSS Distributions kcd_type = getKCDataTypeForID(_type); 429*4f1223e8SApple OSS Distributions NSDictionary * tmpdict = [kcd_type parseData:_datap ofLength:_size]; 430*4f1223e8SApple OSS Distributions if (!tmpdict) { 431*4f1223e8SApple OSS Distributions if (error) 432*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "failed to parse. type=0x%x", (int)_type); 433*4f1223e8SApple OSS Distributions return NULL; 434*4f1223e8SApple OSS Distributions } 435*4f1223e8SApple OSS Distributions NSString *k = [NSString stringWithFormat:@"typedef[%@]", [new_type name]]; 436*4f1223e8SApple OSS Distributions rootObject[k] = tmpdict; 437*4f1223e8SApple OSS Distributions }else { 438*4f1223e8SApple OSS Distributions if (error) 439*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "Failed to parse type definition for type %u", _type); 440*4f1223e8SApple OSS Distributions return NULL; 441*4f1223e8SApple OSS Distributions } 442*4f1223e8SApple OSS Distributions continue; 443*4f1223e8SApple OSS Distributions } 444*4f1223e8SApple OSS Distributions 445*4f1223e8SApple OSS Distributions kcd_type = getKCDataTypeForID(_type); 446*4f1223e8SApple OSS Distributions NSDictionary * tmpdict = [kcd_type parseData:_datap ofLength:_size]; 447*4f1223e8SApple OSS Distributions if (!tmpdict) { 448*4f1223e8SApple OSS Distributions if (error) 449*4f1223e8SApple OSS Distributions *error = GEN_ERRORF(KERN_INVALID_OBJECT, "failed to parse. type=0x%x", (int)_type); 450*4f1223e8SApple OSS Distributions return NULL; 451*4f1223e8SApple OSS Distributions } 452*4f1223e8SApple OSS Distributions if (![kcd_type shouldMergeData]) { 453*4f1223e8SApple OSS Distributions tmpdict = @{[kcd_type name] : tmpdict}; 454*4f1223e8SApple OSS Distributions } 455*4f1223e8SApple OSS Distributions ok = mergedict(rootObject, tmpdict, error); 456*4f1223e8SApple OSS Distributions if (!ok) 457*4f1223e8SApple OSS Distributions return NULL; 458*4f1223e8SApple OSS Distributions } 459*4f1223e8SApple OSS Distributions 460*4f1223e8SApple OSS Distributions if (KCDATA_ITER_FOREACH_FAILED(iter)) { 461*4f1223e8SApple OSS Distributions retval = nil; 462*4f1223e8SApple OSS Distributions if (error) { 463*4f1223e8SApple OSS Distributions *error = GEN_ERROR(KERN_INVALID_OBJECT, "invalid item or missing buffer end marker"); 464*4f1223e8SApple OSS Distributions } 465*4f1223e8SApple OSS Distributions } 466*4f1223e8SApple OSS Distributions 467*4f1223e8SApple OSS Distributions return retval; 468*4f1223e8SApple OSS Distributions} 469