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