1 /* Capstone Disassembly Engine */
2 /* By Nguyen Anh Quynh, 2018 */
3
4 #ifdef CAPSTONE_HAS_EVM
5
6 #include <string.h>
7
8 #include "../../cs_priv.h"
9 #include "../../utils.h"
10
11 #include "EVMMapping.h"
12
13 #ifndef CAPSTONE_DIET
14 static cs_evm insns[256] = {
15 #include "EVMMappingInsn.inc"
16 };
17 #endif
18
19 // look for @id in @insns, given its size in @max.
20 // return -1 if not found
evm_insn_find(cs_evm * insns,unsigned int max,unsigned int id)21 static int evm_insn_find(cs_evm *insns, unsigned int max, unsigned int id)
22 {
23 if (id >= max)
24 return -1;
25
26 if (insns[id].fee == 0xffffffff)
27 // unused opcode
28 return -1;
29
30 return (int)id;
31 }
32
33 // fill in details
EVM_get_insn_id(cs_struct * h,cs_insn * insn,unsigned int id)34 void EVM_get_insn_id(cs_struct *h, cs_insn *insn, unsigned int id)
35 {
36 insn->id = id;
37 #ifndef CAPSTONE_DIET
38 if (evm_insn_find(insns, ARR_SIZE(insns), id) > 0) {
39 if (h->detail) {
40 memcpy(&insn->detail->evm, &insns[id], sizeof(insns[id]));
41 }
42 }
43 #endif
44 }
45
46 #ifndef CAPSTONE_DIET
47 static name_map insn_name_maps[256] = {
48 { EVM_INS_STOP, "stop" },
49 { EVM_INS_ADD, "add" },
50 { EVM_INS_MUL, "mul" },
51 { EVM_INS_SUB, "sub" },
52 { EVM_INS_DIV, "div" },
53 { EVM_INS_SDIV, "sdiv" },
54 { EVM_INS_MOD, "mod" },
55 { EVM_INS_SMOD, "smod" },
56 { EVM_INS_ADDMOD, "addmod" },
57 { EVM_INS_MULMOD, "mulmod" },
58 { EVM_INS_EXP, "exp" },
59 { EVM_INS_SIGNEXTEND, "signextend" },
60 { EVM_INS_INVALID, NULL },
61 { EVM_INS_INVALID, NULL },
62 { EVM_INS_INVALID, NULL },
63 { EVM_INS_INVALID, NULL },
64 { EVM_INS_LT, "lt" },
65 { EVM_INS_GT, "gt" },
66 { EVM_INS_SLT, "slt" },
67 { EVM_INS_SGT, "sgt" },
68 { EVM_INS_EQ, "eq" },
69 { EVM_INS_ISZERO, "iszero" },
70 { EVM_INS_AND, "and" },
71 { EVM_INS_OR, "or" },
72 { EVM_INS_XOR, "xor" },
73 { EVM_INS_NOT, "not" },
74 { EVM_INS_BYTE, "byte" },
75 { EVM_INS_INVALID, NULL },
76 { EVM_INS_INVALID, NULL },
77 { EVM_INS_INVALID, NULL },
78 { EVM_INS_INVALID, NULL },
79 { EVM_INS_INVALID, NULL },
80 { EVM_INS_SHA3, "sha3" },
81 { EVM_INS_INVALID, NULL },
82 { EVM_INS_INVALID, NULL },
83 { EVM_INS_INVALID, NULL },
84 { EVM_INS_INVALID, NULL },
85 { EVM_INS_INVALID, NULL },
86 { EVM_INS_INVALID, NULL },
87 { EVM_INS_INVALID, NULL },
88 { EVM_INS_INVALID, NULL },
89 { EVM_INS_INVALID, NULL },
90 { EVM_INS_INVALID, NULL },
91 { EVM_INS_INVALID, NULL },
92 { EVM_INS_INVALID, NULL },
93 { EVM_INS_INVALID, NULL },
94 { EVM_INS_INVALID, NULL },
95 { EVM_INS_INVALID, NULL },
96 { EVM_INS_ADDRESS, "address" },
97 { EVM_INS_BALANCE, "balance" },
98 { EVM_INS_ORIGIN, "origin" },
99 { EVM_INS_CALLER, "caller" },
100 { EVM_INS_CALLVALUE, "callvalue" },
101 { EVM_INS_CALLDATALOAD, "calldataload" },
102 { EVM_INS_CALLDATASIZE, "calldatasize" },
103 { EVM_INS_CALLDATACOPY, "calldatacopy" },
104 { EVM_INS_CODESIZE, "codesize" },
105 { EVM_INS_CODECOPY, "codecopy" },
106 { EVM_INS_GASPRICE, "gasprice" },
107 { EVM_INS_EXTCODESIZE, "extcodesize" },
108 { EVM_INS_EXTCODECOPY, "extcodecopy" },
109 { EVM_INS_RETURNDATASIZE, "returndatasize" },
110 { EVM_INS_RETURNDATACOPY, "returndatacopy" },
111 { EVM_INS_INVALID, NULL },
112 { EVM_INS_BLOCKHASH, "blockhash" },
113 { EVM_INS_COINBASE, "coinbase" },
114 { EVM_INS_TIMESTAMP, "timestamp" },
115 { EVM_INS_NUMBER, "number" },
116 { EVM_INS_DIFFICULTY, "difficulty" },
117 { EVM_INS_GASLIMIT, "gaslimit" },
118 { EVM_INS_INVALID, NULL },
119 { EVM_INS_INVALID, NULL },
120 { EVM_INS_INVALID, NULL },
121 { EVM_INS_INVALID, NULL },
122 { EVM_INS_INVALID, NULL },
123 { EVM_INS_INVALID, NULL },
124 { EVM_INS_INVALID, NULL },
125 { EVM_INS_INVALID, NULL },
126 { EVM_INS_INVALID, NULL },
127 { EVM_INS_INVALID, NULL },
128 { EVM_INS_POP, "pop" },
129 { EVM_INS_MLOAD, "mload" },
130 { EVM_INS_MSTORE, "mstore" },
131 { EVM_INS_MSTORE8, "mstore8" },
132 { EVM_INS_SLOAD, "sload" },
133 { EVM_INS_SSTORE, "sstore" },
134 { EVM_INS_JUMP, "jump" },
135 { EVM_INS_JUMPI, "jumpi" },
136 { EVM_INS_PC, "pc" },
137 { EVM_INS_MSIZE, "msize" },
138 { EVM_INS_GAS, "gas" },
139 { EVM_INS_JUMPDEST, "jumpdest" },
140 { EVM_INS_INVALID, NULL },
141 { EVM_INS_INVALID, NULL },
142 { EVM_INS_INVALID, NULL },
143 { EVM_INS_INVALID, NULL },
144 { EVM_INS_PUSH1, "push1" },
145 { EVM_INS_PUSH2, "push2" },
146 { EVM_INS_PUSH3, "push3" },
147 { EVM_INS_PUSH4, "push4" },
148 { EVM_INS_PUSH5, "push5" },
149 { EVM_INS_PUSH6, "push6" },
150 { EVM_INS_PUSH7, "push7" },
151 { EVM_INS_PUSH8, "push8" },
152 { EVM_INS_PUSH9, "push9" },
153 { EVM_INS_PUSH10, "push10" },
154 { EVM_INS_PUSH11, "push11" },
155 { EVM_INS_PUSH12, "push12" },
156 { EVM_INS_PUSH13, "push13" },
157 { EVM_INS_PUSH14, "push14" },
158 { EVM_INS_PUSH15, "push15" },
159 { EVM_INS_PUSH16, "push16" },
160 { EVM_INS_PUSH17, "push17" },
161 { EVM_INS_PUSH18, "push18" },
162 { EVM_INS_PUSH19, "push19" },
163 { EVM_INS_PUSH20, "push20" },
164 { EVM_INS_PUSH21, "push21" },
165 { EVM_INS_PUSH22, "push22" },
166 { EVM_INS_PUSH23, "push23" },
167 { EVM_INS_PUSH24, "push24" },
168 { EVM_INS_PUSH25, "push25" },
169 { EVM_INS_PUSH26, "push26" },
170 { EVM_INS_PUSH27, "push27" },
171 { EVM_INS_PUSH28, "push28" },
172 { EVM_INS_PUSH29, "push29" },
173 { EVM_INS_PUSH30, "push30" },
174 { EVM_INS_PUSH31, "push31" },
175 { EVM_INS_PUSH32, "push32" },
176 { EVM_INS_DUP1, "dup1" },
177 { EVM_INS_DUP2, "dup2" },
178 { EVM_INS_DUP3, "dup3" },
179 { EVM_INS_DUP4, "dup4" },
180 { EVM_INS_DUP5, "dup5" },
181 { EVM_INS_DUP6, "dup6" },
182 { EVM_INS_DUP7, "dup7" },
183 { EVM_INS_DUP8, "dup8" },
184 { EVM_INS_DUP9, "dup9" },
185 { EVM_INS_DUP10, "dup10" },
186 { EVM_INS_DUP11, "dup11" },
187 { EVM_INS_DUP12, "dup12" },
188 { EVM_INS_DUP13, "dup13" },
189 { EVM_INS_DUP14, "dup14" },
190 { EVM_INS_DUP15, "dup15" },
191 { EVM_INS_DUP16, "dup16" },
192 { EVM_INS_SWAP1, "swap1" },
193 { EVM_INS_SWAP2, "swap2" },
194 { EVM_INS_SWAP3, "swap3" },
195 { EVM_INS_SWAP4, "swap4" },
196 { EVM_INS_SWAP5, "swap5" },
197 { EVM_INS_SWAP6, "swap6" },
198 { EVM_INS_SWAP7, "swap7" },
199 { EVM_INS_SWAP8, "swap8" },
200 { EVM_INS_SWAP9, "swap9" },
201 { EVM_INS_SWAP10, "swap10" },
202 { EVM_INS_SWAP11, "swap11" },
203 { EVM_INS_SWAP12, "swap12" },
204 { EVM_INS_SWAP13, "swap13" },
205 { EVM_INS_SWAP14, "swap14" },
206 { EVM_INS_SWAP15, "swap15" },
207 { EVM_INS_SWAP16, "swap16" },
208 { EVM_INS_LOG0, "log0" },
209 { EVM_INS_LOG1, "log1" },
210 { EVM_INS_LOG2, "log2" },
211 { EVM_INS_LOG3, "log3" },
212 { EVM_INS_LOG4, "log4" },
213 { EVM_INS_INVALID, NULL },
214 { EVM_INS_INVALID, NULL },
215 { EVM_INS_INVALID, NULL },
216 { EVM_INS_INVALID, NULL },
217 { EVM_INS_INVALID, NULL },
218 { EVM_INS_INVALID, NULL },
219 { EVM_INS_INVALID, NULL },
220 { EVM_INS_INVALID, NULL },
221 { EVM_INS_INVALID, NULL },
222 { EVM_INS_INVALID, NULL },
223 { EVM_INS_INVALID, NULL },
224 { EVM_INS_INVALID, NULL },
225 { EVM_INS_INVALID, NULL },
226 { EVM_INS_INVALID, NULL },
227 { EVM_INS_INVALID, NULL },
228 { EVM_INS_INVALID, NULL },
229 { EVM_INS_INVALID, NULL },
230 { EVM_INS_INVALID, NULL },
231 { EVM_INS_INVALID, NULL },
232 { EVM_INS_INVALID, NULL },
233 { EVM_INS_INVALID, NULL },
234 { EVM_INS_INVALID, NULL },
235 { EVM_INS_INVALID, NULL },
236 { EVM_INS_INVALID, NULL },
237 { EVM_INS_INVALID, NULL },
238 { EVM_INS_INVALID, NULL },
239 { EVM_INS_INVALID, NULL },
240 { EVM_INS_INVALID, NULL },
241 { EVM_INS_INVALID, NULL },
242 { EVM_INS_INVALID, NULL },
243 { EVM_INS_INVALID, NULL },
244 { EVM_INS_INVALID, NULL },
245 { EVM_INS_INVALID, NULL },
246 { EVM_INS_INVALID, NULL },
247 { EVM_INS_INVALID, NULL },
248 { EVM_INS_INVALID, NULL },
249 { EVM_INS_INVALID, NULL },
250 { EVM_INS_INVALID, NULL },
251 { EVM_INS_INVALID, NULL },
252 { EVM_INS_INVALID, NULL },
253 { EVM_INS_INVALID, NULL },
254 { EVM_INS_INVALID, NULL },
255 { EVM_INS_INVALID, NULL },
256 { EVM_INS_INVALID, NULL },
257 { EVM_INS_INVALID, NULL },
258 { EVM_INS_INVALID, NULL },
259 { EVM_INS_INVALID, NULL },
260 { EVM_INS_INVALID, NULL },
261 { EVM_INS_INVALID, NULL },
262 { EVM_INS_INVALID, NULL },
263 { EVM_INS_INVALID, NULL },
264 { EVM_INS_INVALID, NULL },
265 { EVM_INS_INVALID, NULL },
266 { EVM_INS_INVALID, NULL },
267 { EVM_INS_INVALID, NULL },
268 { EVM_INS_INVALID, NULL },
269 { EVM_INS_INVALID, NULL },
270 { EVM_INS_INVALID, NULL },
271 { EVM_INS_INVALID, NULL },
272 { EVM_INS_INVALID, NULL },
273 { EVM_INS_INVALID, NULL },
274 { EVM_INS_INVALID, NULL },
275 { EVM_INS_INVALID, NULL },
276 { EVM_INS_INVALID, NULL },
277 { EVM_INS_INVALID, NULL },
278 { EVM_INS_INVALID, NULL },
279 { EVM_INS_INVALID, NULL },
280 { EVM_INS_INVALID, NULL },
281 { EVM_INS_INVALID, NULL },
282 { EVM_INS_INVALID, NULL },
283 { EVM_INS_INVALID, NULL },
284 { EVM_INS_INVALID, NULL },
285 { EVM_INS_INVALID, NULL },
286 { EVM_INS_INVALID, NULL },
287 { EVM_INS_INVALID, NULL },
288 { EVM_INS_CREATE, "create" },
289 { EVM_INS_CALL, "call" },
290 { EVM_INS_CALLCODE, "callcode" },
291 { EVM_INS_RETURN, "return" },
292 { EVM_INS_DELEGATECALL, "delegatecall" },
293 { EVM_INS_CALLBLACKBOX, "callblackbox" },
294 { EVM_INS_INVALID, NULL },
295 { EVM_INS_INVALID, NULL },
296 { EVM_INS_INVALID, NULL },
297 { EVM_INS_INVALID, NULL },
298 { EVM_INS_STATICCALL, "staticcall" },
299 { EVM_INS_INVALID, NULL },
300 { EVM_INS_INVALID, NULL },
301 { EVM_INS_REVERT, "revert" },
302 { EVM_INS_INVALID, NULL },
303 { EVM_INS_SUICIDE, "suicide" },
304 };
305 #endif
306
EVM_insn_name(csh handle,unsigned int id)307 const char *EVM_insn_name(csh handle, unsigned int id)
308 {
309 #ifndef CAPSTONE_DIET
310 if (id >= ARR_SIZE(insn_name_maps))
311 return NULL;
312 else
313 return insn_name_maps[id].name;
314 #else
315 return NULL;
316 #endif
317 }
318
319 #ifndef CAPSTONE_DIET
320 static name_map group_name_maps[] = {
321 // generic groups
322 { EVM_GRP_INVALID, NULL },
323 { EVM_GRP_JUMP, "jump" },
324 // special groups
325 { EVM_GRP_MATH, "math" },
326 { EVM_GRP_STACK_WRITE, "stack_write" },
327 { EVM_GRP_STACK_READ, "stack_read" },
328 { EVM_GRP_MEM_WRITE, "mem_write" },
329 { EVM_GRP_MEM_READ, "mem_read" },
330 { EVM_GRP_STORE_WRITE, "store_write" },
331 { EVM_GRP_STORE_READ, "store_read" },
332 { EVM_GRP_HALT, "halt" },
333 };
334 #endif
335
EVM_group_name(csh handle,unsigned int id)336 const char *EVM_group_name(csh handle, unsigned int id)
337 {
338 #ifndef CAPSTONE_DIET
339 return id2name(group_name_maps, ARR_SIZE(group_name_maps), id);
340 #else
341 return NULL;
342 #endif
343 }
344 #endif
345