-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathCtu.h
243 lines (220 loc) · 5.1 KB
/
Ctu.h
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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
#pragma once
#include <netinet/in.h>
#include <stdint.h>
#include <string.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <algorithm>
#include <array>
#include <cassert>
#include <climits>
#include <forward_list>
#include <fstream>
#include <functional>
#include <iomanip>
#include <iostream>
#include <list>
#include <map>
#include <memory>
#include <mutex>
#include <numeric>
#include <unordered_map>
#include <queue>
#include <string>
#include <thread>
#include <tuple>
#include <type_traits>
#include <vector>
#include <sstream>
using namespace std;
typedef __int128_t int128_t;
typedef __uint128_t uint128_t;
typedef float float32_t;
typedef double float64_t;
typedef uint64_t gptr;
typedef uint64_t guint;
typedef uint8_t guchar;
typedef uint16_t gushort;
typedef uint32_t ghandle;
typedef uint64_t gpid;
const gptr TERMADDR = 1ULL << 61;
#define FOURCC(a, b, c, d) (((d) << 24) | ((c) << 16) | ((b) << 8) | (a))
#ifdef __APPLE__
#define ADDRFMT "%016llx"
#define LONGFMT "%llx"
#else
#define ADDRFMT "%016lx"
#define LONGFMT "%lx"
#endif
enum LogLevel {
None = 0,
Error = 1,
Warn = 2,
Debug = 3,
Info = 4
};
extern LogLevel g_LogLevel;
#define LOG_ERROR(module, msg, ...) do { \
if(g_LogLevel >= Error) { \
fprintf(stderr, "[" #module "] ERROR: " msg "\n", ##__VA_ARGS__); \
cerr << endl; \
} \
exit(1); \
} while(0)
#define LOG_INFO(module, msg, ...) do { \
if(g_LogLevel >= Info) { \
printf("[" #module "] INFO: " msg, ##__VA_ARGS__); \
cout << endl; \
} \
} while(0)
#define LOG_DEBUG(module, msg, ...) do { \
if(g_LogLevel >= Debug) { \
printf("[" #module "] DEBUG: " msg, ##__VA_ARGS__); \
cout << endl; \
} \
} while(0)
class Ctu;
class LogMessage;
enum class BreakpointType {
None,
Execute,
Read,
Write,
Access
};
typedef struct {
float64_t low;
float64_t high;
} float128;
typedef struct {
guint SP, PC, NZCV;
union {
struct {
guint gprs[31];
float128 fprs[32];
};
struct {
guint X0, X1, X2, X3,
X4, X5, X6, X7,
X8, X9, X10, X11,
X12, X13, X14, X15,
X16, X17, X18, X19,
X20, X21, X22, X23,
X24, X25, X26, X27,
X28, X29, X30;
float128 Q0, Q1, Q2, Q3,
Q4, Q5, Q6, Q7,
Q8, Q9, Q10, Q11,
Q12, Q13, Q14, Q15,
Q16, Q17, Q18, Q19,
Q20, Q21, Q22, Q23,
Q24, Q25, Q26, Q27,
Q28, Q29, Q30, Q31;
};
};
} ThreadRegisters;
#include "optionparser.h"
#include "Lisparser.h"
#include "KObject.h"
#include "Mmio.h"
#include "Cpu.h"
#include "Sync.h"
#include "ThreadManager.h"
#include "Svc.h"
#include "Ipc.h"
#include "Nxo.h"
#include "IpcBridge.h"
#include "GdbStub.h"
template<unsigned long N>
void hexdump(shared_ptr<array<uint8_t, N>> buf, unsigned long count=N) {
if(g_LogLevel < Debug)
return;
for(auto i = 0; i < count; i += 16) {
printf("%04x | ", i);
for(auto j = 0; j < 16; ++j) {
printf("%02x ", buf->data()[i + j]);
if(j == 7)
printf(" ");
}
printf("| ");
for(auto j = 0; j < 16; ++j) {
auto val = buf->data()[i + j];
if(isprint(val))
printf("%c", val);
else
printf(".");
if(j == 7)
printf(" ");
}
printf("\n");
}
printf("%04x\n", (int) count);
}
class Ctu {
public:
Ctu();
void execProgram(gptr ep);
template<typename T>
ghandle newHandle(shared_ptr<T> obj) {
static_assert(std::is_base_of<KObject, T>::value, "T must derive from KObject");
auto hnd = handleId++;
LOG_DEBUG(Ctu, "Creating handle %x", hnd);
handles[hnd] = dynamic_pointer_cast<KObject>(obj);
return hnd;
}
template<typename T>
shared_ptr<T> getHandle(ghandle handle) {
static_assert(std::is_base_of<KObject, T>::value, "T must derive from KObject");
if(handles.find(handle) == handles.end())
LOG_ERROR(Ctu, "Could not find handle with ID 0x%x !", handle);
auto obj = handles[handle];
auto faux = dynamic_pointer_cast<FauxHandle>(obj);
if(faux != nullptr)
LOG_ERROR(Ctu, "Accessing faux handle! 0x%x", faux->val);
auto temp = dynamic_pointer_cast<T>(obj);
if(temp == nullptr)
LOG_ERROR(Ctu, "Got null pointer after cast. Before: 0x%p", (void *) obj.get());
return temp;
}
bool hasHandle(ghandle handle) {
return handles.find(handle) != handles.end();
}
ghandle duplicateHandle(KObject *ptr);
void deleteHandle(ghandle handle);
Cpu cpu;
Mmio mmiohandler;
Svc svc;
Ipc ipc;
ThreadManager tm;
IpcBridge bridge;
GdbStub gdbStub;
guint heapsize;
gptr loadbase, loadsize;
bool socketsEnabled;
bool initializeMemory = false;
private:
ghandle handleId;
unordered_map<ghandle, shared_ptr<KObject>> handles;
};
// TODO: move this/autogenerate this
#define VERSION_1_0_0 0
#define VERSION_2_0_0 1
#define VERSION_2_1_0 2
#define VERSION_2_2_0 3
#define VERSION_2_3_0 4
#define VERSION_3_0_0 5
#define VERSION_3_0_1 6
#define VERSION_3_0_2 7
#define VERSION_4_0_0 8
#define VERSION_4_0_1 9
#define VERSION_4_1_0 10
#define VERSION_5_0_0 11
#define VERSION_5_0_1 12
#define VERSION_5_0_2 13
#define VERSION_5_1_0 14
#define TARGET_VERSION VERSION_5_1_0
#include "IpcStubs.h"