2018-05-11 09:51:14 +00:00
|
|
|
#include "AesCtrWrappedIFile.h"
|
|
|
|
|
2018-05-21 12:48:59 +00:00
|
|
|
AesCtrWrappedIFile::AesCtrWrappedIFile(fnd::IFile* file, bool ownIfile, const crypto::aes::sAes128Key& key, const crypto::aes::sAesIvCtr& ctr) :
|
|
|
|
mOwnIFile(ownIfile),
|
|
|
|
mFile(file),
|
|
|
|
mKey(key),
|
2018-05-22 11:45:40 +00:00
|
|
|
mBaseCtr(ctr),
|
|
|
|
mFileOffset(0)
|
2018-05-21 12:48:59 +00:00
|
|
|
{
|
2018-05-27 10:16:42 +00:00
|
|
|
mCache.alloc(kCacheSizeAllocSize);
|
2018-05-21 12:48:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AesCtrWrappedIFile::~AesCtrWrappedIFile()
|
|
|
|
{
|
|
|
|
if (mOwnIFile)
|
|
|
|
{
|
|
|
|
delete mFile;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-11 09:51:14 +00:00
|
|
|
size_t AesCtrWrappedIFile::size()
|
|
|
|
{
|
2018-05-11 13:20:28 +00:00
|
|
|
return mFile->size();
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AesCtrWrappedIFile::seek(size_t offset)
|
|
|
|
{
|
2018-05-22 11:45:40 +00:00
|
|
|
mFileOffset = offset;
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AesCtrWrappedIFile::read(byte_t* out, size_t len)
|
|
|
|
{
|
2018-05-27 10:16:42 +00:00
|
|
|
//printf("[%x] AesCtrWrappedIFile::read(offset=0x%" PRIx64 ", size=0x%" PRIx64 ")\n", this, mFileOffset, len);
|
2018-05-22 11:45:40 +00:00
|
|
|
|
2018-05-27 10:16:42 +00:00
|
|
|
size_t read_len;
|
|
|
|
size_t read_pos;
|
2018-05-11 09:51:14 +00:00
|
|
|
|
2018-05-27 10:16:42 +00:00
|
|
|
size_t cache_reads = (len / kCacheSize) + ((len % kCacheSize) != 0);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < cache_reads; i++)
|
2018-05-11 09:51:14 +00:00
|
|
|
{
|
2018-06-24 15:01:16 +00:00
|
|
|
read_len = _MIN(len - (i * kCacheSize), kCacheSize);
|
2018-05-27 10:16:42 +00:00
|
|
|
read_pos = ((mFileOffset >> 4) << 4) + (i * kCacheSize);
|
|
|
|
|
|
|
|
//printf("[%x] AesCtrWrappedIFile::read() CACHE READ: readlen=%" PRIx64 "\n", this, read_len);
|
|
|
|
|
|
|
|
mFile->seek(read_pos);
|
2018-06-24 15:01:16 +00:00
|
|
|
mFile->read(mCache.data(), kCacheSizeAllocSize);
|
2018-05-27 10:16:42 +00:00
|
|
|
|
|
|
|
crypto::aes::AesIncrementCounter(mBaseCtr.iv, read_pos>>4, mCurrentCtr.iv);
|
2018-06-24 15:01:16 +00:00
|
|
|
crypto::aes::AesCtr(mCache.data(), kCacheSizeAllocSize, mKey.key, mCurrentCtr.iv, mCache.data());
|
2018-05-27 10:16:42 +00:00
|
|
|
|
2018-06-24 15:01:16 +00:00
|
|
|
memcpy(out + (i * kCacheSize), mCache.data() + (mFileOffset & 0xf), read_len);
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
2018-05-22 11:45:40 +00:00
|
|
|
|
|
|
|
seek(mFileOffset + len);
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AesCtrWrappedIFile::read(byte_t* out, size_t offset, size_t len)
|
|
|
|
{
|
|
|
|
seek(offset);
|
|
|
|
read(out, len);
|
|
|
|
}
|
|
|
|
|
2018-05-27 10:16:42 +00:00
|
|
|
void AesCtrWrappedIFile::write(const byte_t* in, size_t len)
|
2018-05-11 09:51:14 +00:00
|
|
|
{
|
2018-05-27 10:16:42 +00:00
|
|
|
size_t write_len;
|
|
|
|
size_t write_pos;
|
|
|
|
|
|
|
|
size_t cache_writes = (len / kCacheSize) + ((len % kCacheSize) != 0);
|
2018-05-22 11:45:40 +00:00
|
|
|
|
2018-05-27 10:16:42 +00:00
|
|
|
for (size_t i = 0; i < cache_writes; i++)
|
|
|
|
{
|
2018-06-24 15:01:16 +00:00
|
|
|
write_len = _MIN(len - (i * kCacheSize), kCacheSize);
|
2018-05-27 10:16:42 +00:00
|
|
|
write_pos = ((mFileOffset >> 4) << 4) + (i * kCacheSize);
|
|
|
|
|
|
|
|
//printf("[%x] AesCtrWrappedIFile::read() CACHE READ: readlen=%" PRIx64 "\n", this, read_len);
|
|
|
|
|
2018-06-24 15:01:16 +00:00
|
|
|
memcpy(mCache.data() + (mFileOffset & 0xf), in + (i * kCacheSize), write_len);
|
2018-05-27 10:16:42 +00:00
|
|
|
|
|
|
|
crypto::aes::AesIncrementCounter(mBaseCtr.iv, write_pos>>4, mCurrentCtr.iv);
|
2018-06-24 15:01:16 +00:00
|
|
|
crypto::aes::AesCtr(mCache.data(), kCacheSizeAllocSize, mKey.key, mCurrentCtr.iv, mCache.data());
|
2018-05-27 10:16:42 +00:00
|
|
|
|
|
|
|
mFile->seek(write_pos);
|
2018-06-24 15:01:16 +00:00
|
|
|
mFile->write(mCache.data(), kCacheSizeAllocSize);
|
2018-05-27 10:16:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
seek(mFileOffset + len);
|
|
|
|
|
|
|
|
/*
|
2018-05-11 09:51:14 +00:00
|
|
|
for (size_t i = 0; i < (len / kAesCtrScratchSize); i++)
|
|
|
|
{
|
2018-06-24 15:01:16 +00:00
|
|
|
memcpy(mScratch.data() + mBlockOffset, out + (i * kAesCtrScratchSize), kAesCtrScratchSize);
|
|
|
|
crypto::aes::AesCtr(mScratch.data(), kAesCtrScratchAllocSize, mKey.key, mCurrentCtr.iv, mScratch.data());
|
|
|
|
mFile->write(mScratch.data() + mBlockOffset, kAesCtrScratchSize);
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (len % kAesCtrScratchSize)
|
|
|
|
{
|
|
|
|
size_t write_len = len % kAesCtrScratchSize;
|
|
|
|
size_t write_pos = ((len / kAesCtrScratchSize) * kAesCtrScratchSize);
|
2018-06-24 15:01:16 +00:00
|
|
|
memcpy(mScratch.data() + mBlockOffset, out + write_pos, write_len);
|
|
|
|
crypto::aes::AesCtr(mScratch.data(), kAesCtrScratchAllocSize, mKey.key, mCurrentCtr.iv, mScratch.data());
|
|
|
|
mFile->write(mScratch.data() + mBlockOffset, write_len);
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
2018-05-27 10:16:42 +00:00
|
|
|
*/
|
2018-05-22 11:45:40 +00:00
|
|
|
seek(mFileOffset + len);
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|
|
|
|
|
2018-05-27 10:16:42 +00:00
|
|
|
void AesCtrWrappedIFile::write(const byte_t* in, size_t offset, size_t len)
|
2018-05-11 09:51:14 +00:00
|
|
|
{
|
|
|
|
seek(offset);
|
2018-05-27 10:16:42 +00:00
|
|
|
write(in, len);
|
2018-05-11 09:51:14 +00:00
|
|
|
}
|