/*
* Copyright (c) 2015 Apple Inc. All rights reserved.
*
* @APPLE_OSREFERENCE_LICENSE_HEADER_START@
*
* This file contains Original Code and/or Modifications of Original Code
* as defined in and that are subject to the Apple Public Source License
* Version 2.0 (the 'License'). You may not use this file except in
* compliance with the License. The rights granted to you under the License
* may not be used to create, or enable the creation or redistribution of,
* unlawful or unlicensed copies of an Apple operating system, or to
* circumvent, violate, or enable the circumvention or violation of, any
* terms of an Apple operating system software license agreement.
*
* Please obtain a copy of the License at
* http://www.opensource.apple.com/apsl/ and read it before using this file.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
* Please see the License for the specific language governing rights and
* limitations under the License.
*
* @APPLE_OSREFERENCE_LICENSE_HEADER_END@
*/
#import "KCDStructTypeDescription.h"
#ifndef KCDATA_TYPE_MAX_WITH_DESC
#define KCDATA_TYPE_MAX_WITH_DESC 0x6
#endif
@interface
KCDStructTypeDescription () {
unsigned int _typeID;
NSString * _name;
NSMutableArray * _fields;
BOOL _needDescriptionAsKey;
BOOL _flagsRequestedMerge;
}
@end
@implementation KCDStructTypeDescription
- (id)initWithType:(unsigned int)typeID withName:(NSString *)name
{
if ((self = [super init])) {
_typeID = typeID;
_name = name;
_needDescriptionAsKey = NO;
if (typeID >= 0x1 && typeID <= KCDATA_TYPE_MAX_WITH_DESC)
_needDescriptionAsKey = YES;
_fields = [[NSMutableArray alloc] init];
_flagsRequestedMerge = NO;
return self;
}
return NULL;
}
- (void)addFieldBasicType:(KCDBasicTypeDescription *)fieldType
{
[_fields addObject:fieldType];
}
- (void)setFlagsRequestedMerge
{
_flagsRequestedMerge = YES;
}
- (NSDictionary *)parseData:(void *)dataBuffer ofLength:(uint32_t)length
{
NSMutableDictionary * retval = [[NSMutableDictionary alloc] init];
for (KCDataType * fi in _fields) {
NSDictionary * _d = [fi parseData:dataBuffer ofLength:length];
if (!_d) {
return nil;
}
for (NSString * k in [_d keyEnumerator]) {
retval[k] = _d[k];
}
}
if (_typeID == KCDATA_TYPE_TYPEDEFINTION){
uint32_t elem_size = sizeof(struct kcdata_subtype_descriptor);
uint32_t elem_count = (length - offsetof(struct kcdata_type_definition, kct_elements))/elem_size;
NSMutableArray * fields_array = [NSMutableArray arrayWithCapacity:elem_count];
struct kcdata_subtype_descriptor *fields_dsc = (struct kcdata_subtype_descriptor *) ((uintptr_t)dataBuffer + offsetof(struct kcdata_type_definition, kct_elements));
int i = 0;
for (i = 0; i < elem_count; i++) {
KCDBasicTypeDescription * tmpdsc = [[KCDBasicTypeDescription alloc] initWithKCTypeDesc:&fields_dsc[i]];
NSString *field_desc_str = [tmpdsc description];
[fields_array addObject:field_desc_str];
}
retval[@"fields"] = fields_array;
}
if (_needDescriptionAsKey) {
NSString * desc = retval[@"desc"];
NSObject * obj = retval[@"data"];
retval[desc] = obj;
[retval removeObjectForKey:@"desc"];
[retval removeObjectForKey:@"data"];
}
return retval;
}
- (BOOL)shouldMergeData
{
/*
* If this is a type where the kcdata item itself carries the key name, or
* KCS_SUBTYPE_FLAGS_MERGE was used to define the type, then a member of
* this type should have it's dict merged into the parent container,
* instead of being represented as typename => dict.
*/
return _needDescriptionAsKey || _flagsRequestedMerge;
}
- (NSString *)description
{
return [NSString stringWithFormat:@"type: %d => \"%@\" ", _typeID, _name];
}
- (NSString *)name
{
return _name;
}
- (uint32_t)count
{
return (uint32_t)[_fields count];
}
- (unsigned int)typeID
{
return _typeID;
}
@end