/src/capstonenext/arch/X86/X86IntelInstPrinter.c
Line | Count | Source (jump to first uncovered line) |
1 | | //===-- X86IntelInstPrinter.cpp - Intel assembly instruction printing -----===// |
2 | | // |
3 | | // The LLVM Compiler Infrastructure |
4 | | // |
5 | | // This file is distributed under the University of Illinois Open Source |
6 | | // License. See LICENSE.TXT for details. |
7 | | // |
8 | | //===----------------------------------------------------------------------===// |
9 | | // |
10 | | // This file includes code for rendering MCInst instances as Intel-style |
11 | | // assembly. |
12 | | // |
13 | | //===----------------------------------------------------------------------===// |
14 | | |
15 | | /* Capstone Disassembly Engine */ |
16 | | /* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2019 */ |
17 | | |
18 | | #ifdef CAPSTONE_HAS_X86 |
19 | | |
20 | | #ifdef _MSC_VER |
21 | | #pragma warning(disable:4996) // disable MSVC's warning on strncpy() |
22 | | #pragma warning(disable:28719) // disable MSVC's warning on strncpy() |
23 | | #endif |
24 | | |
25 | | #if !defined(CAPSTONE_HAS_OSXKERNEL) |
26 | | #include <ctype.h> |
27 | | #endif |
28 | | #include <capstone/platform.h> |
29 | | |
30 | | #if defined(CAPSTONE_HAS_OSXKERNEL) |
31 | | #include <Availability.h> |
32 | | #include <libkern/libkern.h> |
33 | | #else |
34 | | #include <stdio.h> |
35 | | #include <stdlib.h> |
36 | | #endif |
37 | | #include <string.h> |
38 | | |
39 | | #include "../../utils.h" |
40 | | #include "../../MCInst.h" |
41 | | #include "../../SStream.h" |
42 | | #include "../../MCRegisterInfo.h" |
43 | | |
44 | | #include "X86InstPrinter.h" |
45 | | #include "X86Mapping.h" |
46 | | #include "X86InstPrinterCommon.h" |
47 | | |
48 | | #define GET_INSTRINFO_ENUM |
49 | | #ifdef CAPSTONE_X86_REDUCE |
50 | | #include "X86GenInstrInfo_reduce.inc" |
51 | | #else |
52 | | #include "X86GenInstrInfo.inc" |
53 | | #endif |
54 | | |
55 | | #define GET_REGINFO_ENUM |
56 | | #include "X86GenRegisterInfo.inc" |
57 | | |
58 | | #include "X86BaseInfo.h" |
59 | | |
60 | | static void printMemReference(MCInst *MI, unsigned Op, SStream *O); |
61 | | static void printOperand(MCInst *MI, unsigned OpNo, SStream *O); |
62 | | |
63 | | |
64 | | static void set_mem_access(MCInst *MI, bool status) |
65 | 162k | { |
66 | 162k | if (MI->csh->detail_opt != CS_OPT_ON) |
67 | 0 | return; |
68 | | |
69 | 162k | MI->csh->doing_mem = status; |
70 | 162k | if (!status) |
71 | | // done, create the next operand slot |
72 | 81.0k | MI->flat_insn->detail->x86.op_count++; |
73 | | |
74 | 162k | } |
75 | | |
76 | | static void printopaquemem(MCInst *MI, unsigned OpNo, SStream *O) |
77 | 15.1k | { |
78 | | // FIXME: do this with autogen |
79 | | // printf(">>> ID = %u\n", MI->flat_insn->id); |
80 | 15.1k | switch(MI->flat_insn->id) { |
81 | 4.70k | default: |
82 | 4.70k | SStream_concat0(O, "ptr "); |
83 | 4.70k | break; |
84 | 1.56k | case X86_INS_SGDT: |
85 | 3.01k | case X86_INS_SIDT: |
86 | 4.44k | case X86_INS_LGDT: |
87 | 6.20k | case X86_INS_LIDT: |
88 | 7.13k | case X86_INS_FXRSTOR: |
89 | 7.47k | case X86_INS_FXSAVE: |
90 | 9.18k | case X86_INS_LJMP: |
91 | 10.4k | case X86_INS_LCALL: |
92 | | // do not print "ptr" |
93 | 10.4k | break; |
94 | 15.1k | } |
95 | | |
96 | 15.1k | switch(MI->csh->mode) { |
97 | 5.05k | case CS_MODE_16: |
98 | 5.05k | switch(MI->flat_insn->id) { |
99 | 1.60k | default: |
100 | 1.60k | MI->x86opsize = 2; |
101 | 1.60k | break; |
102 | 583 | case X86_INS_LJMP: |
103 | 1.04k | case X86_INS_LCALL: |
104 | 1.04k | MI->x86opsize = 4; |
105 | 1.04k | break; |
106 | 492 | case X86_INS_SGDT: |
107 | 1.00k | case X86_INS_SIDT: |
108 | 1.50k | case X86_INS_LGDT: |
109 | 2.40k | case X86_INS_LIDT: |
110 | 2.40k | MI->x86opsize = 6; |
111 | 2.40k | break; |
112 | 5.05k | } |
113 | 5.05k | break; |
114 | 6.14k | case CS_MODE_32: |
115 | 6.14k | switch(MI->flat_insn->id) { |
116 | 2.64k | default: |
117 | 2.64k | MI->x86opsize = 4; |
118 | 2.64k | break; |
119 | 421 | case X86_INS_LJMP: |
120 | 1.14k | case X86_INS_JMP: |
121 | 1.54k | case X86_INS_LCALL: |
122 | 2.12k | case X86_INS_SGDT: |
123 | 2.56k | case X86_INS_SIDT: |
124 | 3.06k | case X86_INS_LGDT: |
125 | 3.49k | case X86_INS_LIDT: |
126 | 3.49k | MI->x86opsize = 6; |
127 | 3.49k | break; |
128 | 6.14k | } |
129 | 6.14k | break; |
130 | 6.14k | case CS_MODE_64: |
131 | 3.97k | switch(MI->flat_insn->id) { |
132 | 1.00k | default: |
133 | 1.00k | MI->x86opsize = 8; |
134 | 1.00k | break; |
135 | 715 | case X86_INS_LJMP: |
136 | 1.12k | case X86_INS_LCALL: |
137 | 1.62k | case X86_INS_SGDT: |
138 | 2.12k | case X86_INS_SIDT: |
139 | 2.55k | case X86_INS_LGDT: |
140 | 2.97k | case X86_INS_LIDT: |
141 | 2.97k | MI->x86opsize = 10; |
142 | 2.97k | break; |
143 | 3.97k | } |
144 | 3.97k | break; |
145 | 3.97k | default: // never reach |
146 | 0 | break; |
147 | 15.1k | } |
148 | | |
149 | 15.1k | printMemReference(MI, OpNo, O); |
150 | 15.1k | } |
151 | | |
152 | | static void printi8mem(MCInst *MI, unsigned OpNo, SStream *O) |
153 | 133k | { |
154 | 133k | SStream_concat0(O, "byte ptr "); |
155 | 133k | MI->x86opsize = 1; |
156 | 133k | printMemReference(MI, OpNo, O); |
157 | 133k | } |
158 | | |
159 | | static void printi16mem(MCInst *MI, unsigned OpNo, SStream *O) |
160 | 30.0k | { |
161 | 30.0k | MI->x86opsize = 2; |
162 | 30.0k | SStream_concat0(O, "word ptr "); |
163 | 30.0k | printMemReference(MI, OpNo, O); |
164 | 30.0k | } |
165 | | |
166 | | static void printi32mem(MCInst *MI, unsigned OpNo, SStream *O) |
167 | 56.7k | { |
168 | 56.7k | MI->x86opsize = 4; |
169 | 56.7k | SStream_concat0(O, "dword ptr "); |
170 | 56.7k | printMemReference(MI, OpNo, O); |
171 | 56.7k | } |
172 | | |
173 | | static void printi64mem(MCInst *MI, unsigned OpNo, SStream *O) |
174 | 22.7k | { |
175 | 22.7k | SStream_concat0(O, "qword ptr "); |
176 | 22.7k | MI->x86opsize = 8; |
177 | 22.7k | printMemReference(MI, OpNo, O); |
178 | 22.7k | } |
179 | | |
180 | | static void printi128mem(MCInst *MI, unsigned OpNo, SStream *O) |
181 | 9.05k | { |
182 | 9.05k | SStream_concat0(O, "xmmword ptr "); |
183 | 9.05k | MI->x86opsize = 16; |
184 | 9.05k | printMemReference(MI, OpNo, O); |
185 | 9.05k | } |
186 | | |
187 | | static void printi512mem(MCInst *MI, unsigned OpNo, SStream *O) |
188 | 3.39k | { |
189 | 3.39k | SStream_concat0(O, "zmmword ptr "); |
190 | 3.39k | MI->x86opsize = 64; |
191 | 3.39k | printMemReference(MI, OpNo, O); |
192 | 3.39k | } |
193 | | |
194 | | #ifndef CAPSTONE_X86_REDUCE |
195 | | static void printi256mem(MCInst *MI, unsigned OpNo, SStream *O) |
196 | 5.24k | { |
197 | 5.24k | SStream_concat0(O, "ymmword ptr "); |
198 | 5.24k | MI->x86opsize = 32; |
199 | 5.24k | printMemReference(MI, OpNo, O); |
200 | 5.24k | } |
201 | | |
202 | | static void printf32mem(MCInst *MI, unsigned OpNo, SStream *O) |
203 | 10.0k | { |
204 | 10.0k | switch(MCInst_getOpcode(MI)) { |
205 | 7.90k | default: |
206 | 7.90k | SStream_concat0(O, "dword ptr "); |
207 | 7.90k | MI->x86opsize = 4; |
208 | 7.90k | break; |
209 | 1.17k | case X86_FSTENVm: |
210 | 2.19k | case X86_FLDENVm: |
211 | | // TODO: fix this in tablegen instead |
212 | 2.19k | switch(MI->csh->mode) { |
213 | 0 | default: // never reach |
214 | 0 | break; |
215 | 442 | case CS_MODE_16: |
216 | 442 | MI->x86opsize = 14; |
217 | 442 | break; |
218 | 1.11k | case CS_MODE_32: |
219 | 1.75k | case CS_MODE_64: |
220 | 1.75k | MI->x86opsize = 28; |
221 | 1.75k | break; |
222 | 2.19k | } |
223 | 2.19k | break; |
224 | 10.0k | } |
225 | | |
226 | 10.0k | printMemReference(MI, OpNo, O); |
227 | 10.0k | } |
228 | | |
229 | | static void printf64mem(MCInst *MI, unsigned OpNo, SStream *O) |
230 | 3.52k | { |
231 | | // TODO: fix COMISD in Tablegen instead (#1456) |
232 | 3.52k | if (MI->op1_size == 16) { |
233 | | // printf("printf64mem id = %u\n", MCInst_getOpcode(MI)); |
234 | 1.88k | switch(MCInst_getOpcode(MI)) { |
235 | 1.88k | default: |
236 | 1.88k | SStream_concat0(O, "qword ptr "); |
237 | 1.88k | MI->x86opsize = 8; |
238 | 1.88k | break; |
239 | 0 | case X86_MOVPQI2QImr: |
240 | 0 | SStream_concat0(O, "xmmword ptr "); |
241 | 0 | MI->x86opsize = 16; |
242 | 0 | break; |
243 | 1.88k | } |
244 | 1.88k | } else { |
245 | 1.64k | SStream_concat0(O, "qword ptr "); |
246 | 1.64k | MI->x86opsize = 8; |
247 | 1.64k | } |
248 | | |
249 | 3.52k | printMemReference(MI, OpNo, O); |
250 | 3.52k | } |
251 | | |
252 | | static void printf80mem(MCInst *MI, unsigned OpNo, SStream *O) |
253 | 899 | { |
254 | 899 | switch(MCInst_getOpcode(MI)) { |
255 | 474 | default: |
256 | 474 | SStream_concat0(O, "xword ptr "); |
257 | 474 | break; |
258 | 258 | case X86_FBLDm: |
259 | 425 | case X86_FBSTPm: |
260 | 425 | break; |
261 | 899 | } |
262 | | |
263 | 899 | MI->x86opsize = 10; |
264 | 899 | printMemReference(MI, OpNo, O); |
265 | 899 | } |
266 | | |
267 | | static void printf128mem(MCInst *MI, unsigned OpNo, SStream *O) |
268 | 4.76k | { |
269 | 4.76k | SStream_concat0(O, "xmmword ptr "); |
270 | 4.76k | MI->x86opsize = 16; |
271 | 4.76k | printMemReference(MI, OpNo, O); |
272 | 4.76k | } |
273 | | |
274 | | static void printf256mem(MCInst *MI, unsigned OpNo, SStream *O) |
275 | 3.40k | { |
276 | 3.40k | SStream_concat0(O, "ymmword ptr "); |
277 | 3.40k | MI->x86opsize = 32; |
278 | 3.40k | printMemReference(MI, OpNo, O); |
279 | 3.40k | } |
280 | | |
281 | | static void printf512mem(MCInst *MI, unsigned OpNo, SStream *O) |
282 | 2.14k | { |
283 | 2.14k | SStream_concat0(O, "zmmword ptr "); |
284 | 2.14k | MI->x86opsize = 64; |
285 | 2.14k | printMemReference(MI, OpNo, O); |
286 | 2.14k | } |
287 | | #endif |
288 | | |
289 | | static const char *getRegisterName(unsigned RegNo); |
290 | | static void printRegName(SStream *OS, unsigned RegNo) |
291 | 1.03M | { |
292 | 1.03M | SStream_concat0(OS, getRegisterName(RegNo)); |
293 | 1.03M | } |
294 | | |
295 | | // for MASM syntax, 0x123 = 123h, 0xA123 = 0A123h |
296 | | // this function tell us if we need to have prefix 0 in front of a number |
297 | | static bool need_zero_prefix(uint64_t imm) |
298 | 0 | { |
299 | | // find the first hex letter representing imm |
300 | 0 | while(imm >= 0x10) |
301 | 0 | imm >>= 4; |
302 | |
|
303 | 0 | if (imm < 0xa) |
304 | 0 | return false; |
305 | 0 | else // this need 0 prefix |
306 | 0 | return true; |
307 | 0 | } |
308 | | |
309 | | static void printImm(MCInst *MI, SStream *O, int64_t imm, bool positive) |
310 | 271k | { |
311 | 271k | if (positive) { |
312 | | // always print this number in positive form |
313 | 228k | if (MI->csh->syntax == CS_OPT_SYNTAX_MASM) { |
314 | 0 | if (imm < 0) { |
315 | 0 | if (MI->op1_size) { |
316 | 0 | switch(MI->op1_size) { |
317 | 0 | default: |
318 | 0 | break; |
319 | 0 | case 1: |
320 | 0 | imm &= 0xff; |
321 | 0 | break; |
322 | 0 | case 2: |
323 | 0 | imm &= 0xffff; |
324 | 0 | break; |
325 | 0 | case 4: |
326 | 0 | imm &= 0xffffffff; |
327 | 0 | break; |
328 | 0 | } |
329 | 0 | } |
330 | | |
331 | 0 | if (imm == 0x8000000000000000LL) // imm == -imm |
332 | 0 | SStream_concat0(O, "8000000000000000h"); |
333 | 0 | else if (need_zero_prefix(imm)) |
334 | 0 | SStream_concat(O, "0%"PRIx64"h", imm); |
335 | 0 | else |
336 | 0 | SStream_concat(O, "%"PRIx64"h", imm); |
337 | 0 | } else { |
338 | 0 | if (imm > HEX_THRESHOLD) { |
339 | 0 | if (need_zero_prefix(imm)) |
340 | 0 | SStream_concat(O, "0%"PRIx64"h", imm); |
341 | 0 | else |
342 | 0 | SStream_concat(O, "%"PRIx64"h", imm); |
343 | 0 | } else |
344 | 0 | SStream_concat(O, "%"PRIu64, imm); |
345 | 0 | } |
346 | 228k | } else { // Intel syntax |
347 | 228k | if (imm < 0) { |
348 | 3.59k | if (MI->op1_size) { |
349 | 1.00k | switch(MI->op1_size) { |
350 | 1.00k | default: |
351 | 1.00k | break; |
352 | 1.00k | case 1: |
353 | 0 | imm &= 0xff; |
354 | 0 | break; |
355 | 0 | case 2: |
356 | 0 | imm &= 0xffff; |
357 | 0 | break; |
358 | 0 | case 4: |
359 | 0 | imm &= 0xffffffff; |
360 | 0 | break; |
361 | 1.00k | } |
362 | 1.00k | } |
363 | | |
364 | 3.59k | SStream_concat(O, "0x%"PRIx64, imm); |
365 | 225k | } else { |
366 | 225k | if (imm > HEX_THRESHOLD) |
367 | 208k | SStream_concat(O, "0x%"PRIx64, imm); |
368 | 16.3k | else |
369 | 16.3k | SStream_concat(O, "%"PRIu64, imm); |
370 | 225k | } |
371 | 228k | } |
372 | 228k | } else { |
373 | 42.8k | if (MI->csh->syntax == CS_OPT_SYNTAX_MASM) { |
374 | 0 | if (imm < 0) { |
375 | 0 | if (imm == 0x8000000000000000LL) // imm == -imm |
376 | 0 | SStream_concat0(O, "8000000000000000h"); |
377 | 0 | else if (imm < -HEX_THRESHOLD) { |
378 | 0 | if (need_zero_prefix(imm)) |
379 | 0 | SStream_concat(O, "-0%"PRIx64"h", -imm); |
380 | 0 | else |
381 | 0 | SStream_concat(O, "-%"PRIx64"h", -imm); |
382 | 0 | } else |
383 | 0 | SStream_concat(O, "-%"PRIu64, -imm); |
384 | 0 | } else { |
385 | 0 | if (imm > HEX_THRESHOLD) { |
386 | 0 | if (need_zero_prefix(imm)) |
387 | 0 | SStream_concat(O, "0%"PRIx64"h", imm); |
388 | 0 | else |
389 | 0 | SStream_concat(O, "%"PRIx64"h", imm); |
390 | 0 | } else |
391 | 0 | SStream_concat(O, "%"PRIu64, imm); |
392 | 0 | } |
393 | 42.8k | } else { // Intel syntax |
394 | 42.8k | if (imm < 0) { |
395 | 5.54k | if (imm == 0x8000000000000000LL) // imm == -imm |
396 | 0 | SStream_concat0(O, "0x8000000000000000"); |
397 | 5.54k | else if (imm < -HEX_THRESHOLD) |
398 | 4.77k | SStream_concat(O, "-0x%"PRIx64, -imm); |
399 | 767 | else |
400 | 767 | SStream_concat(O, "-%"PRIu64, -imm); |
401 | | |
402 | 37.3k | } else { |
403 | 37.3k | if (imm > HEX_THRESHOLD) |
404 | 30.9k | SStream_concat(O, "0x%"PRIx64, imm); |
405 | 6.39k | else |
406 | 6.39k | SStream_concat(O, "%"PRIu64, imm); |
407 | 37.3k | } |
408 | 42.8k | } |
409 | 42.8k | } |
410 | 271k | } |
411 | | |
412 | | // local printOperand, without updating public operands |
413 | | static void _printOperand(MCInst *MI, unsigned OpNo, SStream *O) |
414 | 384k | { |
415 | 384k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
416 | 384k | if (MCOperand_isReg(Op)) { |
417 | 384k | printRegName(O, MCOperand_getReg(Op)); |
418 | 384k | } else if (MCOperand_isImm(Op)) { |
419 | 0 | int64_t imm = MCOperand_getImm(Op); |
420 | 0 | printImm(MI, O, imm, MI->csh->imm_unsigned); |
421 | 0 | } |
422 | 384k | } |
423 | | |
424 | | #ifndef CAPSTONE_DIET |
425 | | // copy & normalize access info |
426 | | static void get_op_access(cs_struct *h, unsigned int id, uint8_t *access, uint64_t *eflags) |
427 | 1.86M | { |
428 | 1.86M | #ifndef CAPSTONE_DIET |
429 | 1.86M | uint8_t i; |
430 | 1.86M | const uint8_t *arr = X86_get_op_access(h, id, eflags); |
431 | | |
432 | | // initialize access |
433 | 1.86M | memset(access, 0, CS_X86_MAXIMUM_OPERAND_SIZE * sizeof(access[0])); |
434 | | |
435 | 1.86M | if (!arr) { |
436 | 0 | access[0] = 0; |
437 | 0 | return; |
438 | 0 | } |
439 | | |
440 | | // copy to access but zero out CS_AC_IGNORE |
441 | 5.36M | for(i = 0; arr[i]; i++) { |
442 | 3.49M | if (arr[i] != CS_AC_IGNORE) |
443 | 2.97M | access[i] = arr[i]; |
444 | 522k | else |
445 | 522k | access[i] = 0; |
446 | 3.49M | } |
447 | | |
448 | | // mark the end of array |
449 | 1.86M | access[i] = 0; |
450 | 1.86M | #endif |
451 | 1.86M | } |
452 | | #endif |
453 | | |
454 | | static void printSrcIdx(MCInst *MI, unsigned Op, SStream *O) |
455 | 34.4k | { |
456 | 34.4k | MCOperand *SegReg; |
457 | 34.4k | int reg; |
458 | | |
459 | 34.4k | if (MI->csh->detail_opt) { |
460 | 34.4k | #ifndef CAPSTONE_DIET |
461 | 34.4k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
462 | 34.4k | #endif |
463 | | |
464 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
465 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
466 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
467 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_REG_INVALID; |
468 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_REG_INVALID; |
469 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
470 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
471 | | |
472 | 34.4k | #ifndef CAPSTONE_DIET |
473 | 34.4k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
474 | 34.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
475 | 34.4k | #endif |
476 | 34.4k | } |
477 | | |
478 | 34.4k | SegReg = MCInst_getOperand(MI, Op + 1); |
479 | 34.4k | reg = MCOperand_getReg(SegReg); |
480 | | |
481 | | // If this has a segment register, print it. |
482 | 34.4k | if (reg) { |
483 | 1.09k | _printOperand(MI, Op + 1, O); |
484 | 1.09k | if (MI->csh->detail_opt) { |
485 | 1.09k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_register_map(reg); |
486 | 1.09k | } |
487 | 1.09k | SStream_concat0(O, ":"); |
488 | 1.09k | } |
489 | | |
490 | 34.4k | SStream_concat0(O, "["); |
491 | 34.4k | set_mem_access(MI, true); |
492 | 34.4k | printOperand(MI, Op, O); |
493 | 34.4k | SStream_concat0(O, "]"); |
494 | 34.4k | set_mem_access(MI, false); |
495 | 34.4k | } |
496 | | |
497 | | static void printDstIdx(MCInst *MI, unsigned Op, SStream *O) |
498 | 46.5k | { |
499 | 46.5k | if (MI->csh->detail_opt) { |
500 | 46.5k | #ifndef CAPSTONE_DIET |
501 | 46.5k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
502 | 46.5k | #endif |
503 | | |
504 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
505 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
506 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
507 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_REG_INVALID; |
508 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_REG_INVALID; |
509 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
510 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
511 | | |
512 | 46.5k | #ifndef CAPSTONE_DIET |
513 | 46.5k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
514 | 46.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
515 | 46.5k | #endif |
516 | 46.5k | } |
517 | | |
518 | | // DI accesses are always ES-based on non-64bit mode |
519 | 46.5k | if (MI->csh->mode != CS_MODE_64) { |
520 | 30.4k | SStream_concat0(O, "es:["); |
521 | 30.4k | if (MI->csh->detail_opt) { |
522 | 30.4k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_ES; |
523 | 30.4k | } |
524 | 30.4k | } else |
525 | 16.0k | SStream_concat0(O, "["); |
526 | | |
527 | 46.5k | set_mem_access(MI, true); |
528 | 46.5k | printOperand(MI, Op, O); |
529 | 46.5k | SStream_concat0(O, "]"); |
530 | 46.5k | set_mem_access(MI, false); |
531 | 46.5k | } |
532 | | |
533 | | static void printSrcIdx8(MCInst *MI, unsigned OpNo, SStream *O) |
534 | 14.9k | { |
535 | 14.9k | SStream_concat0(O, "byte ptr "); |
536 | 14.9k | MI->x86opsize = 1; |
537 | 14.9k | printSrcIdx(MI, OpNo, O); |
538 | 14.9k | } |
539 | | |
540 | | static void printSrcIdx16(MCInst *MI, unsigned OpNo, SStream *O) |
541 | 6.31k | { |
542 | 6.31k | SStream_concat0(O, "word ptr "); |
543 | 6.31k | MI->x86opsize = 2; |
544 | 6.31k | printSrcIdx(MI, OpNo, O); |
545 | 6.31k | } |
546 | | |
547 | | static void printSrcIdx32(MCInst *MI, unsigned OpNo, SStream *O) |
548 | 11.2k | { |
549 | 11.2k | SStream_concat0(O, "dword ptr "); |
550 | 11.2k | MI->x86opsize = 4; |
551 | 11.2k | printSrcIdx(MI, OpNo, O); |
552 | 11.2k | } |
553 | | |
554 | | static void printSrcIdx64(MCInst *MI, unsigned OpNo, SStream *O) |
555 | 1.97k | { |
556 | 1.97k | SStream_concat0(O, "qword ptr "); |
557 | 1.97k | MI->x86opsize = 8; |
558 | 1.97k | printSrcIdx(MI, OpNo, O); |
559 | 1.97k | } |
560 | | |
561 | | static void printDstIdx8(MCInst *MI, unsigned OpNo, SStream *O) |
562 | 19.5k | { |
563 | 19.5k | SStream_concat0(O, "byte ptr "); |
564 | 19.5k | MI->x86opsize = 1; |
565 | 19.5k | printDstIdx(MI, OpNo, O); |
566 | 19.5k | } |
567 | | |
568 | | static void printDstIdx16(MCInst *MI, unsigned OpNo, SStream *O) |
569 | 7.77k | { |
570 | 7.77k | SStream_concat0(O, "word ptr "); |
571 | 7.77k | MI->x86opsize = 2; |
572 | 7.77k | printDstIdx(MI, OpNo, O); |
573 | 7.77k | } |
574 | | |
575 | | static void printDstIdx32(MCInst *MI, unsigned OpNo, SStream *O) |
576 | 16.5k | { |
577 | 16.5k | SStream_concat0(O, "dword ptr "); |
578 | 16.5k | MI->x86opsize = 4; |
579 | 16.5k | printDstIdx(MI, OpNo, O); |
580 | 16.5k | } |
581 | | |
582 | | static void printDstIdx64(MCInst *MI, unsigned OpNo, SStream *O) |
583 | 2.69k | { |
584 | 2.69k | SStream_concat0(O, "qword ptr "); |
585 | 2.69k | MI->x86opsize = 8; |
586 | 2.69k | printDstIdx(MI, OpNo, O); |
587 | 2.69k | } |
588 | | |
589 | | static void printMemOffset(MCInst *MI, unsigned Op, SStream *O) |
590 | 8.97k | { |
591 | 8.97k | MCOperand *DispSpec = MCInst_getOperand(MI, Op); |
592 | 8.97k | MCOperand *SegReg = MCInst_getOperand(MI, Op + 1); |
593 | 8.97k | int reg; |
594 | | |
595 | 8.97k | if (MI->csh->detail_opt) { |
596 | 8.97k | #ifndef CAPSTONE_DIET |
597 | 8.97k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
598 | 8.97k | #endif |
599 | | |
600 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
601 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
602 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
603 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_REG_INVALID; |
604 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_REG_INVALID; |
605 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
606 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
607 | | |
608 | 8.97k | #ifndef CAPSTONE_DIET |
609 | 8.97k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
610 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
611 | 8.97k | #endif |
612 | 8.97k | } |
613 | | |
614 | | // If this has a segment register, print it. |
615 | 8.97k | reg = MCOperand_getReg(SegReg); |
616 | 8.97k | if (reg) { |
617 | 511 | _printOperand(MI, Op + 1, O); |
618 | 511 | SStream_concat0(O, ":"); |
619 | 511 | if (MI->csh->detail_opt) { |
620 | 511 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_register_map(reg); |
621 | 511 | } |
622 | 511 | } |
623 | | |
624 | 8.97k | SStream_concat0(O, "["); |
625 | | |
626 | 8.97k | if (MCOperand_isImm(DispSpec)) { |
627 | 8.97k | int64_t imm = MCOperand_getImm(DispSpec); |
628 | 8.97k | if (MI->csh->detail_opt) |
629 | 8.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = imm; |
630 | | |
631 | 8.97k | if (imm < 0) |
632 | 1.61k | printImm(MI, O, arch_masks[MI->csh->mode] & imm, true); |
633 | 7.36k | else |
634 | 7.36k | printImm(MI, O, imm, true); |
635 | 8.97k | } |
636 | | |
637 | 8.97k | SStream_concat0(O, "]"); |
638 | | |
639 | 8.97k | if (MI->csh->detail_opt) |
640 | 8.97k | MI->flat_insn->detail->x86.op_count++; |
641 | | |
642 | 8.97k | if (MI->op1_size == 0) |
643 | 8.97k | MI->op1_size = MI->x86opsize; |
644 | 8.97k | } |
645 | | |
646 | | static void printU8Imm(MCInst *MI, unsigned Op, SStream *O) |
647 | 41.6k | { |
648 | 41.6k | uint8_t val = MCOperand_getImm(MCInst_getOperand(MI, Op)) & 0xff; |
649 | | |
650 | 41.6k | printImm(MI, O, val, true); |
651 | | |
652 | 41.6k | if (MI->csh->detail_opt) { |
653 | 41.6k | #ifndef CAPSTONE_DIET |
654 | 41.6k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
655 | 41.6k | #endif |
656 | | |
657 | 41.6k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM; |
658 | 41.6k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = val; |
659 | 41.6k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = 1; |
660 | | |
661 | 41.6k | #ifndef CAPSTONE_DIET |
662 | 41.6k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
663 | 41.6k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
664 | 41.6k | #endif |
665 | | |
666 | 41.6k | MI->flat_insn->detail->x86.op_count++; |
667 | 41.6k | } |
668 | 41.6k | } |
669 | | |
670 | | static void printMemOffs8(MCInst *MI, unsigned OpNo, SStream *O) |
671 | 4.97k | { |
672 | 4.97k | SStream_concat0(O, "byte ptr "); |
673 | 4.97k | MI->x86opsize = 1; |
674 | 4.97k | printMemOffset(MI, OpNo, O); |
675 | 4.97k | } |
676 | | |
677 | | static void printMemOffs16(MCInst *MI, unsigned OpNo, SStream *O) |
678 | 1.26k | { |
679 | 1.26k | SStream_concat0(O, "word ptr "); |
680 | 1.26k | MI->x86opsize = 2; |
681 | 1.26k | printMemOffset(MI, OpNo, O); |
682 | 1.26k | } |
683 | | |
684 | | static void printMemOffs32(MCInst *MI, unsigned OpNo, SStream *O) |
685 | 2.34k | { |
686 | 2.34k | SStream_concat0(O, "dword ptr "); |
687 | 2.34k | MI->x86opsize = 4; |
688 | 2.34k | printMemOffset(MI, OpNo, O); |
689 | 2.34k | } |
690 | | |
691 | | static void printMemOffs64(MCInst *MI, unsigned OpNo, SStream *O) |
692 | 392 | { |
693 | 392 | SStream_concat0(O, "qword ptr "); |
694 | 392 | MI->x86opsize = 8; |
695 | 392 | printMemOffset(MI, OpNo, O); |
696 | 392 | } |
697 | | |
698 | | static void printInstruction(MCInst *MI, SStream *O); |
699 | | |
700 | | void X86_Intel_printInst(MCInst *MI, SStream *O, void *Info) |
701 | 726k | { |
702 | 726k | x86_reg reg, reg2; |
703 | 726k | enum cs_ac_type access1, access2; |
704 | | |
705 | | // printf("opcode = %u\n", MCInst_getOpcode(MI)); |
706 | | |
707 | | // perhaps this instruction does not need printer |
708 | 726k | if (MI->assembly[0]) { |
709 | 0 | strncpy(O->buffer, MI->assembly, sizeof(O->buffer)); |
710 | 0 | return; |
711 | 0 | } |
712 | | |
713 | 726k | X86_lockrep(MI, O); |
714 | 726k | printInstruction(MI, O); |
715 | | |
716 | 726k | reg = X86_insn_reg_intel(MCInst_getOpcode(MI), &access1); |
717 | 726k | if (MI->csh->detail_opt) { |
718 | 726k | #ifndef CAPSTONE_DIET |
719 | 726k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE] = {0}; |
720 | 726k | #endif |
721 | | |
722 | | // first op can be embedded in the asm by llvm. |
723 | | // so we have to add the missing register as the first operand |
724 | 726k | if (reg) { |
725 | | // shift all the ops right to leave 1st slot for this new register op |
726 | 74.3k | memmove(&(MI->flat_insn->detail->x86.operands[1]), &(MI->flat_insn->detail->x86.operands[0]), |
727 | 74.3k | sizeof(MI->flat_insn->detail->x86.operands[0]) * (ARR_SIZE(MI->flat_insn->detail->x86.operands) - 1)); |
728 | 74.3k | MI->flat_insn->detail->x86.operands[0].type = X86_OP_REG; |
729 | 74.3k | MI->flat_insn->detail->x86.operands[0].reg = reg; |
730 | 74.3k | MI->flat_insn->detail->x86.operands[0].size = MI->csh->regsize_map[reg]; |
731 | 74.3k | MI->flat_insn->detail->x86.operands[0].access = access1; |
732 | 74.3k | MI->flat_insn->detail->x86.op_count++; |
733 | 652k | } else { |
734 | 652k | if (X86_insn_reg_intel2(MCInst_getOpcode(MI), ®, &access1, ®2, &access2)) { |
735 | 13.1k | MI->flat_insn->detail->x86.operands[0].type = X86_OP_REG; |
736 | 13.1k | MI->flat_insn->detail->x86.operands[0].reg = reg; |
737 | 13.1k | MI->flat_insn->detail->x86.operands[0].size = MI->csh->regsize_map[reg]; |
738 | 13.1k | MI->flat_insn->detail->x86.operands[0].access = access1; |
739 | 13.1k | MI->flat_insn->detail->x86.operands[1].type = X86_OP_REG; |
740 | 13.1k | MI->flat_insn->detail->x86.operands[1].reg = reg2; |
741 | 13.1k | MI->flat_insn->detail->x86.operands[1].size = MI->csh->regsize_map[reg2]; |
742 | 13.1k | MI->flat_insn->detail->x86.operands[1].access = access2; |
743 | 13.1k | MI->flat_insn->detail->x86.op_count = 2; |
744 | 13.1k | } |
745 | 652k | } |
746 | | |
747 | 726k | #ifndef CAPSTONE_DIET |
748 | 726k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
749 | 726k | MI->flat_insn->detail->x86.operands[0].access = access[0]; |
750 | 726k | MI->flat_insn->detail->x86.operands[1].access = access[1]; |
751 | 726k | #endif |
752 | 726k | } |
753 | | |
754 | 726k | if (MI->op1_size == 0 && reg) |
755 | 57.4k | MI->op1_size = MI->csh->regsize_map[reg]; |
756 | 726k | } |
757 | | |
758 | | /// printPCRelImm - This is used to print an immediate value that ends up |
759 | | /// being encoded as a pc-relative value. |
760 | | static void printPCRelImm(MCInst *MI, unsigned OpNo, SStream *O) |
761 | 43.8k | { |
762 | 43.8k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
763 | 43.8k | if (MCOperand_isImm(Op)) { |
764 | 43.8k | int64_t imm = MCOperand_getImm(Op) + MI->flat_insn->size + MI->address; |
765 | 43.8k | uint8_t opsize = X86_immediate_size(MI->Opcode, NULL); |
766 | | |
767 | | // truncate imm for non-64bit |
768 | 43.8k | if (MI->csh->mode != CS_MODE_64) { |
769 | 30.2k | imm = imm & 0xffffffff; |
770 | 30.2k | } |
771 | | |
772 | 43.8k | printImm(MI, O, imm, true); |
773 | | |
774 | 43.8k | if (MI->csh->detail_opt) { |
775 | 43.8k | #ifndef CAPSTONE_DIET |
776 | 43.8k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
777 | 43.8k | #endif |
778 | | |
779 | 43.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM; |
780 | | // if op_count > 0, then this operand's size is taken from the destination op |
781 | 43.8k | if (MI->flat_insn->detail->x86.op_count > 0) |
782 | 0 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->flat_insn->detail->x86.operands[0].size; |
783 | 43.8k | else if (opsize > 0) |
784 | 1.80k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = opsize; |
785 | 42.0k | else |
786 | 42.0k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->imm_size; |
787 | 43.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = imm; |
788 | | |
789 | 43.8k | #ifndef CAPSTONE_DIET |
790 | 43.8k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
791 | 43.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
792 | 43.8k | #endif |
793 | | |
794 | 43.8k | MI->flat_insn->detail->x86.op_count++; |
795 | 43.8k | } |
796 | | |
797 | 43.8k | if (MI->op1_size == 0) |
798 | 43.8k | MI->op1_size = MI->imm_size; |
799 | 43.8k | } |
800 | 43.8k | } |
801 | | |
802 | | static void printOperand(MCInst *MI, unsigned OpNo, SStream *O) |
803 | 732k | { |
804 | 732k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
805 | | |
806 | 732k | if (MCOperand_isReg(Op)) { |
807 | 646k | unsigned int reg = MCOperand_getReg(Op); |
808 | | |
809 | 646k | printRegName(O, reg); |
810 | 646k | if (MI->csh->detail_opt) { |
811 | 646k | if (MI->csh->doing_mem) { |
812 | 81.0k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_register_map(reg); |
813 | 565k | } else { |
814 | 565k | #ifndef CAPSTONE_DIET |
815 | 565k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
816 | 565k | #endif |
817 | | |
818 | 565k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_REG; |
819 | 565k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].reg = X86_register_map(reg); |
820 | 565k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->csh->regsize_map[X86_register_map(reg)]; |
821 | | |
822 | 565k | #ifndef CAPSTONE_DIET |
823 | 565k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
824 | 565k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
825 | 565k | #endif |
826 | | |
827 | 565k | MI->flat_insn->detail->x86.op_count++; |
828 | 565k | } |
829 | 646k | } |
830 | | |
831 | 646k | if (MI->op1_size == 0) |
832 | 323k | MI->op1_size = MI->csh->regsize_map[X86_register_map(reg)]; |
833 | 646k | } else if (MCOperand_isImm(Op)) { |
834 | 85.8k | uint8_t encsize; |
835 | 85.8k | int64_t imm = MCOperand_getImm(Op); |
836 | 85.8k | uint8_t opsize = X86_immediate_size(MCInst_getOpcode(MI), &encsize); |
837 | | |
838 | 85.8k | if (opsize == 1) // print 1 byte immediate in positive form |
839 | 38.7k | imm = imm & 0xff; |
840 | | |
841 | | // printf(">>> id = %u\n", MI->flat_insn->id); |
842 | 85.8k | switch(MI->flat_insn->id) { |
843 | 42.8k | default: |
844 | 42.8k | printImm(MI, O, imm, MI->csh->imm_unsigned); |
845 | 42.8k | break; |
846 | | |
847 | 417 | case X86_INS_MOVABS: |
848 | 12.1k | case X86_INS_MOV: |
849 | | // do not print number in negative form |
850 | 12.1k | printImm(MI, O, imm, true); |
851 | 12.1k | break; |
852 | | |
853 | 0 | case X86_INS_IN: |
854 | 0 | case X86_INS_OUT: |
855 | 0 | case X86_INS_INT: |
856 | | // do not print number in negative form |
857 | 0 | imm = imm & 0xff; |
858 | 0 | printImm(MI, O, imm, true); |
859 | 0 | break; |
860 | | |
861 | 1.67k | case X86_INS_LCALL: |
862 | 3.22k | case X86_INS_LJMP: |
863 | 3.22k | case X86_INS_JMP: |
864 | | // always print address in positive form |
865 | 3.22k | if (OpNo == 1) { // ptr16 part |
866 | 1.61k | imm = imm & 0xffff; |
867 | 1.61k | opsize = 2; |
868 | 1.61k | } else |
869 | 1.61k | opsize = 4; |
870 | 3.22k | printImm(MI, O, imm, true); |
871 | 3.22k | break; |
872 | | |
873 | 6.64k | case X86_INS_AND: |
874 | 13.0k | case X86_INS_OR: |
875 | 19.3k | case X86_INS_XOR: |
876 | | // do not print number in negative form |
877 | 19.3k | if (imm >= 0 && imm <= HEX_THRESHOLD) |
878 | 2.61k | printImm(MI, O, imm, true); |
879 | 16.7k | else { |
880 | 16.7k | imm = arch_masks[opsize? opsize : MI->imm_size] & imm; |
881 | 16.7k | printImm(MI, O, imm, true); |
882 | 16.7k | } |
883 | 19.3k | break; |
884 | | |
885 | 6.70k | case X86_INS_RET: |
886 | 8.23k | case X86_INS_RETF: |
887 | | // RET imm16 |
888 | 8.23k | if (imm >= 0 && imm <= HEX_THRESHOLD) |
889 | 945 | printImm(MI, O, imm, true); |
890 | 7.29k | else { |
891 | 7.29k | imm = 0xffff & imm; |
892 | 7.29k | printImm(MI, O, imm, true); |
893 | 7.29k | } |
894 | 8.23k | break; |
895 | 85.8k | } |
896 | | |
897 | 85.8k | if (MI->csh->detail_opt) { |
898 | 85.8k | if (MI->csh->doing_mem) { |
899 | 0 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = imm; |
900 | 85.8k | } else { |
901 | 85.8k | #ifndef CAPSTONE_DIET |
902 | 85.8k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
903 | 85.8k | #endif |
904 | | |
905 | 85.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM; |
906 | 85.8k | if (opsize > 0) { |
907 | 73.5k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = opsize; |
908 | 73.5k | MI->flat_insn->detail->x86.encoding.imm_size = encsize; |
909 | 73.5k | } else if (MI->flat_insn->detail->x86.op_count > 0) { |
910 | 2.62k | if (MI->flat_insn->id != X86_INS_LCALL && MI->flat_insn->id != X86_INS_LJMP) { |
911 | 2.62k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = |
912 | 2.62k | MI->flat_insn->detail->x86.operands[0].size; |
913 | 2.62k | } else |
914 | 0 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->imm_size; |
915 | 2.62k | } else |
916 | 9.68k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->imm_size; |
917 | 85.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = imm; |
918 | | |
919 | 85.8k | #ifndef CAPSTONE_DIET |
920 | 85.8k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
921 | 85.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
922 | 85.8k | #endif |
923 | | |
924 | 85.8k | MI->flat_insn->detail->x86.op_count++; |
925 | 85.8k | } |
926 | 85.8k | } |
927 | 85.8k | } |
928 | 732k | } |
929 | | |
930 | | static void printMemReference(MCInst *MI, unsigned Op, SStream *O) |
931 | 310k | { |
932 | 310k | bool NeedPlus = false; |
933 | 310k | MCOperand *BaseReg = MCInst_getOperand(MI, Op + X86_AddrBaseReg); |
934 | 310k | uint64_t ScaleVal = MCOperand_getImm(MCInst_getOperand(MI, Op + X86_AddrScaleAmt)); |
935 | 310k | MCOperand *IndexReg = MCInst_getOperand(MI, Op + X86_AddrIndexReg); |
936 | 310k | MCOperand *DispSpec = MCInst_getOperand(MI, Op + X86_AddrDisp); |
937 | 310k | MCOperand *SegReg = MCInst_getOperand(MI, Op + X86_AddrSegmentReg); |
938 | 310k | int reg; |
939 | | |
940 | 310k | if (MI->csh->detail_opt) { |
941 | 310k | #ifndef CAPSTONE_DIET |
942 | 310k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
943 | 310k | #endif |
944 | | |
945 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
946 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
947 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
948 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_register_map(MCOperand_getReg(BaseReg)); |
949 | 310k | if (MCOperand_getReg(IndexReg) != X86_EIZ) { |
950 | 308k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_register_map(MCOperand_getReg(IndexReg)); |
951 | 308k | } |
952 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = (int)ScaleVal; |
953 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
954 | | |
955 | 310k | #ifndef CAPSTONE_DIET |
956 | 310k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
957 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
958 | 310k | #endif |
959 | 310k | } |
960 | | |
961 | | // If this has a segment register, print it. |
962 | 310k | reg = MCOperand_getReg(SegReg); |
963 | 310k | if (reg) { |
964 | 8.74k | _printOperand(MI, Op + X86_AddrSegmentReg, O); |
965 | 8.74k | if (MI->csh->detail_opt) { |
966 | 8.74k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_register_map(reg); |
967 | 8.74k | } |
968 | 8.74k | SStream_concat0(O, ":"); |
969 | 8.74k | } |
970 | | |
971 | 310k | SStream_concat0(O, "["); |
972 | | |
973 | 310k | if (MCOperand_getReg(BaseReg)) { |
974 | 304k | _printOperand(MI, Op + X86_AddrBaseReg, O); |
975 | 304k | NeedPlus = true; |
976 | 304k | } |
977 | | |
978 | 310k | if (MCOperand_getReg(IndexReg) && MCOperand_getReg(IndexReg) != X86_EIZ) { |
979 | 69.8k | if (NeedPlus) SStream_concat0(O, " + "); |
980 | 69.8k | _printOperand(MI, Op + X86_AddrIndexReg, O); |
981 | 69.8k | if (ScaleVal != 1) |
982 | 10.0k | SStream_concat(O, "*%u", ScaleVal); |
983 | 69.8k | NeedPlus = true; |
984 | 69.8k | } |
985 | | |
986 | 310k | if (MCOperand_isImm(DispSpec)) { |
987 | 310k | int64_t DispVal = MCOperand_getImm(DispSpec); |
988 | 310k | if (MI->csh->detail_opt) |
989 | 310k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = DispVal; |
990 | 310k | if (DispVal) { |
991 | 91.4k | if (NeedPlus) { |
992 | 86.9k | if (DispVal < 0) { |
993 | 34.7k | SStream_concat0(O, " - "); |
994 | 34.7k | printImm(MI, O, -DispVal, true); |
995 | 52.1k | } else { |
996 | 52.1k | SStream_concat0(O, " + "); |
997 | 52.1k | printImm(MI, O, DispVal, true); |
998 | 52.1k | } |
999 | 86.9k | } else { |
1000 | | // memory reference to an immediate address |
1001 | 4.48k | if (MI->csh->mode == CS_MODE_64) |
1002 | 449 | MI->op1_size = 8; |
1003 | 4.48k | if (DispVal < 0) { |
1004 | 1.62k | printImm(MI, O, arch_masks[MI->csh->mode] & DispVal, true); |
1005 | 2.85k | } else { |
1006 | 2.85k | printImm(MI, O, DispVal, true); |
1007 | 2.85k | } |
1008 | 4.48k | } |
1009 | | |
1010 | 219k | } else { |
1011 | | // DispVal = 0 |
1012 | 219k | if (!NeedPlus) // [0] |
1013 | 550 | SStream_concat0(O, "0"); |
1014 | 219k | } |
1015 | 310k | } |
1016 | | |
1017 | 310k | SStream_concat0(O, "]"); |
1018 | | |
1019 | 310k | if (MI->csh->detail_opt) |
1020 | 310k | MI->flat_insn->detail->x86.op_count++; |
1021 | | |
1022 | 310k | if (MI->op1_size == 0) |
1023 | 211k | MI->op1_size = MI->x86opsize; |
1024 | 310k | } |
1025 | | |
1026 | | static void printanymem(MCInst *MI, unsigned OpNo, SStream *O) |
1027 | 7.16k | { |
1028 | 7.16k | switch(MI->Opcode) { |
1029 | 287 | default: break; |
1030 | 658 | case X86_LEA16r: |
1031 | 658 | MI->x86opsize = 2; |
1032 | 658 | break; |
1033 | 796 | case X86_LEA32r: |
1034 | 1.49k | case X86_LEA64_32r: |
1035 | 1.49k | MI->x86opsize = 4; |
1036 | 1.49k | break; |
1037 | 417 | case X86_LEA64r: |
1038 | 417 | MI->x86opsize = 8; |
1039 | 417 | break; |
1040 | 0 | #ifndef CAPSTONE_X86_REDUCE |
1041 | 336 | case X86_BNDCL32rm: |
1042 | 788 | case X86_BNDCN32rm: |
1043 | 1.21k | case X86_BNDCU32rm: |
1044 | 1.92k | case X86_BNDSTXmr: |
1045 | 2.57k | case X86_BNDLDXrm: |
1046 | 2.87k | case X86_BNDCL64rm: |
1047 | 3.43k | case X86_BNDCN64rm: |
1048 | 4.30k | case X86_BNDCU64rm: |
1049 | 4.30k | MI->x86opsize = 16; |
1050 | 4.30k | break; |
1051 | 7.16k | #endif |
1052 | 7.16k | } |
1053 | | |
1054 | 7.16k | printMemReference(MI, OpNo, O); |
1055 | 7.16k | } |
1056 | | |
1057 | | #ifdef CAPSTONE_X86_REDUCE |
1058 | | #include "X86GenAsmWriter1_reduce.inc" |
1059 | | #else |
1060 | | #include "X86GenAsmWriter1.inc" |
1061 | | #endif |
1062 | | |
1063 | | #include "X86GenRegisterName1.inc" |
1064 | | |
1065 | | #endif |