]> git.saurik.com Git - apple/xnu.git/blobdiff - libkdd/KCDStructTypeDescription.m
xnu-3789.1.32.tar.gz
[apple/xnu.git] / libkdd / KCDStructTypeDescription.m
diff --git a/libkdd/KCDStructTypeDescription.m b/libkdd/KCDStructTypeDescription.m
new file mode 100644 (file)
index 0000000..7cc3b20
--- /dev/null
@@ -0,0 +1,141 @@
+/*
+ * 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