You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

148 lines
3.6 KiB

/**
* @file encrypt.cpp
*
* Implementation of functions for compression and decompressing MPQ data.
*/
#include <SDL.h>
#include <array>
#include <cctype>
#include <memory>
#include "encrypt.h"
#include "pkware.h"
namespace devilution {
namespace {
unsigned int PkwareBufferRead(char *buf, unsigned int *size, void *param) // NOLINT(readability-non-const-parameter)
{
auto *pInfo = reinterpret_cast<TDataInfo *>(param);
uint32_t sSize;
if (*size >= pInfo->size - pInfo->srcOffset) {
sSize = pInfo->size - pInfo->srcOffset;
} else {
sSize = *size;
}
memcpy(buf, pInfo->srcData + pInfo->srcOffset, sSize);
pInfo->srcOffset += sSize;
return sSize;
}
void PkwareBufferWrite(char *buf, unsigned int *size, void *param) // NOLINT(readability-non-const-parameter)
{
auto *pInfo = reinterpret_cast<TDataInfo *>(param);
memcpy(pInfo->destData + pInfo->destOffset, buf, *size);
pInfo->destOffset += *size;
}
const std::array<std::array<uint32_t, 256>, 5> hashtable = []() {
uint32_t seed = 0x00100001;
std::array<std::array<uint32_t, 256>, 5> ret = {};
for (int i = 0; i < 256; i++) {
for (int j = 0; j < 5; j++) { // NOLINT(modernize-loop-convert)
seed = (125 * seed + 3) % 0x2AAAAB;
uint32_t ch = (seed & 0xFFFF);
seed = (125 * seed + 3) % 0x2AAAAB;
ret[j][i] = ch << 16 | (seed & 0xFFFF);
}
}
return ret;
}();
} // namespace
void Decrypt(uint32_t *castBlock, uint32_t size, uint32_t key)
{
uint32_t seed = 0xEEEEEEEE;
for (uint32_t i = 0; i < (size >> 2); i++) {
uint32_t t = SDL_SwapLE32(*castBlock);
seed += hashtable[4][(key & 0xFF)];
t ^= seed + key;
*castBlock = t;
seed += t + (seed << 5) + 3;
castBlock++;
key = (((key << 0x15) ^ 0xFFE00000) + 0x11111111) | (key >> 0x0B);
}
}
void Encrypt(uint32_t *castBlock, uint32_t size, uint32_t key)
{
uint32_t seed = 0xEEEEEEEE;
for (unsigned i = 0; i < (size >> 2); i++) {
uint32_t ch = *castBlock;
uint32_t t = ch;
seed += hashtable[4][(key & 0xFF)];
t ^= seed + key;
*castBlock = SDL_SwapLE32(t);
castBlock++;
seed += ch + (seed << 5) + 3;
key = (((key << 0x15) ^ 0xFFE00000) + 0x11111111) | (key >> 0x0B);
}
}
uint32_t Hash(const char *s, int type)
{
uint32_t seed1 = 0x7FED7FED;
uint32_t seed2 = 0xEEEEEEEE;
while (s != nullptr && (*s != '\0')) {
int8_t ch = *s++;
ch = toupper(ch);
seed1 = hashtable[type][ch] ^ (seed1 + seed2);
seed2 += ch + seed1 + (seed2 << 5) + 3;
}
return seed1;
}
uint32_t PkwareCompress(byte *srcData, uint32_t size)
{
std::unique_ptr<char[]> ptr { new char[CMP_BUFFER_SIZE] };
unsigned destSize = 2 * size;
if (destSize < 2 * 4096)
destSize = 2 * 4096;
std::unique_ptr<byte[]> destData { new byte[destSize] };
TDataInfo param;
param.srcData = srcData;
param.srcOffset = 0;
param.destData = destData.get();
param.destOffset = 0;
param.size = size;
unsigned type = 0;
unsigned dsize = 4096;
implode(PkwareBufferRead, PkwareBufferWrite, ptr.get(), &param, &type, &dsize);
if (param.destOffset < size) {
memcpy(srcData, destData.get(), param.destOffset);
size = param.destOffset;
}
return size;
}
void PkwareDecompress(byte *inBuff, int recvSize, int maxBytes)
{
TDataInfo info;
std::unique_ptr<char[]> ptr { new char[CMP_BUFFER_SIZE] };
std::unique_ptr<byte[]> outBuff { new byte[maxBytes] };
info.srcData = inBuff;
info.srcOffset = 0;
info.destData = outBuff.get();
info.destOffset = 0;
info.size = recvSize;
explode(PkwareBufferRead, PkwareBufferWrite, ptr.get(), &info);
memcpy(inBuff, outBuff.get(), info.destOffset);
}
} // namespace devilution