forked from asmjit/asmjit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
asmjit_test_opcode.cpp
92 lines (74 loc) · 2.39 KB
/
asmjit_test_opcode.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
// [AsmJit]
// Complete x86/x64 JIT and Remote Assembler for C++.
//
// [License]
// Zlib - See LICENSE.md file in the package.
// This file is used to test opcodes generated by AsmJit. Output can be
// disassembled in your IDE or by your favorite disassembler. Instructions
// are grouped by category and then sorted alphabetically.
// [Dependencies]
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "./asmjit.h"
#include "./asmjit_test_opcode.h"
using namespace asmjit;
struct OpcodeDumpInfo {
uint32_t archType;
bool useRex1;
bool useRex2;
};
static const char* archTypeToString(uint32_t archType) {
switch (archType) {
case ArchInfo::kTypeNone : return "None";
case ArchInfo::kTypeX86 : return "X86";
case ArchInfo::kTypeX64 : return "X64";
case ArchInfo::kTypeA32 : return "A32";
case ArchInfo::kTypeA64 : return "A64";
default:
return "<unknown>";
}
}
struct TestErrorHandler : public ErrorHandler {
virtual bool handleError(Error err, const char* message, CodeEmitter* origin) {
printf("ERROR 0x%08X: %s\n", err, message);
return true;
}
};
typedef void (*VoidFunc)(void);
int main(int argc, char* argv[]) {
TestErrorHandler eh;
OpcodeDumpInfo infoList[] = {
{ ArchInfo::kTypeX86, false, false },
{ ArchInfo::kTypeX64, false, false },
{ ArchInfo::kTypeX64, false, true },
{ ArchInfo::kTypeX64, true , false },
{ ArchInfo::kTypeX64, true , true }
};
for (int i = 0; i < ASMJIT_ARRAY_SIZE(infoList); i++) {
const OpcodeDumpInfo& info = infoList[i];
printf("Opcodes [ARCH=%s REX1=%s REX2=%s]\n",
archTypeToString(info.archType),
info.useRex1 ? "true" : "false",
info.useRex2 ? "true" : "false");
CodeHolder code;
code.init(CodeInfo(info.archType));
code.setErrorHandler(&eh);
#if !defined(ASMJIT_DISABLE_LOGGING)
FileLogger logger(stdout);
logger.addOptions(Logger::kOptionBinaryForm);
code.setLogger(&logger);
#endif // ASMJIT_DISABLE_LOGGING
X86Assembler a(&code);
asmtest::generateOpcodes(a, info.useRex1, info.useRex2);
// If this is the host architecture the code generated can be executed
// for debugging purposes (the first instruction is ret anyway).
if (code.getArchType() == ArchInfo::kTypeHost) {
JitRuntime runtime;
VoidFunc p;
Error err = runtime.add(&p, &code);
if (err == kErrorOk) p();
}
}
return 0;
}