mirror of
https://github.com/iriselia/xgmsv.git
synced 2025-04-03 14:28:26 +08:00
.
This commit is contained in:
parent
cd2b367e25
commit
64b2793519
@ -17,8 +17,8 @@ namespace server
|
||||
marked_for_delete(false),
|
||||
parent(nullptr),
|
||||
socket(std::move(new_socket)),
|
||||
rbuffer(1024),
|
||||
wbuffer(1024),
|
||||
rbuffer(4096),
|
||||
wbuffer(4096),
|
||||
rqueue(this),
|
||||
wqueue(this)
|
||||
{
|
||||
@ -55,6 +55,8 @@ namespace server
|
||||
uint32 id = connection->id;
|
||||
if (connection->marked_for_delete)
|
||||
{
|
||||
|
||||
/*
|
||||
if (connection->parent)
|
||||
{
|
||||
connection->parent->remove_connection(connection);
|
||||
@ -64,6 +66,7 @@ namespace server
|
||||
connection = nullptr;
|
||||
this->conneciton_count--;
|
||||
connections_purged++;
|
||||
*/
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -32,7 +32,7 @@ namespace server
|
||||
|
||||
frame_data() : wpos(0), rpos(0), wsize(0), packet_count(0)
|
||||
{
|
||||
data.resize(1024);
|
||||
data.resize(4096);
|
||||
packets.resize(128);
|
||||
}
|
||||
};
|
||||
@ -199,7 +199,8 @@ namespace server
|
||||
{
|
||||
memcpy(wptr, source, size);
|
||||
rpos = wpos;
|
||||
wpos += size;
|
||||
rend += size;
|
||||
wpos = rend;
|
||||
return wptr;
|
||||
}
|
||||
|
||||
|
@ -15,6 +15,85 @@
|
||||
#pragma warning(push)
|
||||
#pragma warning(disable : 26444)
|
||||
|
||||
BOOL utf8ToGBK(char* str, int size)
|
||||
{
|
||||
char temp[2048];
|
||||
|
||||
// UTF-8 to UTF-16
|
||||
int size_needed = MultiByteToWideChar(CP_UTF8, 0, str, size, NULL, 0);
|
||||
MultiByteToWideChar(CP_UTF8, 0, str, size, (LPWSTR)temp, size_needed);
|
||||
|
||||
if (size_needed > 1024)
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// UTF-16 to Big5(950)
|
||||
int size_needed2 = WideCharToMultiByte(936, 0, (LPCWCH)temp, size_needed, NULL, 0, NULL, NULL);
|
||||
WideCharToMultiByte(936, 0, (LPCWCH)temp, size_needed, str, size, NULL, NULL);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
BOOL GBKToUtf8(char* str, int size)
|
||||
{
|
||||
char temp[2048];
|
||||
|
||||
// UTF-8 to UTF-16
|
||||
int size_needed = MultiByteToWideChar(936, 0, str, size, NULL, 0);
|
||||
MultiByteToWideChar(936, 0, str, size, (LPWSTR)temp, size_needed);
|
||||
|
||||
if (size_needed > 1024)
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// UTF-16 to Big5(950)
|
||||
int size_needed2 = WideCharToMultiByte(CP_UTF8, 0, (LPCWCH)temp, size_needed, NULL, 0, NULL, NULL);
|
||||
WideCharToMultiByte(CP_UTF8, 0, (LPCWCH)temp, size_needed, str, size, NULL, NULL);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
BOOL utf8ToBig5(char* str, int size)
|
||||
{
|
||||
char temp[2048];
|
||||
|
||||
// UTF-8 to UTF-16
|
||||
int size_needed = MultiByteToWideChar(CP_UTF8, 0, str, size, NULL, 0);
|
||||
MultiByteToWideChar(CP_UTF8, 0, str, size, (LPWSTR)temp, size_needed);
|
||||
|
||||
if (size_needed > 1024)
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// UTF-16 to Big5(950)
|
||||
int size_needed2 = WideCharToMultiByte(950, 0, (LPCWCH)temp, size_needed, NULL, 0, NULL, NULL);
|
||||
WideCharToMultiByte(950, 0, (LPCWCH)temp, size_needed, str, size, NULL, NULL);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
BOOL big5ToUtf8(char* str, int size)
|
||||
{
|
||||
char temp[2048];
|
||||
// UTF-8 to UTF-16
|
||||
int size_needed = MultiByteToWideChar(950, 0, str, size, NULL, 0);
|
||||
MultiByteToWideChar(950, 0, str, size, (LPWSTR)temp, size_needed);
|
||||
|
||||
if (size_needed > 1024)
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// UTF-16 to Big5(950)
|
||||
int size_needed2 = WideCharToMultiByte(CP_UTF8, 0, (LPCWCH)temp, size_needed, NULL, 0, NULL, NULL);
|
||||
WideCharToMultiByte(CP_UTF8, 0, (LPCWCH)temp, size_needed, str, size, NULL, NULL);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
namespace mmo_server
|
||||
{
|
||||
using namespace server;
|
||||
@ -93,13 +172,23 @@ namespace mmo_server
|
||||
});
|
||||
|
||||
//char packet4[] = "MLVlll6KdKyJmGKKII5pnmuNyMUp31qWQ7QqzcfGII5pnmuNN8X5wcMqzCuyJMAX2";
|
||||
char packet[] = "CmMMlxipiG42g1E";
|
||||
//char packet[] = "CmMMlxipiG42g1E";
|
||||
//crossgate::decrypt_message(packet, sizeof(packet));
|
||||
//char packet2[] = R"(CharList 0 jason|0\\z241400\\z1\\z15\\z0\\z0\\z0\\z15\\z0\\z0\\z100\\z0\\z2\\zjason\\z2\\zÓÎÃñ\\z2\\z106002\\z-1| )";
|
||||
|
||||
char packet2[] = "Echo nr ";
|
||||
//char packet[] = "ClientLogin 0 block ";
|
||||
//char packet3[] = "FC ";
|
||||
|
||||
//crossgate::encrypt_message(packet3, sizeof(packet2) - 1);
|
||||
//char packet[] = "QWHrI4t3y0x4gnZO+XBw+LGL7XYV3o5Fi98-507yuCnzU0T7YZRHP8OxQsbzdM9K-6fC9EnlhJ-eRRlhU0d4ZsWJzgBDs4Z1WPkbKs2kDNb47dw8Ff8w0tH3WMcheib0TxHZ1t0ZOrch9ID2nvd7xiZzTZHagPNmLT8RrtGLGXNAOzPph-w3eSC-HQps2DIekTuaxcYng1vgpYIgAZ8Tj9P2YjLMtU+Szf-1adkDOAcQstXdz3TukE85NMHYWfoHTv0lyhYeUk1qechP9j6FSj11vvX8HPEcvJyNLVhJI-o35jBF64unpAR+XsY1zfn3-ElvdsB7ZgXZDk7lkXoNQ6sbjmB1ath9brzXbJLzXdQb3mmDrRQ3F+4fjkiwoGn8hhzuxgLUb9BlXRc9PafSzn+C0HI5cSHLfyyoBI+41G5gQOr9-xzuxlPUb9BlXT49PaeDzn8C0mBExTCAagCAI8ZLc7W1qX5Rs0KATXAAO2Mc53Ip440JozG443chGFi-cBHklsg4kfAhEMIAyNOBlHrh3Ydz1os-c4sE-JDF9BDkhJSdGoN5LuLL3gopnMvN4DEQK7OevOQIGiO-0m3jc8t8bLhQWvUi4nse6zz7UwzdPN2wM9j95maFs6vA2EmW81VvOB5LoLDXMgIwWsE5FA44OR13216pPgyv2Lyxn6PUZLQoWYgjfYic9GY2c96MInI+OEVvSmqv3ccSysTev-4FxLjNpQDZDtG7wVyQRr3GZDBaHLFujbmI6vbvR6HeRCWzasfEcSa8gsNpZdlRC58Ce2e2JwSCdt+m50g8r22PSYCArfE8Gh87Wr3nnO4yQJSYNzBzIWV4TjEfhUwSefAmA5yd0oDzp0Ty4lHPQE9xEBJxfP1+eNGzQnwd5B1RSlWUifBr-j4A64GteBZPRZJy8Js5aQX4uJgWYCoVveOmZTAFfxWpr-cVJg3WYz6knLrpnUeFja544skgZILFG5+BKeBoUB42hoNjIeAlNwoHBRJemHwL5HPIzRWSW+9LOcHk1Z61KRU9sSBlq+oC-lf9ArCI7rk1U1v1GxngdFR+salmFfcMLX50HmY8NuQW5K9jjsUkFFan";
|
||||
char packet[] = "s7Oar2m1BIPYpdaZ1eXOVKDYpotoD5GWg-i3gjin0a3ddKFVX-TgJZBGp2PRbDC0VavRTa08l9drNWsItjIDQA";
|
||||
core::byte_buffer buf(4096);
|
||||
buf.write(packet, sizeof(packet) - 1);
|
||||
crossgate::decrypt_message(buf);
|
||||
//crossgate::encrypt_message(buf);
|
||||
|
||||
//big5ToUtf8((char*)buf.data(), buf.wpos);
|
||||
|
||||
//printf("string:%s", buf.data());
|
||||
|
||||
core::async_every(1000ms, []()
|
||||
{
|
||||
|
@ -5,10 +5,66 @@ namespace server
|
||||
namespace crossgate
|
||||
{
|
||||
#define SEED 4595
|
||||
#define MESSAGE_SIZE(a) sizeof(a)/sizeof(char)
|
||||
|
||||
char mapping_table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+-";
|
||||
|
||||
|
||||
char char9_str[] = {
|
||||
0x43, 0xd0, 0x84, 0x91, 0xc3, 0x24, 0xcd, 0x1c,
|
||||
0x3a, 0x20, 0x60, 0x80, 0x50, 0x13, 0x66, 0xce,
|
||||
0x1b, 0x37, 0x7c, 0x60, 0x70, 0xe1, 0xa2, 0x47,
|
||||
0x06, 0x8d, 0x18, 0x34, 0x60, 0x48, 0xa4, 0x18,
|
||||
0x63, 0xa2, 0x9e, 0x18, 0x35, 0x3c, 0x6e, 0xd4,
|
||||
0x33, 0x72, 0x24, 0x48, 0x91, 0x28, 0x39, 0x6a,
|
||||
0x4c, 0x59, 0xd1, 0x23, 0x43, 0x87, 0x6e, 0x3c,
|
||||
0xca, 0xf0, 0x38, 0xcd, 0xd9, 0xb0, 0x78, 0x32,
|
||||
0x3d, 0xc6, 0x80, 0x61, 0x43, 0xe3, 0x4c, 0x8a,
|
||||
0x2d, 0x62, 0xf0, 0x01, 0x01, 0x10, 0x00
|
||||
};
|
||||
|
||||
|
||||
int weird_kernel1[] =
|
||||
{
|
||||
// 0 1 2 3 4 5 6 7 8
|
||||
0x00, 0x01, 0x03, 0x07, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF, // 0
|
||||
0x00, 0x03, 0x07, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF, 0xFF, // 1
|
||||
0x00, 0x07, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF, 0xFF, 0xFF, // 2
|
||||
0x00, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, // 3
|
||||
0x00, 0x1F, 0x3F, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, // 4
|
||||
0x00, 0x3F, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, // 5
|
||||
0x00, 0x7F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, // 6
|
||||
0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF // 7
|
||||
};
|
||||
|
||||
int weird_kernel2[] =
|
||||
{
|
||||
// 0 1 2 3 4 5 6 7 8
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // 0
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, // 1
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x03, // 2
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x03, 0x07, // 3
|
||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x03, 0x07, 0x0F, // 4
|
||||
0x00, 0x00, 0x00, 0x00, 0x01, 0x03, 0x07, 0x0F, 0x1F, // 5
|
||||
0x00, 0x00, 0x00, 0x01, 0x03, 0x07, 0x0F, 0x1F, 0x3F, // 6
|
||||
0x00, 0x00, 0x01, 0x03, 0x07, 0x0F, 0x1F, 0x3F, 0x7F // 7
|
||||
};
|
||||
|
||||
// odd length encryption
|
||||
struct dictionary_entry
|
||||
{
|
||||
unsigned char9_left : 9;
|
||||
unsigned : 7;
|
||||
unsigned char9_right : 9;
|
||||
unsigned : 7;
|
||||
|
||||
unsigned prev : 9;
|
||||
unsigned : 23;
|
||||
};
|
||||
|
||||
|
||||
// decrypt stage 1
|
||||
int util_64to256(char* dst, const char* src, char* table, uint64 length)
|
||||
int util_64to256(uint8* dst, const uint8* src, char* table, uint64 length)
|
||||
{
|
||||
unsigned int dw, dwcounter, i;
|
||||
char* ptr = NULL;
|
||||
@ -34,7 +90,7 @@ namespace server
|
||||
}
|
||||
|
||||
// decrypt stage 2
|
||||
uint64 remove_salt(char* dst, const char* src, uint64 length)
|
||||
uint64 remove_salt(uint8* dst, const uint8* src, uint64 length)
|
||||
{
|
||||
// remove conditional salt
|
||||
int seed = SEED % length; //38 when message_size = 49
|
||||
@ -60,7 +116,8 @@ namespace server
|
||||
return length - 1;
|
||||
}
|
||||
|
||||
void remove_conditional_bit_reverse(char* dst, const char* src, uint64 length)
|
||||
// decrypt stage 3
|
||||
void remove_conditional_bit_reverse(uint8* dst, const uint8* src, uint64 length)
|
||||
{
|
||||
// remove conditional bit reverse, seed = 4595
|
||||
for (int i = 0; i < (length + 1); i++) // +1 for checksum
|
||||
@ -78,49 +135,351 @@ namespace server
|
||||
}
|
||||
}
|
||||
|
||||
std::string decrypt_message_str(std::string packet)
|
||||
// decrypt stage 4
|
||||
uint64 remove_conditional_compression(uint8* dst, uint64 dst_size, const uint8* src, uint64 length)
|
||||
{
|
||||
std::string stage_1(packet.length(), 0);
|
||||
int read_counter = 0;
|
||||
const uint8* read_buffer = 0;
|
||||
int read_length = 0;
|
||||
|
||||
// 6 bit to 8 bit string
|
||||
int size = util_64to256(stage_1.data(), packet.c_str(), mapping_table, packet.length());
|
||||
// important trim right here
|
||||
stage_1 = stage_1.substr(0, size - 1);
|
||||
auto read_bits = [&read_counter, &read_buffer, &read_length](int input) -> int
|
||||
{
|
||||
int mod8 = read_counter % 8; // row 0 1 2 3 4 5 6 7
|
||||
int div8 = read_counter / 8; // column 0 1 2 3 4 5 6 7 8
|
||||
|
||||
std::string stage_2(stage_1.length(), 0);
|
||||
size = remove_salt(stage_2.data(), stage_1.c_str(), stage_1.length());
|
||||
stage_2 = stage_2.substr(0, strlen(stage_2.data()));
|
||||
// valid input: 1 2 3 4 5 6 7 8
|
||||
if (div8 >= read_length) // || input < 1 || input > 8) unnecessary for now
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
std::string stage_3(stage_2.length(), 0);
|
||||
remove_conditional_bit_reverse(stage_3.data(), stage_2.c_str(), stage_2.length());
|
||||
stage_3 = stage_3.substr(1, strlen(stage_3.data()) - 1);
|
||||
int index = input + mod8 * 9;
|
||||
|
||||
int lo = (*(read_buffer + div8) & weird_kernel1[index]) >> mod8;
|
||||
int hi = (*(read_buffer + div8 + 1) & weird_kernel2[index]) << (8 - mod8);
|
||||
|
||||
read_counter += input;
|
||||
|
||||
return hi | lo;
|
||||
|
||||
};
|
||||
|
||||
auto read_9bits = [&read_bits](int input = 9) -> int
|
||||
{
|
||||
int lo = read_bits(8); // row + 1
|
||||
int hi = read_bits(1) << 8; // column + 1
|
||||
return hi + lo;
|
||||
};
|
||||
|
||||
|
||||
return stage_3;
|
||||
//input variables
|
||||
read_counter = 0;
|
||||
read_buffer = src;
|
||||
read_length = length; // message_size - 1
|
||||
|
||||
uint8 dictionary_indices[512];
|
||||
dictionary_entry dictionary_entries[256];
|
||||
uint8 dictionary_size = 0;
|
||||
|
||||
memset(dictionary_indices, 0, 512);
|
||||
memset(dictionary_entries, 0, sizeof(dictionary_entry) * 256);
|
||||
memset(dst, 0, dst_size);
|
||||
dictionary_size = 0;
|
||||
|
||||
uint16 next_char9 = 0;
|
||||
uint16 curr_char9 = 0;// packet2_6[0];
|
||||
unsigned char max_dictionary_size = 0;
|
||||
unsigned char curr_entry_index = 0;
|
||||
char char_stack[512];
|
||||
unsigned int char_stack_size = 0;
|
||||
memset(char_stack, 0, 512);
|
||||
|
||||
int bytes_decoded = 0;
|
||||
|
||||
for (int index = 0; index < length; index++)
|
||||
{
|
||||
int bits_read = read_9bits(9);
|
||||
int last_char_or_something = bits_read;
|
||||
|
||||
if (bits_read == 0x100)
|
||||
{
|
||||
break; // EOF for extended ascii
|
||||
}
|
||||
|
||||
//dictionary_size = max_dictionary_size;
|
||||
|
||||
if (bits_read < (dictionary_size + 1 + 0x100))
|
||||
{
|
||||
next_char9 = bits_read;
|
||||
}
|
||||
else
|
||||
{
|
||||
char_stack[char_stack_size++] = next_char9;
|
||||
if (char_stack_size >= 512)
|
||||
{
|
||||
assert(false);
|
||||
}
|
||||
next_char9 = curr_char9;
|
||||
}
|
||||
|
||||
if (next_char9 > 0x100) // if extended ascii
|
||||
{
|
||||
while ((unsigned int)next_char9 < 512)
|
||||
{
|
||||
const dictionary_entry& curr_entry = dictionary_entries[next_char9 - 0x101];
|
||||
|
||||
char_stack[char_stack_size++] = curr_entry.char9_right;
|
||||
next_char9 = curr_entry.char9_left;
|
||||
if (char_stack_size >= 512)
|
||||
{
|
||||
break;
|
||||
printf("error\n");
|
||||
printf("so far:%s", dst);
|
||||
assert(false);
|
||||
}
|
||||
if (next_char9 <= 256)
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
char_stack[char_stack_size++] = next_char9;
|
||||
if (char_stack_size >= 512)
|
||||
{
|
||||
assert(false);
|
||||
//return -1;
|
||||
}
|
||||
|
||||
if (char_stack_size)
|
||||
{
|
||||
int increment = char_stack_size - 1;
|
||||
|
||||
while (bytes_decoded < dst_size && char_stack_size)
|
||||
{
|
||||
dst[bytes_decoded] = char_stack[char_stack_size - 1];
|
||||
bytes_decoded++;
|
||||
char_stack_size--;
|
||||
}
|
||||
|
||||
if (char_stack_size)
|
||||
{
|
||||
//error
|
||||
assert(false);
|
||||
break;
|
||||
}
|
||||
|
||||
//dictionary_size = max_dictionary_size;
|
||||
bits_read = last_char_or_something;
|
||||
}
|
||||
|
||||
if (bytes_decoded > 1 && dictionary_size < 256)
|
||||
{
|
||||
auto& new_entry = dictionary_entries[dictionary_size];
|
||||
new_entry.char9_left = curr_char9;
|
||||
new_entry.char9_right = next_char9;
|
||||
new_entry.prev = dictionary_size;
|
||||
|
||||
bits_read = last_char_or_something;
|
||||
dictionary_indices[curr_char9] = dictionary_size++;
|
||||
}
|
||||
curr_char9 = bits_read;
|
||||
}
|
||||
|
||||
return bytes_decoded;
|
||||
}
|
||||
|
||||
void decrypt_message(core::byte_buffer& payload)
|
||||
{
|
||||
char* buffer = (char*)payload.data() + payload.rpos;
|
||||
uint8* buffer = payload.data() + payload.rpos;
|
||||
|
||||
uint64 payload_size = payload.rend - payload.rpos;
|
||||
|
||||
// 6 bit to 8 bit string
|
||||
uint64 descrypted_size = util_64to256(buffer, buffer, mapping_table, payload_size);
|
||||
uint64 decrypted_size = util_64to256(buffer, buffer, mapping_table, payload_size);
|
||||
|
||||
descrypted_size--; // -1 for checksum
|
||||
decrypted_size--; // -1 for checksum
|
||||
|
||||
descrypted_size = remove_salt(buffer, buffer, descrypted_size);
|
||||
remove_conditional_bit_reverse(buffer, buffer, descrypted_size);
|
||||
decrypted_size = remove_salt(buffer, buffer, decrypted_size);
|
||||
remove_conditional_bit_reverse(buffer, buffer, decrypted_size);
|
||||
|
||||
buffer[descrypted_size] = '\0';
|
||||
uint8 workbuf[4096];
|
||||
if (buffer[0] % 2)
|
||||
{
|
||||
decrypted_size = remove_conditional_compression(workbuf, sizeof(workbuf), buffer + 1, decrypted_size);
|
||||
memcpy(buffer, workbuf, decrypted_size);
|
||||
}
|
||||
|
||||
buffer[decrypted_size] = '\0';
|
||||
|
||||
buffer++;
|
||||
payload.rpos++;
|
||||
payload.rend += descrypted_size - payload_size;
|
||||
payload.rend += decrypted_size - payload_size;
|
||||
}
|
||||
|
||||
uint8 apply_conditional_bit_reverse(char* dst, const char* src, uint64 length)
|
||||
uint64 apply_conditional_compression(uint8* dst, uint64 dst_size, const uint8* src, uint64 length)
|
||||
{
|
||||
int write_counter = 0;
|
||||
uint8* write_buffer = 0;
|
||||
int write_length = 0;
|
||||
|
||||
auto write_bits = [&write_counter, &write_buffer, &write_length](int num_bits, char input) -> int
|
||||
{
|
||||
int mod8 = write_counter % 8; // row 0 1 2 3 4 5 6 7
|
||||
int div8 = write_counter / 8; // column 0 1 2 3 4 5 6 7 8
|
||||
int div8_plus_1 = div8 + 1;
|
||||
// valid input: 1 2 3 4 5 6 7 8
|
||||
if (write_length < div8) // || input < 1 || input > 8) unnecessary for now
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
int index = num_bits + mod8 * 9;
|
||||
|
||||
*(write_buffer + div8) &= weird_kernel1[index];
|
||||
*(write_buffer + div8) |= weird_kernel1[index] & (input << mod8);
|
||||
*(write_buffer + div8 + 1) &= weird_kernel2[index];
|
||||
*(write_buffer + div8 + 1) |= weird_kernel2[index] & (input >> (8 - mod8));
|
||||
|
||||
write_counter += num_bits;
|
||||
|
||||
return div8 + 1;
|
||||
|
||||
};
|
||||
|
||||
auto write_9bits = [&write_bits](int num_bits, int input) -> int
|
||||
{
|
||||
int result = 0;
|
||||
if (write_bits(8, (char)(input & 0xFF)) >= 0) // encode_magic_number(8, bits) -> encode_magic_number(1, bits)
|
||||
{
|
||||
result = write_bits(1, (char)((input >> 8) & 0x1));
|
||||
if (result < 0)
|
||||
{
|
||||
result = -1;
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
};
|
||||
|
||||
unsigned char dictionary_indices[512];
|
||||
dictionary_entry dictionary_entries[256];
|
||||
unsigned char dictionary_size = 0;
|
||||
|
||||
memset(dictionary_indices, 0, 512);
|
||||
memset(dictionary_entries, 0, sizeof(dictionary_entry) * 256);
|
||||
dictionary_size = 0;
|
||||
|
||||
unsigned curr_char9;
|
||||
unsigned next_char9;
|
||||
|
||||
int bytes_encoded = 0;
|
||||
|
||||
write_buffer = dst;
|
||||
write_length = length;
|
||||
write_counter = 0;
|
||||
memset(dst, 0, dst_size);
|
||||
|
||||
//printf("Compression:\n");
|
||||
|
||||
curr_char9 = src[0];
|
||||
for (int index = 1; index < length + 1; index++)
|
||||
{
|
||||
if (index < length - 1)
|
||||
{
|
||||
next_char9 = src[index] & 0xFF; // 0xFF required because of visual studio bug
|
||||
}
|
||||
else
|
||||
{
|
||||
next_char9 = 0x100; // EOF in 9-bit extended ascii
|
||||
}
|
||||
|
||||
// Is there a dictionary entry?
|
||||
unsigned char head_entry_index = dictionary_indices[curr_char9];
|
||||
dictionary_entry* head_entry = &dictionary_entries[head_entry_index];
|
||||
|
||||
|
||||
unsigned char curr_entry_index = head_entry_index;
|
||||
|
||||
while (curr_entry_index)
|
||||
{
|
||||
const dictionary_entry* curr_entry = &dictionary_entries[curr_entry_index];
|
||||
|
||||
if (curr_entry->char9_right == next_char9)
|
||||
{
|
||||
if (curr_entry_index > 0)
|
||||
{
|
||||
if (index != length + 1)
|
||||
{
|
||||
// bytes_encoded = bytes_encoded_old;
|
||||
curr_char9 = curr_entry_index + 0x101; // 0x101 is the offset to make extended 9 bit ascii
|
||||
goto skip_char;
|
||||
}
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
assert(false);
|
||||
//printf("error");
|
||||
}
|
||||
}
|
||||
|
||||
curr_entry_index = curr_entry->prev;
|
||||
}
|
||||
char out_char = '?';
|
||||
|
||||
if (curr_char9 < 0x7F)
|
||||
{
|
||||
out_char = curr_char9;
|
||||
}
|
||||
|
||||
if ((index - 1) && !((index - 1) % 8))
|
||||
{
|
||||
//printf("\n");
|
||||
}
|
||||
|
||||
/*
|
||||
char char9_str[] = {
|
||||
43, d0, 84, 91, c3, 24, cd, 1c,
|
||||
3a, 20, 60, 80, 50, 13, 66, ce,
|
||||
1b, 37, 7c, 60, 70, e1, a2, 47,
|
||||
06, 8d, 18, 34, 60, 48, a4, 18,
|
||||
63, a2, 9e, 18, 35, 3c, 6e, d4,
|
||||
33, 72, 24, 48, 91, 28, 39, 6a,
|
||||
4c, 59, d1, 23, 43, 87, 6e, 3c,
|
||||
ca, f0, 38, cd, d9, b0, 78, 32,
|
||||
3d, c6, 80, 61, 43, e3, 4c, 8a,
|
||||
2d, 62, f0, 01, 01, 10, 00
|
||||
};
|
||||
*/
|
||||
//printf("|%c:0x%03x, ", out_char, curr_char9);
|
||||
bytes_encoded = write_9bits(9, curr_char9);
|
||||
//bytes_encoded_old = bytes_encoded;
|
||||
|
||||
//if (bytes_encoded > buffer_size)
|
||||
// return -1;
|
||||
|
||||
if (dictionary_size < 256)
|
||||
{
|
||||
auto& new_entry = dictionary_entries[dictionary_size];
|
||||
new_entry.char9_left = curr_char9;
|
||||
new_entry.char9_right = next_char9;
|
||||
new_entry.prev = head_entry_index;
|
||||
|
||||
dictionary_indices[curr_char9] = dictionary_size++;
|
||||
}
|
||||
|
||||
curr_char9 = next_char9;
|
||||
skip_char:
|
||||
if (index == length + 1)
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return bytes_encoded;
|
||||
}
|
||||
|
||||
uint8 apply_conditional_bit_reverse(uint8* dst, const uint8* src, uint64 length)
|
||||
{
|
||||
char sum = 0;
|
||||
|
||||
@ -138,7 +497,7 @@ namespace server
|
||||
return sum;
|
||||
}
|
||||
|
||||
int apply_salt_and_add_checksum(char* dst, const char* src, uint64 length, uint8 checksum)
|
||||
int apply_salt_and_add_checksum(uint8* dst, const uint8* src, uint64 length, uint8 checksum)
|
||||
{
|
||||
// conditional salt
|
||||
int seed = SEED % length; //38 when message_size = 49
|
||||
@ -165,11 +524,11 @@ namespace server
|
||||
return 0;
|
||||
}
|
||||
|
||||
int util_256to64(char *dst, char *src, char* table, uint64 length)
|
||||
int util_256to64(uint8* dst, uint8* src, char* table, uint64 length)
|
||||
{
|
||||
uint32 byte_counter, i;
|
||||
uint32 byte_counter;
|
||||
|
||||
uint16 reg;
|
||||
uint16 reg, i;
|
||||
|
||||
if (!dst || !src || !table) return 0;
|
||||
reg = 0;
|
||||
@ -193,7 +552,7 @@ namespace server
|
||||
|
||||
void encrypt_message(core::byte_buffer& payload)
|
||||
{
|
||||
char* buffer = (char*)payload.data() + payload.rpos;
|
||||
uint8* buffer = payload.data() + payload.rpos;
|
||||
const uint64 payload_size = payload.rend - payload.rpos;
|
||||
|
||||
uint64 encrypted_size = payload_size + 1; // packet_length padding
|
||||
@ -206,7 +565,6 @@ namespace server
|
||||
header = encrypted_size;
|
||||
}
|
||||
|
||||
assert(false);
|
||||
/*
|
||||
*(_BYTE*)message_work_buffer2 = checksum;
|
||||
message_length = odd_length_encrypt(
|
||||
@ -215,6 +573,25 @@ namespace server
|
||||
(unsigned __int8*)message,
|
||||
strlen(message));
|
||||
*/
|
||||
uint8 workbuf[4096];
|
||||
encrypted_size = apply_conditional_compression(workbuf + 1, sizeof(workbuf) - 1, buffer, encrypted_size);
|
||||
|
||||
/*
|
||||
char char9_str[] = {
|
||||
43, d0, 84, 91, c3, 24, cd, 1c,
|
||||
3a, 20, 60, 80, 50, 13, 66, ce,
|
||||
1b, 37, 7c, 60, 70, e1, a2, 47,
|
||||
06, 8d, 18, 34, 60, 48, a4, 18,
|
||||
63, a2, 9e, 18, 35, 3c, 6e, d4,
|
||||
33, 72, 24, 48, 91, 28, 39, 6a,
|
||||
4c, 59, d1, 23, 43, 87, 6e, 3c,
|
||||
ca, f0, 38, cd, d9, b0, 78, 32,
|
||||
3d, c6, 80, 61, 43, e3, 4c, 8a,
|
||||
2d, 62, f0, 01, 01, 10, 00
|
||||
};
|
||||
*/
|
||||
//workbuf[0] = header;
|
||||
memcpy(buffer, workbuf, encrypted_size + 1);
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -231,7 +608,7 @@ namespace server
|
||||
|
||||
encrypted_size++; // null terminator;
|
||||
|
||||
char workbuf[4096];
|
||||
uint8 workbuf[4096];
|
||||
uint8 checksum = apply_conditional_bit_reverse(buffer, buffer, encrypted_size);
|
||||
apply_salt_and_add_checksum(workbuf, buffer, encrypted_size, checksum);
|
||||
|
||||
|
@ -33,9 +33,89 @@ namespace server
|
||||
|
||||
void handle_client_login(xg_packet* packet)
|
||||
{
|
||||
enum status_code
|
||||
{
|
||||
success = 0,
|
||||
wrong_login_or_banned = 1,
|
||||
refuse = 2
|
||||
};
|
||||
|
||||
auto connection = packet->connection;
|
||||
|
||||
if (!connection->send_raw("ClientLogin 0 block "))
|
||||
{
|
||||
printf("handle_client_login failed.\n");
|
||||
}
|
||||
|
||||
printf("handle_client_login!\n");
|
||||
}
|
||||
|
||||
void handle_char_list(xg_packet* packet)
|
||||
{
|
||||
enum fields
|
||||
{
|
||||
name = 0,
|
||||
something = 1,
|
||||
portrait = 2,
|
||||
level = 3,
|
||||
vitality = 4,
|
||||
strength = 5,
|
||||
toughness = 6,
|
||||
quickness = 7,
|
||||
magic = 8,
|
||||
fire = 9,
|
||||
wind = 10,
|
||||
earth = 11,
|
||||
water = 12,
|
||||
};
|
||||
|
||||
auto connection = packet->connection;
|
||||
|
||||
if (!connection->send_raw(R"(CharList 0 jason|0\\z241400\\z2\\z15\\z0\\z0\\z0\\z15\\z0\\z0\\z100\\z0\\z2\\zjason\\z2\\z游民\\z2\\z106002\\z-1| )"))
|
||||
{
|
||||
printf("handle_char_list failed.\n");
|
||||
}
|
||||
|
||||
printf("handle_char_list!\n");
|
||||
}
|
||||
|
||||
void handle_char_login(xg_packet* packet)
|
||||
{
|
||||
auto connection = packet->connection;
|
||||
connection->send_raw(R"(LI 3I3 1JayvY 1 0 )");
|
||||
connection->send_raw(R"(CC 0 hV p J c 6 0 0 0 0 -1 )");
|
||||
connection->send_raw(R"(MN 城西医院\\z0 )");
|
||||
connection->send_raw(R"(CP 1|155|155|185|185|15|0|0|0|15|50|50|50|0|16|1|23|23|23|107|107|60|100|100|0|0|0|5000|0|jason|| )");
|
||||
connection->send_raw(R"(CP2 1|0|0|0|0|0|0|0|0|0|0|100|0|0|0|0|241400|10|100| )");
|
||||
connection->send_raw(R"(CJ 1 见习传教士 )");
|
||||
connection->send_raw(R"(CS 0|||||||||||1|||||||||||2|||||||||||3|||||||||||4|||||||||||5|||||||||||6|||||||||||7|||||||||||8|||||||||||9|||||||||||10|||||||||||11|||||||||||12|||||||||||13|||||||||||14||||||||||| )");
|
||||
connection->send_raw(R"(TITLE 敬畏的寂静|0|17|见习传教士|1|161||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||)");
|
||||
connection->send_raw(R"(I )");
|
||||
connection->send_raw(R"(EP 0 0 )");
|
||||
connection->send_raw(R"(KP 0 1|101321|4|93|93|99|99|1|16|1|48|40|34|101|100|24|0|40|60|0|10|1|新手红螳螂||0|0| )");
|
||||
connection->send_raw(R"(KP2 0 1|2|9|4|5|3|25|0|45|-10|0|0|0|0|0|0|1| )");
|
||||
connection->send_raw(R"(PT 0 0|7300|攻击|能以普通物理攻击给与打击|0|1|1141|0|1|7400|防御|能防守来自物理攻击的打击|0|1|72|1|2|407|气功弹\SLV8|给予对象前后列位置一体或数体的伤害,依等级改变攻击数。|40|1|117|2|3|1238|明净止水-Ⅴ|集中精神回复一定比例的体力,技 能和最大生命值越高回复比例上限越高(注意和回复力无关),使用后将无法闪躲物理攻击。\S|135|1|72|3|)");
|
||||
connection->send_raw(R"(FS 0 )");
|
||||
connection->send_raw(R"(MC 0 hV 0 0 p k bFX agh 0 )");
|
||||
connection->send_raw(R"(AB |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||)");
|
||||
connection->send_raw(R"(ABG |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||)");
|
||||
connection->send_raw(R"(C 1|3I3|12|6|0|106002|1|0|jason|\\z\\z|0|0|0|敬畏的寂静|0 )");
|
||||
connection->send_raw(R"(C 2|1tq|10|5|4|14088|1|0|伯舒医师|\\z\\z|0|0|0||0 37|1ty|10|17|4|14088|1|0|姆涅医师|\\z\\z|0|0|0||0 2|1ti|12|5|4|14090|1|0|实习药剂师吉可|\\z\\z|0|0|0||0 29|1xH|16|9|6|14151|1|0|实习生蜜雅|\\z\\z|0|0|0||0)");
|
||||
connection->send_raw(R"(BT 5Co 0 )");
|
||||
connection->send_raw(R"(POS 0 )");
|
||||
connection->send_raw(R"(AL 1 0|0|0|0|0|0|0|0|0|0| )");
|
||||
connection->send_raw(R"(IP 127.0.0.1 )");
|
||||
connection->send_raw(R"(MAC Y O )");
|
||||
connection->send_raw(R"(EF 0 0 )");
|
||||
connection->send_raw(R"(TK -1 P|感谢购买大灰狼魔力。wow335.taobao.com 4 0 )");
|
||||
connection->send_raw(R"(TK -1 P|[版本申明]GMSV\SAvaritia\SFeb\S\S1\S2014\S共享版 4 0 )");
|
||||
connection->send_raw(R"(TK -1 P|本服务端仅供研究使用,请勿用作商业用途。 4 0 )");
|
||||
connection->send_raw(R"(TK -1 P|项目主页\S&\S交流论坛:http://www.cgdev.me/ 4 0 )");
|
||||
connection->send_raw(R"(STK GA\SLua引擎运行正常。 )");
|
||||
connection->send_raw(R"(STK [二键魔力公告]可用\S/help\S指令查看当前可用LuaTalk指令 )");
|
||||
//connection->send_raw(R"()");
|
||||
}
|
||||
|
||||
struct opcode_entry
|
||||
{
|
||||
enum xg_opcode opcode;
|
||||
@ -49,6 +129,8 @@ namespace server
|
||||
XG_FC,
|
||||
XG_ECHO,
|
||||
XG_CLIENT_LOGIN,
|
||||
XG_CHAR_LIST,
|
||||
XG_CHAR_LOGIN,
|
||||
XG_OPCODE_COUNT
|
||||
};
|
||||
|
||||
@ -59,7 +141,9 @@ namespace server
|
||||
{
|
||||
add_opcode_entry(XG_FC, "FC", &handle_fc),
|
||||
add_opcode_entry(XG_ECHO, "Echo", &handle_echo),
|
||||
add_opcode_entry(XG_CLIENT_LOGIN, "ClientLogin", &handle_client_login)
|
||||
add_opcode_entry(XG_CLIENT_LOGIN, "ClientLogin", &handle_client_login),
|
||||
add_opcode_entry(XG_CHAR_LIST, "CharList", &handle_char_list),
|
||||
add_opcode_entry(XG_CHAR_LOGIN, "CharLogin", &handle_char_login),
|
||||
};
|
||||
|
||||
template<typename T, size_t N>
|
||||
@ -80,8 +164,11 @@ namespace server
|
||||
{
|
||||
auto handler = xg_opcode_table[i].handler;
|
||||
handler(&xg_packet);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
printf("Unhandled packet:%s\n", xg_packet.data);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user