|
@@ -16,7 +16,6 @@ cdef extern from "openssl/evp.h":
|
|
ctypedef struct EVP_CIPHER:
|
|
ctypedef struct EVP_CIPHER:
|
|
pass
|
|
pass
|
|
ctypedef struct EVP_CIPHER_CTX:
|
|
ctypedef struct EVP_CIPHER_CTX:
|
|
- unsigned char *iv
|
|
|
|
pass
|
|
pass
|
|
ctypedef struct ENGINE:
|
|
ctypedef struct ENGINE:
|
|
pass
|
|
pass
|
|
@@ -40,12 +39,40 @@ import struct
|
|
|
|
|
|
_int = struct.Struct('>I')
|
|
_int = struct.Struct('>I')
|
|
_long = struct.Struct('>Q')
|
|
_long = struct.Struct('>Q')
|
|
|
|
+_2long = struct.Struct('>QQ')
|
|
|
|
|
|
bytes_to_int = lambda x, offset=0: _int.unpack_from(x, offset)[0]
|
|
bytes_to_int = lambda x, offset=0: _int.unpack_from(x, offset)[0]
|
|
bytes_to_long = lambda x, offset=0: _long.unpack_from(x, offset)[0]
|
|
bytes_to_long = lambda x, offset=0: _long.unpack_from(x, offset)[0]
|
|
long_to_bytes = lambda x: _long.pack(x)
|
|
long_to_bytes = lambda x: _long.pack(x)
|
|
|
|
|
|
|
|
|
|
|
|
+def bytes16_to_int(b, offset=0):
|
|
|
|
+ h, l = _2long.unpack_from(b, offset)
|
|
|
|
+ return (h << 64) + l
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+def int_to_bytes16(i):
|
|
|
|
+ max_uint64 = 0xffffffffffffffff
|
|
|
|
+ l = i & max_uint64
|
|
|
|
+ h = (i >> 64) & max_uint64
|
|
|
|
+ return _2long.pack(h, l)
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+def increment_iv(iv, amount=1):
|
|
|
|
+ """
|
|
|
|
+ Increment the IV by the given amount (default 1).
|
|
|
|
+
|
|
|
|
+ :param iv: input IV, 16 bytes (128 bit)
|
|
|
|
+ :param amount: increment value
|
|
|
|
+ :return: input_IV + amount, 16 bytes (128 bit)
|
|
|
|
+ """
|
|
|
|
+ assert len(iv) == 16
|
|
|
|
+ iv = bytes16_to_int(iv)
|
|
|
|
+ iv += amount
|
|
|
|
+ iv = int_to_bytes16(iv)
|
|
|
|
+ return iv
|
|
|
|
+
|
|
|
|
+
|
|
def num_aes_blocks(int length):
|
|
def num_aes_blocks(int length):
|
|
"""Return the number of AES blocks required to encrypt/decrypt *length* bytes of data.
|
|
"""Return the number of AES blocks required to encrypt/decrypt *length* bytes of data.
|
|
Note: this is only correct for modes without padding, like AES-CTR.
|
|
Note: this is only correct for modes without padding, like AES-CTR.
|
|
@@ -58,6 +85,8 @@ cdef class AES:
|
|
"""
|
|
"""
|
|
cdef EVP_CIPHER_CTX *ctx
|
|
cdef EVP_CIPHER_CTX *ctx
|
|
cdef int is_encrypt
|
|
cdef int is_encrypt
|
|
|
|
+ cdef unsigned char iv_orig[16]
|
|
|
|
+ cdef int blocks
|
|
|
|
|
|
def __cinit__(self, is_encrypt, key, iv=None):
|
|
def __cinit__(self, is_encrypt, key, iv=None):
|
|
self.ctx = EVP_CIPHER_CTX_new()
|
|
self.ctx = EVP_CIPHER_CTX_new()
|
|
@@ -82,6 +111,10 @@ cdef class AES:
|
|
key2 = key
|
|
key2 = key
|
|
if iv:
|
|
if iv:
|
|
iv2 = iv
|
|
iv2 = iv
|
|
|
|
+ assert isinstance(iv, bytes) and len(iv) == 16
|
|
|
|
+ for i in range(16):
|
|
|
|
+ self.iv_orig[i] = iv[i]
|
|
|
|
+ self.blocks = 0 # number of AES blocks encrypted starting with iv_orig
|
|
# Initialise key and IV
|
|
# Initialise key and IV
|
|
if self.is_encrypt:
|
|
if self.is_encrypt:
|
|
if not EVP_EncryptInit_ex(self.ctx, NULL, NULL, key2, iv2):
|
|
if not EVP_EncryptInit_ex(self.ctx, NULL, NULL, key2, iv2):
|
|
@@ -92,7 +125,7 @@ cdef class AES:
|
|
|
|
|
|
@property
|
|
@property
|
|
def iv(self):
|
|
def iv(self):
|
|
- return self.ctx.iv[:16]
|
|
|
|
|
|
+ return increment_iv(self.iv_orig[:16], self.blocks)
|
|
|
|
|
|
def encrypt(self, data):
|
|
def encrypt(self, data):
|
|
cdef int inl = len(data)
|
|
cdef int inl = len(data)
|
|
@@ -109,6 +142,7 @@ cdef class AES:
|
|
if not EVP_EncryptFinal_ex(self.ctx, out+ctl, &outl):
|
|
if not EVP_EncryptFinal_ex(self.ctx, out+ctl, &outl):
|
|
raise Exception('EVP_EncryptFinal failed')
|
|
raise Exception('EVP_EncryptFinal failed')
|
|
ctl += outl
|
|
ctl += outl
|
|
|
|
+ self.blocks += num_aes_blocks(ctl)
|
|
return out[:ctl]
|
|
return out[:ctl]
|
|
finally:
|
|
finally:
|
|
free(out)
|
|
free(out)
|
|
@@ -133,6 +167,7 @@ cdef class AES:
|
|
# CTR mode does not use padding nor authentication.
|
|
# CTR mode does not use padding nor authentication.
|
|
raise Exception('EVP_DecryptFinal failed')
|
|
raise Exception('EVP_DecryptFinal failed')
|
|
ptl += outl
|
|
ptl += outl
|
|
|
|
+ self.blocks += num_aes_blocks(inl)
|
|
return out[:ptl]
|
|
return out[:ptl]
|
|
finally:
|
|
finally:
|
|
free(out)
|
|
free(out)
|