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