1*19c3b8c2SApple OSS Distributions/* 2*19c3b8c2SApple OSS Distributions * Copyright (c) 2015 Apple Inc. All rights reserved. 3*19c3b8c2SApple OSS Distributions * 4*19c3b8c2SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_START@ 5*19c3b8c2SApple OSS Distributions * 6*19c3b8c2SApple OSS Distributions * This file contains Original Code and/or Modifications of Original Code 7*19c3b8c2SApple OSS Distributions * as defined in and that are subject to the Apple Public Source License 8*19c3b8c2SApple OSS Distributions * Version 2.0 (the 'License'). You may not use this file except in 9*19c3b8c2SApple OSS Distributions * compliance with the License. The rights granted to you under the License 10*19c3b8c2SApple OSS Distributions * may not be used to create, or enable the creation or redistribution of, 11*19c3b8c2SApple OSS Distributions * unlawful or unlicensed copies of an Apple operating system, or to 12*19c3b8c2SApple OSS Distributions * circumvent, violate, or enable the circumvention or violation of, any 13*19c3b8c2SApple OSS Distributions * terms of an Apple operating system software license agreement. 14*19c3b8c2SApple OSS Distributions * 15*19c3b8c2SApple OSS Distributions * Please obtain a copy of the License at 16*19c3b8c2SApple OSS Distributions * http://www.opensource.apple.com/apsl/ and read it before using this file. 17*19c3b8c2SApple OSS Distributions * 18*19c3b8c2SApple OSS Distributions * The Original Code and all software distributed under the License are 19*19c3b8c2SApple OSS Distributions * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 20*19c3b8c2SApple OSS Distributions * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, 21*19c3b8c2SApple OSS Distributions * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, 22*19c3b8c2SApple OSS Distributions * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 23*19c3b8c2SApple OSS Distributions * Please see the License for the specific language governing rights and 24*19c3b8c2SApple OSS Distributions * limitations under the License. 25*19c3b8c2SApple OSS Distributions * 26*19c3b8c2SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_END@ 27*19c3b8c2SApple OSS Distributions */ 28*19c3b8c2SApple OSS Distributions 29*19c3b8c2SApple OSS Distributions#import "KCDBasicTypeDescription.h" 30*19c3b8c2SApple OSS Distributions 31*19c3b8c2SApple OSS Distributionsconst char * name_for_subtype(uint8_t elem_type); 32*19c3b8c2SApple OSS Distributions 33*19c3b8c2SApple OSS Distributionsconst char * name_for_subtype(uint8_t elem_type) 34*19c3b8c2SApple OSS Distributions{ 35*19c3b8c2SApple OSS Distributions char * retval = "unknown"; 36*19c3b8c2SApple OSS Distributions 37*19c3b8c2SApple OSS Distributions switch (elem_type) { 38*19c3b8c2SApple OSS Distributions case KC_ST_CHAR: retval = "char"; break; 39*19c3b8c2SApple OSS Distributions case KC_ST_INT8: retval = "int8_t"; break; 40*19c3b8c2SApple OSS Distributions case KC_ST_UINT8: retval = "uint8_t"; break; 41*19c3b8c2SApple OSS Distributions case KC_ST_INT16: retval = "int16_t"; break; 42*19c3b8c2SApple OSS Distributions case KC_ST_UINT16: retval = "uint16_t"; break; 43*19c3b8c2SApple OSS Distributions case KC_ST_INT32: retval = "int32_t"; break; 44*19c3b8c2SApple OSS Distributions case KC_ST_UINT32: retval = "uint32_t"; break; 45*19c3b8c2SApple OSS Distributions case KC_ST_INT64: retval = "int64_t"; break; 46*19c3b8c2SApple OSS Distributions case KC_ST_UINT64: retval = "uint64_t"; break; 47*19c3b8c2SApple OSS Distributions 48*19c3b8c2SApple OSS Distributions default: retval = "Unknown"; break; 49*19c3b8c2SApple OSS Distributions } 50*19c3b8c2SApple OSS Distributions 51*19c3b8c2SApple OSS Distributions return retval; 52*19c3b8c2SApple OSS Distributions} 53*19c3b8c2SApple OSS Distributions 54*19c3b8c2SApple OSS Distributions 55*19c3b8c2SApple OSS Distributions@interface 56*19c3b8c2SApple OSS DistributionsKCDBasicTypeDescription () { 57*19c3b8c2SApple OSS Distributions unsigned int _typeID; 58*19c3b8c2SApple OSS Distributions uint32_t _size; 59*19c3b8c2SApple OSS Distributions uint32_t _count; 60*19c3b8c2SApple OSS Distributions NSString * _name; 61*19c3b8c2SApple OSS Distributions struct kcdata_subtype_descriptor _subtype_desc; 62*19c3b8c2SApple OSS Distributions} 63*19c3b8c2SApple OSS Distributions 64*19c3b8c2SApple OSS Distributions@end 65*19c3b8c2SApple OSS Distributions 66*19c3b8c2SApple OSS Distributions@implementation KCDBasicTypeDescription 67*19c3b8c2SApple OSS Distributions 68*19c3b8c2SApple OSS Distributions- (id)initWithKCTypeDesc:(kcdata_subtype_descriptor_t)sub_type_desc 69*19c3b8c2SApple OSS Distributions{ 70*19c3b8c2SApple OSS Distributions _typeID = sub_type_desc->kcs_elem_type; 71*19c3b8c2SApple OSS Distributions _count = kcs_get_elem_count(sub_type_desc); 72*19c3b8c2SApple OSS Distributions _size = kcs_get_elem_size(sub_type_desc); 73*19c3b8c2SApple OSS Distributions 74*19c3b8c2SApple OSS Distributions memcpy(&_subtype_desc, sub_type_desc, sizeof(_subtype_desc)); 75*19c3b8c2SApple OSS Distributions _name = [NSString stringWithFormat:@"%s", _subtype_desc.kcs_name]; 76*19c3b8c2SApple OSS Distributions 77*19c3b8c2SApple OSS Distributions return self; 78*19c3b8c2SApple OSS Distributions} 79*19c3b8c2SApple OSS Distributions 80*19c3b8c2SApple OSS Distributions- (id)createDefaultForType:(uint32_t)typeID 81*19c3b8c2SApple OSS Distributions{ 82*19c3b8c2SApple OSS Distributions struct kcdata_subtype_descriptor subtype; 83*19c3b8c2SApple OSS Distributions subtype.kcs_flags = KCS_SUBTYPE_FLAGS_ARRAY; 84*19c3b8c2SApple OSS Distributions subtype.kcs_elem_type = KC_ST_UINT8; 85*19c3b8c2SApple OSS Distributions subtype.kcs_elem_offset = 0; 86*19c3b8c2SApple OSS Distributions subtype.kcs_elem_size = KCS_SUBTYPE_PACK_SIZE(UINT16_MAX, (uint16_t)sizeof(uint8_t)); 87*19c3b8c2SApple OSS Distributions subtype.kcs_name[0] = '\0'; 88*19c3b8c2SApple OSS Distributions (void)[self initWithKCTypeDesc:&subtype]; 89*19c3b8c2SApple OSS Distributions _name = [NSString stringWithFormat:@"Type_0x%x", typeID]; 90*19c3b8c2SApple OSS Distributions return self; 91*19c3b8c2SApple OSS Distributions} 92*19c3b8c2SApple OSS Distributions 93*19c3b8c2SApple OSS Distributions#define read_unaligned(type, data) ({ \ 94*19c3b8c2SApple OSS Distributions type x; \ 95*19c3b8c2SApple OSS Distributions memcpy((void*)&x, (void*)(data), sizeof(type)); \ 96*19c3b8c2SApple OSS Distributions x; }) 97*19c3b8c2SApple OSS Distributions 98*19c3b8c2SApple OSS Distributions- (NSObject *)objectForType:(kctype_subtype_t)elem_type withData:(uint8_t *)data 99*19c3b8c2SApple OSS Distributions{ 100*19c3b8c2SApple OSS Distributions NSObject * obj; 101*19c3b8c2SApple OSS Distributions 102*19c3b8c2SApple OSS Distributions switch (elem_type) { 103*19c3b8c2SApple OSS Distributions case KC_ST_CHAR: obj = [NSString stringWithFormat:@"%c", *(char *)data]; break; 104*19c3b8c2SApple OSS Distributions case KC_ST_INT8: obj = [NSNumber numberWithInt:read_unaligned(int8_t, data)]; break; 105*19c3b8c2SApple OSS Distributions case KC_ST_UINT8: obj = [NSNumber numberWithInt:read_unaligned(uint8_t, data)]; break; 106*19c3b8c2SApple OSS Distributions case KC_ST_INT16: obj = [NSNumber numberWithShort:read_unaligned(int16_t, data)]; break; 107*19c3b8c2SApple OSS Distributions case KC_ST_UINT16: obj = [NSNumber numberWithUnsignedShort:read_unaligned(uint16_t, data)]; break; 108*19c3b8c2SApple OSS Distributions case KC_ST_INT32: obj = [NSNumber numberWithInt:read_unaligned(int32_t, data)]; break; 109*19c3b8c2SApple OSS Distributions case KC_ST_UINT32: obj = [NSNumber numberWithUnsignedInt:read_unaligned(uint32_t, data)]; break; 110*19c3b8c2SApple OSS Distributions case KC_ST_INT64: obj = [NSNumber numberWithLongLong:read_unaligned(int64_t, data)]; break; 111*19c3b8c2SApple OSS Distributions case KC_ST_UINT64: obj = [NSNumber numberWithUnsignedLongLong:read_unaligned(uint64_t, data)]; break; 112*19c3b8c2SApple OSS Distributions 113*19c3b8c2SApple OSS Distributions default: obj = @"<Unknown error occurred>"; break; 114*19c3b8c2SApple OSS Distributions } 115*19c3b8c2SApple OSS Distributions 116*19c3b8c2SApple OSS Distributions return obj; 117*19c3b8c2SApple OSS Distributions} 118*19c3b8c2SApple OSS Distributions 119*19c3b8c2SApple OSS Distributions- (NSDictionary *)parseData:(void *)dataBuffer ofLength:(uint32_t)length 120*19c3b8c2SApple OSS Distributions{ 121*19c3b8c2SApple OSS Distributions NSMutableDictionary * retval = [[NSMutableDictionary alloc] init]; 122*19c3b8c2SApple OSS Distributions if (length <= _subtype_desc.kcs_elem_offset) 123*19c3b8c2SApple OSS Distributions return retval; 124*19c3b8c2SApple OSS Distributions uint8_t * data = (uint8_t *)dataBuffer; 125*19c3b8c2SApple OSS Distributions /* 126*19c3b8c2SApple OSS Distributions * Calculate the maximum number of data elements we can parse, Taking into 127*19c3b8c2SApple OSS Distributions * account the maximum size specified by the type description, and also the 128*19c3b8c2SApple OSS Distributions * actual length of the data buffer and the offset into the buffer where we 129*19c3b8c2SApple OSS Distributions * begin parsing. 130*19c3b8c2SApple OSS Distributions */ 131*19c3b8c2SApple OSS Distributions uint32_t elem_count = MIN(_count, (length - _subtype_desc.kcs_elem_offset) / (_size / _count)); 132*19c3b8c2SApple OSS Distributions uint32_t elem_size = _size / _count; 133*19c3b8c2SApple OSS Distributions if (elem_count == 0) { 134*19c3b8c2SApple OSS Distributions return retval; 135*19c3b8c2SApple OSS Distributions } else if (elem_count == 1) { 136*19c3b8c2SApple OSS Distributions retval[_name] = [self objectForType:_subtype_desc.kcs_elem_type withData:&data[_subtype_desc.kcs_elem_offset]]; 137*19c3b8c2SApple OSS Distributions } else if (_subtype_desc.kcs_elem_type == KC_ST_CHAR) { 138*19c3b8c2SApple OSS Distributions char *s = (char *)&data[_subtype_desc.kcs_elem_offset]; 139*19c3b8c2SApple OSS Distributions retval[_name] = [NSString stringWithFormat:@"%.*s", (int)elem_count, s]; 140*19c3b8c2SApple OSS Distributions } else { 141*19c3b8c2SApple OSS Distributions NSMutableArray * objArray = [NSMutableArray arrayWithCapacity:elem_count]; 142*19c3b8c2SApple OSS Distributions for (unsigned int i = 0; i < elem_count; i++) { 143*19c3b8c2SApple OSS Distributions [objArray addObject:[self objectForType:_subtype_desc.kcs_elem_type 144*19c3b8c2SApple OSS Distributions withData:&data[(_subtype_desc.kcs_elem_offset + (elem_size * i))]]]; 145*19c3b8c2SApple OSS Distributions } 146*19c3b8c2SApple OSS Distributions retval[_name] = objArray; 147*19c3b8c2SApple OSS Distributions } 148*19c3b8c2SApple OSS Distributions return retval; 149*19c3b8c2SApple OSS Distributions} 150*19c3b8c2SApple OSS Distributions 151*19c3b8c2SApple OSS Distributions- (NSString *)description 152*19c3b8c2SApple OSS Distributions{ 153*19c3b8c2SApple OSS Distributions if (_subtype_desc.kcs_flags & KCS_SUBTYPE_FLAGS_ARRAY) { 154*19c3b8c2SApple OSS Distributions return [NSString stringWithFormat:@"[%d,%d] %s %s[%d];", _subtype_desc.kcs_elem_offset, kcs_get_elem_size(&_subtype_desc), name_for_subtype(_subtype_desc.kcs_elem_type), _subtype_desc.kcs_name, kcs_get_elem_count(&_subtype_desc) ]; 155*19c3b8c2SApple OSS Distributions }else { 156*19c3b8c2SApple OSS Distributions return [NSString stringWithFormat:@"[%d,%d] %s %s;", _subtype_desc.kcs_elem_offset, kcs_get_elem_size(&_subtype_desc), name_for_subtype(_subtype_desc.kcs_elem_type), _subtype_desc.kcs_name ]; 157*19c3b8c2SApple OSS Distributions } 158*19c3b8c2SApple OSS Distributions //return [NSString stringWithFormat:@"type: %d => \"%@\" ", [self typeID], [self name]]; 159*19c3b8c2SApple OSS Distributions} 160*19c3b8c2SApple OSS Distributions 161*19c3b8c2SApple OSS Distributions- (NSString *)name 162*19c3b8c2SApple OSS Distributions{ 163*19c3b8c2SApple OSS Distributions return _name; 164*19c3b8c2SApple OSS Distributions} 165*19c3b8c2SApple OSS Distributions 166*19c3b8c2SApple OSS Distributions- (uint32_t)count 167*19c3b8c2SApple OSS Distributions{ 168*19c3b8c2SApple OSS Distributions return _count; 169*19c3b8c2SApple OSS Distributions} 170*19c3b8c2SApple OSS Distributions 171*19c3b8c2SApple OSS Distributions- (unsigned int)typeID 172*19c3b8c2SApple OSS Distributions{ 173*19c3b8c2SApple OSS Distributions return _typeID; 174*19c3b8c2SApple OSS Distributions} 175*19c3b8c2SApple OSS Distributions 176*19c3b8c2SApple OSS Distributions- (BOOL) shouldMergeData 177*19c3b8c2SApple OSS Distributions{ 178*19c3b8c2SApple OSS Distributions return TRUE; 179*19c3b8c2SApple OSS Distributions} 180*19c3b8c2SApple OSS Distributions 181*19c3b8c2SApple OSS Distributions@end 182