forked from temisu/oneKpaq
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathAsmDecode.cpp
101 lines (84 loc) · 3.66 KB
/
AsmDecode.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
/* Copyright (C) Teemu Suutari */
#include <string.h> /* memcpy */
#include <vector>
#include <sys/mman.h>
#ifndef __linux__
#include <mach/vm_param.h>
#else
#include <unistd.h>
#define PAGE_SIZE (sysconf(_SC_PAGESIZE))
#endif
#include "onekpaq_common.h"
#include "AsmDecode.hpp"
#include "Timer.hpp"
extern "C" void *onekpaq_decompressor_mode1;
extern "C" void *onekpaq_decompressor_mode2;
extern "C" void *onekpaq_decompressor_mode3;
extern "C" void *onekpaq_decompressor_mode4;
extern "C" void *onekpaq_decompressor_mode1_shift;
extern "C" void *onekpaq_decompressor_mode2_shift;
extern "C" void *onekpaq_decompressor_mode3_shift;
extern "C" void *onekpaq_decompressor_mode4_shift;
extern "C" void *onekpaq_decompressor_mode1_end;
extern "C" void *onekpaq_decompressor_mode2_end;
extern "C" void *onekpaq_decompressor_mode3_end;
extern "C" void *onekpaq_decompressor_mode4_end;
// Please be PIC!
static void *makeCodeblock(const void *prog,ulong size)
{
// not executable yet!
void *ret=mmap(nullptr,(size+PAGE_SIZE-1)&~(PAGE_SIZE-1),PROT_READ|PROT_WRITE,MAP_ANON|MAP_PRIVATE,-1,0);
ASSERT(ret!=MAP_FAILED,"mmap failed");
memcpy(ret,prog,size);
return ret;
}
static void makeRunnable(void *ptr,ulong size)
{
mprotect(ptr,(size+PAGE_SIZE-1)&~(PAGE_SIZE-1),PROT_READ|PROT_EXEC);
}
std::vector<u8> AsmDecode(const std::vector<u8> &src1,const std::vector<u8> &src2,StreamCodec::EncodeMode mode,uint shift) {
typedef void (*AsmDecompressor)(void*,void*);
static void *decompressors[4]={
&onekpaq_decompressor_mode1,
&onekpaq_decompressor_mode2,
&onekpaq_decompressor_mode3,
&onekpaq_decompressor_mode4
};
static ulong decompressorShiftOffsets[4]={
(ulong)&onekpaq_decompressor_mode1_shift-(ulong)(void*)&onekpaq_decompressor_mode1,
(ulong)&onekpaq_decompressor_mode2_shift-(ulong)(void*)&onekpaq_decompressor_mode2,
(ulong)&onekpaq_decompressor_mode3_shift-(ulong)(void*)&onekpaq_decompressor_mode3,
(ulong)&onekpaq_decompressor_mode4_shift-(ulong)(void*)&onekpaq_decompressor_mode4
};
static ulong decompressorSizes[4]={
(ulong)&onekpaq_decompressor_mode1_end-(ulong)(void*)&onekpaq_decompressor_mode1,
(ulong)&onekpaq_decompressor_mode2_end-(ulong)(void*)&onekpaq_decompressor_mode2,
(ulong)&onekpaq_decompressor_mode3_end-(ulong)(void*)&onekpaq_decompressor_mode3,
(ulong)&onekpaq_decompressor_mode4_end-(ulong)(void*)&onekpaq_decompressor_mode4
};
// we have no way of calculating the real size for dest without decompressing.
// lets just put some huge number here
uint expectedSize=262144;
std::vector<u8> ret(expectedSize);
std::vector<u8> combine=src1;
combine.insert(combine.end(),src2.begin(),src2.end());
for (int i=0;i<4;i++) combine.push_back(0);
const uint destStartMargin=13;
uint dIndex=static_cast<uint>(mode)-1;
AsmDecompressor decompr=(AsmDecompressor)makeCodeblock(decompressors[dIndex],decompressorSizes[dIndex]);
((u8*)(void*)decompr)[decompressorShiftOffsets[dIndex]]=shift;
makeRunnable((void*)decompr,decompressorSizes[dIndex]);
/*fprintf(stderr, "offset = %zu, mode=%zu, dind=%zu, shift=%zu\n", src1.size(), mode, dIndex, shift);
for (size_t i = 0; i < ((src1.size() < 4) ? src1.size() : 4); ++i) {
fprintf(stderr, "src1[%zu] = 0x%x\n", i, src1[i]);
}
for (size_t i = 0; i < ((src2.size() < 4) ? src2.size() : 4); ++i) {
fprintf(stderr, "src2[%zu] = 0x%x\n", i, src2[i]);
}*/
//INFO("Running asm decompressor Xbx=%p Xdi=%p",combine.data()+src1.size(),ret.data()+destStartMargin);
auto timeTaken=Timer([&]() {
decompr(combine.data()+src1.size(),ret.data()+destStartMargin);
});
INFO("Asm decompressor done. Time taken %f seconds",float(timeTaken));
return std::vector<u8>(ret.begin()+destStartMargin,ret.begin()+expectedSize-destStartMargin);
}