NSData+AES.m 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. #import "NSData+AES.h"
  2. #import <CommonCrypto/CommonCryptor.h>
  3. @implementation NSData (AES)
  4. - (NSData *)AES128EncryptedDataWithKey:(NSString *)key
  5. {
  6. return [self AES128EncryptedDataWithKey:key iv:nil];
  7. }
  8. - (NSData *)AES128DecryptedDataWithKey:(NSString *)key
  9. {
  10. return [self AES128DecryptedDataWithKey:key iv:nil];
  11. }
  12. - (NSData *)AES128EncryptedDataWithKey:(NSString *)key iv:(NSString *)iv
  13. {
  14. return [self AES128Operation:kCCEncrypt key:key iv:iv];
  15. }
  16. - (NSData *)AES128DecryptedDataWithKey:(NSString *)key iv:(NSString *)iv
  17. {
  18. return [self AES128Operation:kCCDecrypt key:key iv:iv];
  19. }
  20. - (NSData *)AES128Operation:(CCOperation)operation key:(NSString *)key iv:(NSString *)iv
  21. {
  22. char keyPtr[kCCKeySizeAES128 + 1];
  23. bzero(keyPtr, sizeof(keyPtr));
  24. [key getCString:keyPtr maxLength:sizeof(keyPtr) encoding:NSUTF8StringEncoding];
  25. char ivPtr[kCCBlockSizeAES128 + 1];
  26. bzero(ivPtr, sizeof(ivPtr));
  27. if (iv) {
  28. [iv getCString:ivPtr maxLength:sizeof(ivPtr) encoding:NSUTF8StringEncoding];
  29. }
  30. NSUInteger dataLength = [self length];
  31. size_t bufferSize = dataLength + kCCBlockSizeAES128;
  32. void *buffer = malloc(bufferSize);
  33. size_t numBytesEncrypted = 0;
  34. CCCryptorStatus cryptStatus = CCCrypt(operation,
  35. kCCAlgorithmAES128,
  36. kCCOptionPKCS7Padding | kCCOptionECBMode,
  37. keyPtr,
  38. kCCBlockSizeAES128,
  39. ivPtr,
  40. [self bytes],
  41. dataLength,
  42. buffer,
  43. bufferSize,
  44. &numBytesEncrypted);
  45. if (cryptStatus == kCCSuccess) {
  46. return [NSData dataWithBytesNoCopy:buffer length:numBytesEncrypted];
  47. }
  48. free(buffer);
  49. return nil;
  50. }
  51. - (NSString*)hexUppercaseString
  52. {
  53. return [self hexStringWithFormat:"%02X"];
  54. }
  55. - (NSString*)hexStringWithFormat:(const char*)format
  56. {
  57. if (self.length == 0) return @"";
  58. NSUInteger length = self.length;
  59. NSMutableData* data = [NSMutableData dataWithLength:length * 2];
  60. char *dest = data.mutableBytes;
  61. unsigned const char *src = self.bytes;
  62. for (int i = 0; i < length; ++i)
  63. {
  64. sprintf(dest + i*2, format, (unsigned int)(src[i]));
  65. }
  66. return [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
  67. }
  68. - (id)initWithHexString:(NSString*)hexString
  69. {
  70. return [self initWithHexCString:[hexString cStringUsingEncoding:NSUTF8StringEncoding]];
  71. }
  72. - (id) initWithHexCString:(const char *)hexCString
  73. {
  74. if (!hexCString) return [self init];
  75. const unsigned char *psz = (const unsigned char*)hexCString;
  76. while (isspace(*psz)) psz++;
  77. // Skip optional 0x prefix
  78. if (psz[0] == '0' && tolower(psz[1]) == 'x') psz += 2;
  79. while (isspace(*psz)) psz++;
  80. size_t len = strlen((const char*)psz);
  81. // If the string is not full number of bytes (each byte 2 hex characters), return nil.
  82. if (len % 2 != 0) return nil;
  83. unsigned char* buf = (unsigned char*)malloc(len/2);
  84. static const signed char digits[256] = {
  85. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  86. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  87. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  88. 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -1, -1, -1, -1, -1,
  89. -1,0xa,0xb,0xc,0xd,0xe,0xf, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  90. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  91. -1,0xa,0xb,0xc,0xd,0xe,0xf, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  92. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  93. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  94. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  95. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  96. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  97. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  98. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  99. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  100. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1
  101. };
  102. unsigned char* bufpointer = buf;
  103. while (1)
  104. {
  105. unsigned char c1 = (unsigned char)*psz++;
  106. signed char n1 = digits[c1];
  107. if (n1 == (signed char)-1) break; // break when null-terminator is hit
  108. unsigned char c2 = (unsigned char)*psz++;
  109. signed char n2 = digits[c2];
  110. if (n2 == (signed char)-1) break; // break when null-terminator is hit
  111. *bufpointer = (unsigned char)((n1 << 4) | n2);
  112. bufpointer++;
  113. }
  114. return [self initWithBytesNoCopy:buf length:len/2];
  115. }
  116. @end