MYBERParser.m
author Jens Alfke <jens@mooseyard.com>
Tue Jun 02 13:16:28 2009 -0700 (2009-06-02)
changeset 16 c409dbc4f068
child 17 90a70925562b
permissions -rw-r--r--
* Added ASN.1 / BER / DER utilities, to be used in generating and parsing X.509 certs.
* Added Keychain user-interaction-allowed setter. Added doc comments to MYSymmetricKey.
     1 //
     2 //  MYBERParser.m
     3 //  MYCrypto
     4 //
     5 //  Created by Jens Alfke on 6/2/09.
     6 //  Copyright 2009 Jens Alfke. All rights reserved.
     7 //
     8 
     9 #import "MYBERParser.h"
    10 #import "MYASN1Object.h"
    11 #import "MYOID.h"
    12 #import "MYErrorUtils.h"
    13 #import "CollectionUtils.h"
    14 #import "Test.h"
    15 
    16 
    17 #define MYBERParserException @"MYBERParserException"
    18 
    19 
    20 
    21 typedef struct {
    22     const uint8_t *nextChar;
    23     size_t length;
    24 } InputData;
    25 
    26 
    27 static void requireLength (size_t length, size_t expectedLength) {
    28     if (length != expectedLength)
    29         [NSException raise: MYBERParserException format: @"Unexpected value length"];
    30 }
    31 
    32 
    33 static const void* readOrDie (InputData *input, size_t len) {
    34     if (len > input->length)
    35         [NSException raise: MYBERParserException format: @"Unexpected EOF on input"];
    36     const void *bytes = input->nextChar;
    37     input->nextChar += len;
    38     input->length -= len;
    39     return bytes;
    40 }
    41 
    42 
    43 static NSData* readDataOrDie(InputData *input, size_t length) {
    44     return [NSMutableData dataWithBytes: readOrDie(input,length) length: length];
    45 }
    46 
    47 
    48 static NSString* readStringOrDie(InputData *input, size_t length, NSStringEncoding encoding) {
    49     NSString *str = [[NSString alloc] initWithBytes: readOrDie(input,length) 
    50                                              length: length
    51                                            encoding: encoding];
    52     if (!str)
    53         [NSException raise: MYBERParserException format: @"Unparseable string"];
    54     return [str autorelease];
    55 }    
    56 
    57 
    58 static uint32_t readBigEndianUnsignedInteger (InputData *input, size_t length) {
    59     if (length == 0 || length > 4)
    60         [NSException raise: MYBERParserException format: @"Invalid integer length"];
    61     uint32_t result = 0;
    62     memcpy(((uint8_t*)&result)+(4-length), readOrDie(input, length), length);
    63     return result;
    64 }
    65 
    66 static int32_t readBigEndianSignedInteger (InputData *input, size_t length) {
    67     int32_t result = (int32_t) readBigEndianUnsignedInteger(input,length);
    68     uint8_t *dst = ((uint8_t*)&result)+(4-length);
    69     if (*dst & 0x80) { // sign-extend negative value
    70         while (--dst >= (uint8_t*)&result)
    71             *dst = 0xFF;
    72     }
    73     return result;
    74 }
    75 
    76 
    77 NSDateFormatter* MYBERGeneralizedTimeFormatter() {
    78     static NSDateFormatter *sFmt;
    79     if (!sFmt) {
    80         sFmt = [[NSDateFormatter alloc] init];
    81         sFmt.dateFormat = @"yyyyMMddHHmmss'Z'";
    82         sFmt.timeZone = [NSTimeZone timeZoneWithName: @"GMT"];
    83     }
    84     return sFmt;
    85 }
    86 
    87 NSDateFormatter* MYBERUTCTimeFormatter() {
    88     static NSDateFormatter *sFmt;
    89     if (!sFmt) {
    90         sFmt = [[NSDateFormatter alloc] init];
    91         sFmt.dateFormat = @"yyMMddHHmmss'Z'";
    92         sFmt.timeZone = [NSTimeZone timeZoneWithName: @"GMT"];
    93     }
    94     return sFmt;
    95 }
    96 
    97 static NSDate* parseDate (NSString *dateStr, unsigned tag) {
    98     NSDateFormatter *fmt = (tag==23 ?MYBERUTCTimeFormatter() :MYBERGeneralizedTimeFormatter());
    99     NSDate *date = [fmt dateFromString: dateStr];
   100     if (!date)
   101         [NSException raise: MYBERParserException format: @"Unparseable date '%@'", dateStr];
   102     return date;
   103 }
   104 
   105 
   106 static id parseBER(InputData *input) {
   107     struct {
   108         unsigned tag            :5;
   109         unsigned isConstructed  :1;
   110         unsigned tagClass       :2;
   111         unsigned length         :7;
   112         unsigned isLengthLong   :1;
   113     } header;
   114     memcpy(&header, readOrDie(input,2), 2);
   115     
   116     if (header.tag == 0x1F)
   117         [NSException raise: MYBERParserException format: @"Long tags not supported"];
   118     
   119     // Parse the length:
   120     size_t length;
   121     if (!header.isLengthLong)
   122         length = header.length;
   123     else if (header.length == 0)
   124         [NSException raise: MYBERParserException format: @"Indefinite length not supported"];
   125     else
   126         length = NSSwapBigIntToHost(readBigEndianUnsignedInteger(input,header.length));
   127     
   128     Class defaultClass = [MYASN1Object class];
   129     
   130     // Tag values can be found in <Security/x509defs.h>. I'm not using them here because that
   131     // header does not exist on iPhone!
   132     
   133     if (header.isConstructed) {
   134         // Constructed:
   135         NSMutableArray *items = $marray();
   136         InputData subInput = {input->nextChar, length};
   137         while (subInput.length > 0) {
   138             [items addObject: parseBER(&subInput)];
   139         }
   140         input->nextChar += length;
   141         input->length -= length;
   142 
   143         switch (header.tag) {
   144             case 16: // sequence
   145                 return items;
   146             case 17: // set
   147                 return [NSSet setWithArray: items];
   148             default:
   149                 return [[[MYASN1Object alloc] initWithTag: header.tag
   150                                                   ofClass: header.tagClass
   151                                                components: items] autorelease];
   152         }
   153     } else {
   154         // Primitive:
   155         switch (header.tag) {
   156             case 1: { // boolean
   157                 requireLength(length,1);
   158                 return *(const uint8_t*)readOrDie(input, 1) ?$true :$false;
   159             }
   160             case 2: // integer
   161             case 10: // enum
   162             {
   163                 if (length <= 4) {
   164                     int32_t value = NSSwapBigIntToHost(readBigEndianSignedInteger(input,length));
   165                     return [NSNumber numberWithInteger: value];
   166                 } else {
   167                     // Big integer!
   168                     defaultClass = [MYASN1BigInteger class];
   169                     break;
   170                 }
   171             }
   172             case 3: // bitstring
   173             {
   174                 UInt8 unusedBits = *(const UInt8*) readOrDie(input, 1);
   175                 if (unusedBits)
   176                     Log(@"Bit-string has %u unused bits", (unsigned)unusedBits);
   177                 if (unusedBits > 7 || length < 1)
   178                     [NSException raise: MYBERParserException format: @"Bogus bit-string"];
   179                 return [[[MYBitString alloc] initWithBits: readDataOrDie(input, length-1)
   180                                                     count: 8*(length-1) - unusedBits] autorelease];
   181             }
   182             case 4: // octetstring
   183                 return readDataOrDie(input, length);
   184             case 5: // null
   185                 requireLength(length,0);
   186                 return [NSNull null];
   187             case 6: // OID
   188                 return [[[MYOID alloc] initWithBEREncoding: readDataOrDie(input, length)] autorelease];
   189             case 12: // UTF8String
   190                 return readStringOrDie(input,length,NSUTF8StringEncoding);
   191             case 18: // numeric string
   192             case 19: // printable string:
   193                 return readStringOrDie(input,length,NSASCIIStringEncoding);
   194             case 23: // UTC time:
   195             case 24: // Generalized time:
   196                 return parseDate(readStringOrDie(input,length,NSASCIIStringEncoding), header.tag);
   197             default:
   198                 break;
   199         }
   200     }
   201 
   202     // Generic case -- create and return a MYASN1Object:
   203     NSData *value = readDataOrDie(input, length);
   204     id result = [[[defaultClass alloc] initWithTag: header.tag
   205                                            ofClass: header.tagClass 
   206                                        constructed: header.isConstructed
   207                                              value: value] autorelease];
   208     if( defaultClass == [MYASN1Object class])
   209         Warn(@"parseBER: Returning default %@", result);
   210     return result;
   211 }
   212 
   213 
   214 id MYBERParse (NSData *ber, NSError **outError) {
   215     @try{
   216         InputData input = {ber.bytes, ber.length};
   217         return parseBER(&input);
   218     }@catch (NSException *x) {
   219         if ($equal(x.name, MYBERParserException)) {
   220             *outError = MYError(1,MYASN1ErrorDomain, @"%@", x.reason);
   221         } else {
   222             @throw(x);
   223         }
   224     }
   225     return nil;
   226 }
   227 
   228 
   229 
   230 
   231 #pragma mark -
   232 #pragma mark TEST CASES:
   233 
   234 
   235 #define $data(BYTES...)    ({const uint8_t bytes[] = {BYTES}; [NSData dataWithBytes: bytes length: sizeof(bytes)];})
   236 
   237 TestCase(ParseBER) {
   238     CAssertEqual(MYBERParse($data(0x05, 0x00), nil),
   239                  [NSNull null]);
   240     CAssertEqual(MYBERParse($data(0x01, 0x01, 0xFF), nil),
   241                  $true);
   242     CAssertEqual(MYBERParse($data(0x01, 0x01, 0x00), nil),
   243                  $false);
   244     
   245     // integers:
   246     CAssertEqual(MYBERParse($data(0x02, 0x01, 0x00), nil),
   247                  $object(0));
   248     CAssertEqual(MYBERParse($data(0x02, 0x01, 0x48), nil),
   249                  $object(72));
   250     CAssertEqual(MYBERParse($data(0x02, 0x01, 0x80), nil),
   251                  $object(-128));
   252     CAssertEqual(MYBERParse($data(0x02, 0x02, 0x00, 0x80), nil),
   253                  $object(128));
   254     CAssertEqual(MYBERParse($data(0x02, 0x02, 0x30,0x39), nil),
   255                  $object(12345));
   256     CAssertEqual(MYBERParse($data(0x02, 0x02, 0xCF, 0xC7), nil),
   257                  $object(-12345));
   258     CAssertEqual(MYBERParse($data(0x02, 0x04, 0x07, 0x5B, 0xCD, 0x15), nil),
   259                  $object(123456789));
   260     CAssertEqual(MYBERParse($data(0x02, 0x04, 0xF8, 0xA4, 0x32, 0xEB), nil),
   261                  $object(-123456789));
   262     CAssertEqual(MYBERParse($data(0x02, 0x04, 0xF8, 0xA4, 0x32, 0xEB), nil),
   263                  $object(-123456789));
   264     
   265     // octet strings:
   266     CAssertEqual(MYBERParse($data(0x04, 0x05, 'h', 'e', 'l', 'l', 'o'), nil),
   267                  [@"hello" dataUsingEncoding: NSASCIIStringEncoding]);
   268     CAssertEqual(MYBERParse($data(0x04, 0x00), nil),
   269                  [NSData data]);
   270     CAssertEqual(MYBERParse($data(0x0C, 0x05, 'h', 'e', 'l', 'l', 'o'), nil),
   271                  @"hello");
   272     
   273     // sequences:
   274     CAssertEqual(MYBERParse($data(0x30, 0x06,  0x02, 0x01, 0x48,  0x01, 0x01, 0xFF), nil),
   275                  $array($object(72), $true));
   276     CAssertEqual(MYBERParse($data(0x30, 0x10,  
   277                                   0x30, 0x06,  0x02, 0x01, 0x48,  0x01, 0x01, 0xFF,
   278                                   0x30, 0x06,  0x02, 0x01, 0x48,  0x01, 0x01, 0xFF), nil),
   279                  $array( $array($object(72), $true), $array($object(72), $true)));
   280 }
   281 
   282 
   283 TestCase(ParseCert) {
   284     NSData *cert = [NSData dataWithContentsOfFile: @"../../Tests/selfsigned.cer"];
   285     NSError *error = nil;
   286     id parsed = MYBERParse(cert,&error);
   287     CAssert(parsed);
   288     CAssertNil(error);
   289     NSString *dump = [MYASN1Object dump: parsed];
   290     CAssert(dump);
   291 
   292     cert = [NSData dataWithContentsOfFile: @"../../Tests/iphonedev.cer"];
   293     parsed = MYBERParse(cert,&error);
   294     CAssert(parsed);
   295     CAssertNil(error);
   296     dump = [MYASN1Object dump: parsed];
   297     CAssert(dump);
   298 }