582 lines
17 KiB
C++
582 lines
17 KiB
C++
#include "PSOEncryption.hh"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include <stdexcept>
|
|
#include <string>
|
|
#include <phosg/Encoding.hh>
|
|
#include <phosg/Strings.hh>
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
// TODO: fix style in this file, especially in psobb functions
|
|
|
|
|
|
|
|
// Most ciphers used by PSO are symmetric; alias decrypt to encrypt by default
|
|
void PSOEncryption::decrypt(void* data, size_t size, bool advance) {
|
|
this->encrypt(data, size, advance);
|
|
}
|
|
|
|
|
|
|
|
void PSOPCEncryption::update_stream() {
|
|
uint32_t esi, edi, eax, ebp, edx;
|
|
edi = 1;
|
|
edx = 0x18;
|
|
eax = edi;
|
|
while (edx > 0) {
|
|
esi = this->stream[eax + 0x1F];
|
|
ebp = this->stream[eax] - esi;
|
|
this->stream[eax] = ebp;
|
|
eax++;
|
|
edx--;
|
|
}
|
|
edi = 0x19;
|
|
edx = 0x1F;
|
|
eax = edi;
|
|
while (edx > 0) {
|
|
esi = this->stream[eax - 0x18];
|
|
ebp = this->stream[eax] - esi;
|
|
this->stream[eax] = ebp;
|
|
eax++;
|
|
edx--;
|
|
}
|
|
}
|
|
|
|
PSOPCEncryption::PSOPCEncryption(uint32_t seed) : offset(1) {
|
|
uint32_t esi, ebx, edi, eax, edx, var1;
|
|
esi = 1;
|
|
ebx = seed;
|
|
edi = 0x15;
|
|
this->stream[56] = ebx;
|
|
this->stream[55] = ebx;
|
|
while (edi <= 0x46E) {
|
|
eax = edi;
|
|
var1 = eax / 55;
|
|
edx = eax - (var1 * 55);
|
|
ebx = ebx - esi;
|
|
edi = edi + 0x15;
|
|
this->stream[edx] = esi;
|
|
esi = ebx;
|
|
ebx = this->stream[edx];
|
|
}
|
|
for (size_t x = 0; x < 5; x++) {
|
|
this->update_stream();
|
|
}
|
|
}
|
|
|
|
uint32_t PSOPCEncryption::next(bool advance) {
|
|
if (this->offset == PC_STREAM_LENGTH) {
|
|
this->update_stream();
|
|
this->offset = 1;
|
|
}
|
|
uint32_t ret = this->stream[this->offset];
|
|
if (advance) {
|
|
this->offset++;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void PSOPCEncryption::encrypt(void* vdata, size_t size, bool advance) {
|
|
if (size & 3) {
|
|
throw invalid_argument("size must be a multiple of 4");
|
|
}
|
|
if (!advance && (size != 4)) {
|
|
throw logic_error("cannot peek-encrypt/decrypt with size > 4");
|
|
}
|
|
size >>= 2;
|
|
|
|
uint32_t* data = reinterpret_cast<uint32_t*>(vdata);
|
|
for (size_t x = 0; x < size; x++) {
|
|
data[x] ^= this->next(advance);
|
|
}
|
|
}
|
|
|
|
void PSOPCEncryption::skip(size_t size) {
|
|
if (size & 3) {
|
|
throw invalid_argument("size must be a multiple of 4");
|
|
}
|
|
size >>= 2;
|
|
|
|
// TODO: Do something smarter than just calling next() in a loop here
|
|
size_t new_offset = this->offset + size;
|
|
while (new_offset > PC_STREAM_LENGTH) {
|
|
this->update_stream();
|
|
// The PC encryption apparently always skips the first key in the stream
|
|
new_offset -= (PC_STREAM_LENGTH - 1);
|
|
}
|
|
this->offset = new_offset;
|
|
}
|
|
|
|
|
|
|
|
|
|
void PSOGCEncryption::update_stream() {
|
|
uint32_t r5, r6, r7;
|
|
r5 = 0;
|
|
r6 = 489;
|
|
r7 = 0;
|
|
|
|
while (r6 != GC_STREAM_LENGTH) {
|
|
this->stream[r5++] ^= this->stream[r6++];
|
|
}
|
|
|
|
while (r5 != GC_STREAM_LENGTH) {
|
|
this->stream[r5++] ^= this->stream[r7++];
|
|
}
|
|
|
|
this->offset = 0;
|
|
}
|
|
|
|
uint32_t PSOGCEncryption::next(bool advance) {
|
|
if (this->offset == GC_STREAM_LENGTH) {
|
|
this->update_stream();
|
|
}
|
|
uint32_t ret = this->stream[this->offset];
|
|
if (advance) {
|
|
this->offset++;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
PSOGCEncryption::PSOGCEncryption(uint32_t seed) : offset(0) {
|
|
uint32_t x, y, basekey, source1, source2, source3;
|
|
basekey = 0;
|
|
|
|
for (x = 0; x <= 16; x++) {
|
|
for (y = 0; y < 32; y++) {
|
|
seed = seed * 0x5D588B65;
|
|
basekey = basekey >> 1;
|
|
seed++;
|
|
if (seed & 0x80000000) {
|
|
basekey = basekey | 0x80000000;
|
|
} else {
|
|
basekey = basekey & 0x7FFFFFFF;
|
|
}
|
|
}
|
|
this->stream[this->offset++] = basekey;
|
|
}
|
|
|
|
this->stream[this->offset - 1] = (((this->stream[0] >> 9) ^ (this->stream[this->offset - 1] << 23)) ^ this->stream[15]);
|
|
|
|
source1 = 0;
|
|
source2 = 1;
|
|
source3 = this->offset - 1;
|
|
while (this->offset != GC_STREAM_LENGTH) {
|
|
this->stream[this->offset++] = (this->stream[source3++] ^ (((this->stream[source1++] << 23) & 0xFF800000) ^ ((this->stream[source2++] >> 9) & 0x007FFFFF)));
|
|
}
|
|
|
|
for (size_t x = 0; x < 4; x++) {
|
|
this->update_stream();
|
|
}
|
|
}
|
|
|
|
void PSOGCEncryption::encrypt(void* vdata, size_t size, bool advance) {
|
|
if (size & 3) {
|
|
throw invalid_argument("size must be a multiple of 4");
|
|
}
|
|
if (!advance && (size != 4)) {
|
|
throw logic_error("cannot peek-encrypt/decrypt with size > 4");
|
|
}
|
|
size >>= 2;
|
|
|
|
uint32_t* data = reinterpret_cast<uint32_t*>(vdata);
|
|
for (size_t x = 0; x < size; x++) {
|
|
data[x] ^= this->next(advance);
|
|
}
|
|
}
|
|
|
|
void PSOGCEncryption::skip(size_t size) {
|
|
if (size & 3) {
|
|
throw invalid_argument("size must be a multiple of 4");
|
|
}
|
|
size >>= 2;
|
|
|
|
size_t new_offset = this->offset + size;
|
|
while (new_offset > GC_STREAM_LENGTH) {
|
|
this->update_stream();
|
|
new_offset -= GC_STREAM_LENGTH;
|
|
}
|
|
this->offset = new_offset;
|
|
}
|
|
|
|
|
|
|
|
void PSOBBEncryption::decrypt(void* vdata, size_t size, bool) {
|
|
if (size & 7) {
|
|
throw invalid_argument("size must be a multiple of 8");
|
|
}
|
|
size >>= 2;
|
|
|
|
uint32_t* data = reinterpret_cast<uint32_t*>(vdata);
|
|
uint32_t edx, ebx, ebp, esi, edi;
|
|
|
|
edx = 0;
|
|
while (edx < size) {
|
|
ebx = data[edx];
|
|
ebx = ebx ^ this->stream[5];
|
|
ebp = ((this->stream[(ebx >> 0x18) + 0x12] + this->stream[((ebx >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebx >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebx & 0xFF) + 0x312];
|
|
ebp = ebp ^ this->stream[4];
|
|
ebp ^= data[edx + 1];
|
|
edi = ((this->stream[(ebp >> 0x18) + 0x12] + this->stream[((ebp >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebp >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebp & 0xFF) + 0x312];
|
|
edi = edi ^ this->stream[3];
|
|
ebx = ebx ^ edi;
|
|
esi = ((this->stream[(ebx >> 0x18) + 0x12] + this->stream[((ebx >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebx >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebx & 0xFF) + 0x312];
|
|
ebp = ebp ^ esi ^ this->stream[2];
|
|
edi = ((this->stream[(ebp >> 0x18) + 0x12] + this->stream[((ebp >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebp >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebp & 0xFF) + 0x312];
|
|
edi = edi ^ this->stream[1];
|
|
ebp = ebp ^ this->stream[0];
|
|
ebx = ebx ^ edi;
|
|
data[edx] = ebp;
|
|
data[edx + 1] = ebx;
|
|
edx += 2;
|
|
}
|
|
}
|
|
|
|
void PSOBBEncryption::encrypt(void* vdata, size_t size, bool) {
|
|
if (size & 7) {
|
|
throw invalid_argument("size must be a multiple of 8");
|
|
}
|
|
size >>= 2;
|
|
|
|
uint32_t* data = reinterpret_cast<uint32_t*>(vdata);
|
|
uint32_t edx, ebx, ebp, esi, edi;
|
|
|
|
edx = 0;
|
|
while (edx < size) {
|
|
ebx = data[edx] ^ this->stream[0];
|
|
ebp = ((this->stream[(ebx >> 0x18) + 0x12] + this->stream[((ebx >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebx >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebx & 0xFF) + 0x312];
|
|
ebp = ebp ^ this->stream[1];
|
|
ebp ^= data[edx + 1];
|
|
edi = ((this->stream[(ebp >> 0x18) + 0x12] + this->stream[((ebp >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebp >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebp & 0xFF) + 0x312];
|
|
edi = edi ^ this->stream[2];
|
|
ebx = ebx ^ edi;
|
|
esi = ((this->stream[(ebx >> 0x18) + 0x12] + this->stream[((ebx >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebx >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebx & 0xFF) + 0x312];
|
|
ebp = ebp ^ esi ^ this->stream[3];
|
|
edi = ((this->stream[(ebp >> 0x18) + 0x12] + this->stream[((ebp >> 0x10) & 0xFF) + 0x112])
|
|
^ this->stream[((ebp >> 0x8) & 0xFF) + 0x212]) + this->stream[(ebp & 0xFF) + 0x312];
|
|
edi = edi ^ this->stream[4];
|
|
ebp = ebp ^ this->stream[5];
|
|
ebx = ebx ^ edi;
|
|
data[edx] = ebp;
|
|
data[edx + 1] = ebx;
|
|
edx += 2;
|
|
}
|
|
}
|
|
|
|
void PSOBBEncryption::skip(size_t) { }
|
|
|
|
PSOBBEncryption::PSOBBEncryption(
|
|
const KeyFile& key, const void* original_seed, size_t seed_size)
|
|
: stream(this->generate_stream(key, original_seed, seed_size)) { }
|
|
|
|
PSOBBEncryption::PSOBBEncryption() { }
|
|
|
|
|
|
|
|
vector<uint32_t> PSOBBEncryption::generate_stream(
|
|
const KeyFile& key, const void* original_seed, size_t seed_size) {
|
|
if (seed_size % 3) {
|
|
throw invalid_argument("seed size must be divisible by 3");
|
|
}
|
|
|
|
vector<uint32_t> stream(BB_STREAM_LENGTH, 0);
|
|
|
|
// Note: This part is done in the 03 command handler in the BB client, and
|
|
// isn't actually part of the encryption library. (Why did they do this?)
|
|
string seed;
|
|
const uint8_t* original_seed_data = reinterpret_cast<const uint8_t*>(
|
|
original_seed);
|
|
for (size_t x = 0; x < seed_size; x += 3) {
|
|
seed.push_back(original_seed_data[x] ^ 0x19);
|
|
seed.push_back(original_seed_data[x + 1] ^ 0x16);
|
|
seed.push_back(original_seed_data[x + 2] ^ 0x18);
|
|
}
|
|
|
|
if (key.is_modcrypt) {
|
|
for (size_t x = 0; x < 0x12; x++) {
|
|
uint8_t a = key.initial_keys[4 * x + 0];
|
|
uint8_t b = key.initial_keys[4 * x + 1];
|
|
uint8_t c = key.initial_keys[4 * x + 2];
|
|
uint8_t d = key.initial_keys[4 * x + 3];
|
|
stream[x] = ((a ^ d) << 24) | ((b ^ c) << 16) | (a << 8) | b;
|
|
}
|
|
memcpy(stream.data() + 0x12, &key.private_keys, sizeof(key.private_keys));
|
|
|
|
} else {
|
|
memcpy(stream.data(), &key, sizeof(key));
|
|
}
|
|
|
|
// This block was formerly postprocess_initial_stream
|
|
{
|
|
uint32_t eax, ecx, edx, ebx, ebp, esi, edi, ou, x;
|
|
|
|
ecx = 0;
|
|
ebx = 0;
|
|
|
|
while (ebx < 0x12) {
|
|
ebp = static_cast<uint32_t>(seed[ecx]) << 0x18;
|
|
eax = ecx + 1;
|
|
edx = eax % seed.size();
|
|
eax = (static_cast<uint32_t>(seed[edx]) << 0x10) & 0x00FF0000;
|
|
ebp = (ebp | eax) & 0xFFFF00FF;
|
|
eax = ecx + 2;
|
|
edx = eax % seed.size();
|
|
eax = (static_cast<uint32_t>(seed[edx]) << 0x08) & 0x0000FF00;
|
|
ebp = (ebp | eax) & 0xFFFFFF00;
|
|
eax = ecx + 3;
|
|
ecx = ecx + 4;
|
|
edx = eax % seed.size();
|
|
eax = static_cast<uint32_t>(seed[edx]) & 0x000000FF;
|
|
ebp = ebp | eax;
|
|
eax = ecx;
|
|
edx = eax % seed.size();
|
|
stream[ebx] ^= ebp;
|
|
ecx = edx;
|
|
ebx++;
|
|
}
|
|
|
|
ebp = 0;
|
|
esi = 0;
|
|
ecx = 0;
|
|
edi = 0;
|
|
ebx = 0;
|
|
edx = 0x48;
|
|
|
|
while (edi < edx) {
|
|
esi = esi ^ stream[0];
|
|
eax = esi >> 0x18;
|
|
ebx = (esi >> 0x10) & 0xFF;
|
|
eax = stream[eax + 0x12] + stream[ebx + 0x112];
|
|
ebx = (esi >> 8) & 0xFF;
|
|
eax = eax ^ stream[ebx + 0x212];
|
|
ebx = esi & 0xFF;
|
|
eax = eax + stream[ebx + 0x312];
|
|
|
|
eax = eax ^ stream[1];
|
|
ecx = ecx ^ eax;
|
|
ebx = ecx >> 0x18;
|
|
eax = (ecx >> 0x10) & 0xFF;
|
|
ebx = stream[ebx + 0x12] + stream[eax + 0x112];
|
|
eax = (ecx >> 8) & 0xFF;
|
|
ebx = ebx ^ stream[eax + 0x212];
|
|
eax = ecx & 0xFF;
|
|
ebx = ebx + stream[eax + 0x312];
|
|
|
|
for (x = 0; x <= 5; x++) {
|
|
ebx = ebx ^ stream[(x * 2) + 2];
|
|
esi = esi ^ ebx;
|
|
ebx = esi >> 0x18;
|
|
eax = (esi >> 0x10) & 0xFF;
|
|
ebx = stream[ebx + 0x12] + stream[eax + 0x112];
|
|
eax = (esi >> 8) & 0xFF;
|
|
ebx = ebx ^ stream[eax + 0x212];
|
|
eax = esi & 0xFF;
|
|
ebx = ebx + stream[eax + 0x312];
|
|
|
|
ebx = ebx ^ stream[(x * 2) + 3];
|
|
ecx = ecx ^ ebx;
|
|
ebx = ecx >> 0x18;
|
|
eax = (ecx >> 0x10) & 0xFF;
|
|
ebx = stream[ebx + 0x12] + stream[eax + 0x112];
|
|
eax = (ecx >> 8) & 0xFF;
|
|
ebx = ebx ^ stream[eax + 0x212];
|
|
eax = ecx & 0xFF;
|
|
ebx = ebx + stream[eax + 0x312];
|
|
}
|
|
|
|
ebx = ebx ^ stream[14];
|
|
esi = esi ^ ebx;
|
|
eax = esi >> 0x18;
|
|
ebx = (esi >> 0x10) & 0xFF;
|
|
eax = stream[eax + 0x12] + stream[ebx + 0x112];
|
|
ebx = (esi >> 8) & 0xFF;
|
|
eax = eax ^ stream[ebx + 0x212];
|
|
ebx = esi & 0xFF;
|
|
eax = eax + stream[ebx + 0x312];
|
|
|
|
eax = eax ^ stream[15];
|
|
eax = ecx ^ eax;
|
|
ecx = eax >> 0x18;
|
|
ebx = (eax >> 0x10) & 0xFF;
|
|
ecx = stream[ecx + 0x12] + stream[ebx + 0x112];
|
|
ebx = (eax >> 8) & 0xFF;
|
|
ecx = ecx ^ stream[ebx + 0x212];
|
|
ebx = eax & 0xFF;
|
|
ecx = ecx + stream[ebx + 0x312];
|
|
|
|
ecx = ecx ^ stream[16];
|
|
ecx = ecx ^ esi;
|
|
esi = stream[17];
|
|
esi = esi ^ eax;
|
|
stream[(edi / 4)] = esi;
|
|
stream[(edi / 4)+1] = ecx;
|
|
edi = edi + 8;
|
|
}
|
|
|
|
eax = 0;
|
|
edx = 0;
|
|
ou = 0;
|
|
while (ou < 0x1000) {
|
|
edi = 0x48;
|
|
edx = 0x448;
|
|
|
|
while (edi < edx) {
|
|
esi = esi ^ stream[0];
|
|
eax = esi >> 0x18;
|
|
ebx = (esi >> 0x10) & 0xFF;
|
|
eax = stream[eax + 0x12] + stream[ebx + 0x112];
|
|
ebx = (esi >> 8) & 0xFF;
|
|
eax = eax ^ stream[ebx + 0x212];
|
|
ebx = esi & 0xFF;
|
|
eax = eax + stream[ebx + 0x312];
|
|
|
|
eax = eax ^ stream[1];
|
|
ecx = ecx ^ eax;
|
|
ebx = ecx >> 0x18;
|
|
eax = (ecx >> 0x10) & 0xFF;
|
|
ebx = stream[ebx + 0x12] + stream[eax + 0x112];
|
|
eax = (ecx >> 8) & 0xFF;
|
|
ebx = ebx ^ stream[eax + 0x212];
|
|
eax = ecx & 0xFF;
|
|
ebx = ebx + stream[eax + 0x312];
|
|
|
|
for (x = 0; x <= 5; x++) {
|
|
ebx = ebx ^ stream[(x * 2) + 2];
|
|
esi = esi ^ ebx;
|
|
ebx = esi >> 0x18;
|
|
eax = (esi >> 0x10) & 0xFF;
|
|
ebx = stream[ebx + 0x12] + stream[eax + 0x112];
|
|
eax = (esi >> 8) & 0xFF;
|
|
ebx = ebx ^ stream[eax + 0x212];
|
|
eax = esi & 0xFF;
|
|
ebx = ebx + stream[eax + 0x312];
|
|
|
|
ebx = ebx ^ stream[(x * 2) + 3];
|
|
ecx = ecx ^ ebx;
|
|
ebx = ecx >> 0x18;
|
|
eax = (ecx >> 0x10) & 0xFF;
|
|
ebx = stream[ebx + 0x12] + stream[eax + 0x112];
|
|
eax = (ecx >> 8) & 0xFF;
|
|
ebx = ebx ^ stream[eax + 0x212];
|
|
eax = ecx & 0xFF;
|
|
ebx = ebx + stream[eax + 0x312];
|
|
}
|
|
|
|
ebx = ebx ^ stream[14];
|
|
esi = esi ^ ebx;
|
|
eax = esi >> 0x18;
|
|
ebx = (esi >> 0x10) & 0xFF;
|
|
eax = stream[eax + 0x12] + stream[ebx + 0x112];
|
|
ebx = (esi >> 8) & 0xFF;
|
|
eax = eax ^ stream[ebx + 0x212];
|
|
ebx = esi & 0xFF;
|
|
eax = eax + stream[ebx + 0x312];
|
|
|
|
eax = eax ^ stream[15];
|
|
eax = ecx ^ eax;
|
|
ecx = eax >> 0x18;
|
|
ebx = (eax >> 0x10) & 0xFF;
|
|
ecx = stream[ecx + 0x12] + stream[ebx + 0x112];
|
|
ebx = (eax >> 8) & 0xFF;
|
|
ecx = ecx ^ stream[ebx + 0x212];
|
|
ebx = eax & 0xFF;
|
|
ecx = ecx + stream[ebx + 0x312];
|
|
|
|
ecx = ecx ^ stream[16];
|
|
ecx = ecx ^ esi;
|
|
esi = stream[17];
|
|
esi = esi ^ eax;
|
|
stream[(ou / 4) + (edi / 4)] = esi;
|
|
stream[(ou / 4) + (edi / 4) + 1] = ecx;
|
|
edi = edi + 8;
|
|
}
|
|
ou = ou + 0x400;
|
|
}
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
|
|
|
|
|
|
PSOBBMultiKeyClientEncryption::PSOBBMultiKeyClientEncryption(
|
|
const vector<shared_ptr<const KeyFile>>& possible_keys,
|
|
const string& expected_first_data,
|
|
const void* seed,
|
|
size_t seed_size)
|
|
: possible_keys(possible_keys),
|
|
expected_first_data(expected_first_data),
|
|
seed(reinterpret_cast<const char*>(seed), seed_size) { }
|
|
|
|
void PSOBBMultiKeyClientEncryption::encrypt(void* data, size_t size, bool advance) {
|
|
if (this->stream.empty()) {
|
|
throw logic_error("PSOBB multi-key encryption requires client input first");
|
|
}
|
|
this->PSOBBEncryption::encrypt(data, size, advance);
|
|
}
|
|
|
|
void PSOBBMultiKeyClientEncryption::decrypt(void* data, size_t size, bool advance) {
|
|
if (this->stream.empty()) {
|
|
if (size != this->expected_first_data.size()) {
|
|
throw logic_error("initial decryption size does not match expected first data size");
|
|
}
|
|
|
|
for (const auto& key : this->possible_keys) {
|
|
this->active_key = key;
|
|
this->stream = PSOBBEncryption::generate_stream(
|
|
*this->active_key.get(), this->seed.data(), this->seed.size());
|
|
string test_data(reinterpret_cast<const char*>(data), size);
|
|
this->PSOBBEncryption::decrypt(test_data.data(), test_data.size());
|
|
if (test_data == this->expected_first_data) {
|
|
break;
|
|
}
|
|
this->active_key.reset();
|
|
this->stream.clear();
|
|
}
|
|
if (!this->active_key.get()) {
|
|
throw runtime_error("none of the registered private keys are valid for this client");
|
|
}
|
|
}
|
|
this->PSOBBEncryption::decrypt(data, size, advance);
|
|
}
|
|
|
|
PSOBBMultiKeyServerEncryption::PSOBBMultiKeyServerEncryption(
|
|
shared_ptr<const PSOBBMultiKeyClientEncryption> client_crypt,
|
|
const void* seed,
|
|
size_t seed_size)
|
|
: client_crypt(client_crypt),
|
|
seed(reinterpret_cast<const char*>(seed), seed_size) { }
|
|
|
|
void PSOBBMultiKeyServerEncryption::encrypt(void* data, size_t size, bool advance) {
|
|
this->ensure_stream_ready();
|
|
this->PSOBBEncryption::encrypt(data, size, advance);
|
|
}
|
|
|
|
void PSOBBMultiKeyServerEncryption::decrypt(void* data, size_t size, bool advance) {
|
|
this->ensure_stream_ready();
|
|
this->PSOBBEncryption::decrypt(data, size, advance);
|
|
}
|
|
|
|
void PSOBBMultiKeyServerEncryption::ensure_stream_ready() {
|
|
if (this->stream.empty()) {
|
|
if (!this->client_crypt->active_key.get()) {
|
|
throw logic_error("server crypt cannot be initialized because client crypt is not ready");
|
|
}
|
|
log(INFO, "[PSOBB/MK] Generating server stream");
|
|
this->stream = PSOBBEncryption::generate_stream(
|
|
*this->client_crypt->active_key, this->seed.data(), this->seed.size());
|
|
}
|
|
}
|