/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 | | // disable MSVC's warning on strncpy() |
22 | | #pragma warning(disable : 4996) |
23 | | // disable MSVC's warning on strncpy() |
24 | | #pragma warning(disable : 28719) |
25 | | #endif |
26 | | |
27 | | #if !defined(CAPSTONE_HAS_OSXKERNEL) |
28 | | #include <ctype.h> |
29 | | #endif |
30 | | #include <capstone/platform.h> |
31 | | |
32 | | #if defined(CAPSTONE_HAS_OSXKERNEL) |
33 | | #include <Availability.h> |
34 | | #include <libkern/libkern.h> |
35 | | #else |
36 | | #include <stdio.h> |
37 | | #include <stdlib.h> |
38 | | #endif |
39 | | #include <string.h> |
40 | | |
41 | | #include "../../utils.h" |
42 | | #include "../../MCInst.h" |
43 | | #include "../../SStream.h" |
44 | | #include "../../MCRegisterInfo.h" |
45 | | |
46 | | #include "X86InstPrinter.h" |
47 | | #include "X86Mapping.h" |
48 | | #include "X86InstPrinterCommon.h" |
49 | | |
50 | | #define GET_INSTRINFO_ENUM |
51 | | #ifdef CAPSTONE_X86_REDUCE |
52 | | #include "X86GenInstrInfo_reduce.inc" |
53 | | #else |
54 | | #include "X86GenInstrInfo.inc" |
55 | | #endif |
56 | | |
57 | | #define GET_REGINFO_ENUM |
58 | | #include "X86GenRegisterInfo.inc" |
59 | | |
60 | | #include "X86BaseInfo.h" |
61 | | |
62 | | static void printMemReference(MCInst *MI, unsigned Op, SStream *O); |
63 | | static void printOperand(MCInst *MI, unsigned OpNo, SStream *O); |
64 | | |
65 | | static void set_mem_access(MCInst *MI, bool status) |
66 | 150k | { |
67 | 150k | if (MI->csh->detail_opt != CS_OPT_ON) |
68 | 0 | return; |
69 | | |
70 | 150k | MI->csh->doing_mem = status; |
71 | 150k | if (!status) |
72 | | // done, create the next operand slot |
73 | 75.1k | MI->flat_insn->detail->x86.op_count++; |
74 | 150k | } |
75 | | |
76 | | static void printopaquemem(MCInst *MI, unsigned OpNo, SStream *O) |
77 | 13.7k | { |
78 | | // FIXME: do this with autogen |
79 | | // printf(">>> ID = %u\n", MI->flat_insn->id); |
80 | 13.7k | switch (MI->flat_insn->id) { |
81 | 4.73k | default: |
82 | 4.73k | SStream_concat0(O, "ptr "); |
83 | 4.73k | break; |
84 | 1.41k | case X86_INS_SGDT: |
85 | 2.86k | case X86_INS_SIDT: |
86 | 4.03k | case X86_INS_LGDT: |
87 | 5.30k | case X86_INS_LIDT: |
88 | 5.85k | case X86_INS_FXRSTOR: |
89 | 6.48k | case X86_INS_FXSAVE: |
90 | 7.66k | case X86_INS_LJMP: |
91 | 9.06k | case X86_INS_LCALL: |
92 | | // do not print "ptr" |
93 | 9.06k | break; |
94 | 13.7k | } |
95 | | |
96 | 13.7k | switch (MI->csh->mode) { |
97 | 4.43k | case CS_MODE_16: |
98 | 4.43k | switch (MI->flat_insn->id) { |
99 | 1.60k | default: |
100 | 1.60k | MI->x86opsize = 2; |
101 | 1.60k | break; |
102 | 450 | case X86_INS_LJMP: |
103 | 963 | case X86_INS_LCALL: |
104 | 963 | MI->x86opsize = 4; |
105 | 963 | break; |
106 | 437 | case X86_INS_SGDT: |
107 | 907 | case X86_INS_SIDT: |
108 | 1.45k | case X86_INS_LGDT: |
109 | 1.87k | case X86_INS_LIDT: |
110 | 1.87k | MI->x86opsize = 6; |
111 | 1.87k | break; |
112 | 4.43k | } |
113 | 4.43k | break; |
114 | 5.31k | case CS_MODE_32: |
115 | 5.31k | switch (MI->flat_insn->id) { |
116 | 2.17k | default: |
117 | 2.17k | MI->x86opsize = 4; |
118 | 2.17k | break; |
119 | 274 | case X86_INS_LJMP: |
120 | 985 | case X86_INS_JMP: |
121 | 1.38k | case X86_INS_LCALL: |
122 | 1.87k | case X86_INS_SGDT: |
123 | 2.35k | case X86_INS_SIDT: |
124 | 2.69k | case X86_INS_LGDT: |
125 | 3.13k | case X86_INS_LIDT: |
126 | 3.13k | MI->x86opsize = 6; |
127 | 3.13k | break; |
128 | 5.31k | } |
129 | 5.31k | break; |
130 | 5.31k | case CS_MODE_64: |
131 | 4.04k | switch (MI->flat_insn->id) { |
132 | 1.42k | default: |
133 | 1.42k | MI->x86opsize = 8; |
134 | 1.42k | break; |
135 | 462 | case X86_INS_LJMP: |
136 | 948 | case X86_INS_LCALL: |
137 | 1.43k | case X86_INS_SGDT: |
138 | 1.93k | case X86_INS_SIDT: |
139 | 2.21k | case X86_INS_LGDT: |
140 | 2.62k | case X86_INS_LIDT: |
141 | 2.62k | MI->x86opsize = 10; |
142 | 2.62k | break; |
143 | 4.04k | } |
144 | 4.04k | break; |
145 | 4.04k | default: // never reach |
146 | 0 | break; |
147 | 13.7k | } |
148 | | |
149 | 13.7k | printMemReference(MI, OpNo, O); |
150 | 13.7k | } |
151 | | |
152 | | static void printi8mem(MCInst *MI, unsigned OpNo, SStream *O) |
153 | 99.8k | { |
154 | 99.8k | SStream_concat0(O, "byte ptr "); |
155 | 99.8k | MI->x86opsize = 1; |
156 | 99.8k | printMemReference(MI, OpNo, O); |
157 | 99.8k | } |
158 | | |
159 | | static void printi16mem(MCInst *MI, unsigned OpNo, SStream *O) |
160 | 26.3k | { |
161 | 26.3k | MI->x86opsize = 2; |
162 | 26.3k | SStream_concat0(O, "word ptr "); |
163 | 26.3k | printMemReference(MI, OpNo, O); |
164 | 26.3k | } |
165 | | |
166 | | static void printi32mem(MCInst *MI, unsigned OpNo, SStream *O) |
167 | 50.5k | { |
168 | 50.5k | MI->x86opsize = 4; |
169 | 50.5k | SStream_concat0(O, "dword ptr "); |
170 | 50.5k | printMemReference(MI, OpNo, O); |
171 | 50.5k | } |
172 | | |
173 | | static void printi64mem(MCInst *MI, unsigned OpNo, SStream *O) |
174 | 22.6k | { |
175 | 22.6k | SStream_concat0(O, "qword ptr "); |
176 | 22.6k | MI->x86opsize = 8; |
177 | 22.6k | printMemReference(MI, OpNo, O); |
178 | 22.6k | } |
179 | | |
180 | | static void printi128mem(MCInst *MI, unsigned OpNo, SStream *O) |
181 | 7.60k | { |
182 | 7.60k | SStream_concat0(O, "xmmword ptr "); |
183 | 7.60k | MI->x86opsize = 16; |
184 | 7.60k | printMemReference(MI, OpNo, O); |
185 | 7.60k | } |
186 | | |
187 | | static void printi512mem(MCInst *MI, unsigned OpNo, SStream *O) |
188 | 4.65k | { |
189 | 4.65k | SStream_concat0(O, "zmmword ptr "); |
190 | 4.65k | MI->x86opsize = 64; |
191 | 4.65k | printMemReference(MI, OpNo, O); |
192 | 4.65k | } |
193 | | |
194 | | #ifndef CAPSTONE_X86_REDUCE |
195 | | static void printi256mem(MCInst *MI, unsigned OpNo, SStream *O) |
196 | 4.90k | { |
197 | 4.90k | SStream_concat0(O, "ymmword ptr "); |
198 | 4.90k | MI->x86opsize = 32; |
199 | 4.90k | printMemReference(MI, OpNo, O); |
200 | 4.90k | } |
201 | | |
202 | | static void printf32mem(MCInst *MI, unsigned OpNo, SStream *O) |
203 | 7.96k | { |
204 | 7.96k | switch (MCInst_getOpcode(MI)) { |
205 | 6.06k | default: |
206 | 6.06k | SStream_concat0(O, "dword ptr "); |
207 | 6.06k | MI->x86opsize = 4; |
208 | 6.06k | break; |
209 | 551 | case X86_FSTENVm: |
210 | 1.89k | case X86_FLDENVm: |
211 | | // TODO: fix this in tablegen instead |
212 | 1.89k | switch (MI->csh->mode) { |
213 | 0 | default: // never reach |
214 | 0 | break; |
215 | 634 | case CS_MODE_16: |
216 | 634 | MI->x86opsize = 14; |
217 | 634 | break; |
218 | 736 | case CS_MODE_32: |
219 | 1.26k | case CS_MODE_64: |
220 | 1.26k | MI->x86opsize = 28; |
221 | 1.26k | break; |
222 | 1.89k | } |
223 | 1.89k | break; |
224 | 7.96k | } |
225 | | |
226 | 7.96k | printMemReference(MI, OpNo, O); |
227 | 7.96k | } |
228 | | |
229 | | static void printf64mem(MCInst *MI, unsigned OpNo, SStream *O) |
230 | 3.05k | { |
231 | | // TODO: fix COMISD in Tablegen instead (#1456) |
232 | 3.05k | if (MI->op1_size == 16) { |
233 | | // printf("printf64mem id = %u\n", MCInst_getOpcode(MI)); |
234 | 1.28k | switch (MCInst_getOpcode(MI)) { |
235 | 1.28k | default: |
236 | 1.28k | SStream_concat0(O, "qword ptr "); |
237 | 1.28k | MI->x86opsize = 8; |
238 | 1.28k | break; |
239 | 0 | case X86_MOVPQI2QImr: |
240 | 0 | SStream_concat0(O, "xmmword ptr "); |
241 | 0 | MI->x86opsize = 16; |
242 | 0 | break; |
243 | 1.28k | } |
244 | 1.76k | } else { |
245 | 1.76k | SStream_concat0(O, "qword ptr "); |
246 | 1.76k | MI->x86opsize = 8; |
247 | 1.76k | } |
248 | | |
249 | 3.05k | printMemReference(MI, OpNo, O); |
250 | 3.05k | } |
251 | | |
252 | | static void printf80mem(MCInst *MI, unsigned OpNo, SStream *O) |
253 | 811 | { |
254 | 811 | switch (MCInst_getOpcode(MI)) { |
255 | 334 | default: |
256 | 334 | SStream_concat0(O, "xword ptr "); |
257 | 334 | break; |
258 | 288 | case X86_FBLDm: |
259 | 477 | case X86_FBSTPm: |
260 | 477 | break; |
261 | 811 | } |
262 | | |
263 | 811 | MI->x86opsize = 10; |
264 | 811 | printMemReference(MI, OpNo, O); |
265 | 811 | } |
266 | | |
267 | | static void printf128mem(MCInst *MI, unsigned OpNo, SStream *O) |
268 | 4.93k | { |
269 | 4.93k | SStream_concat0(O, "xmmword ptr "); |
270 | 4.93k | MI->x86opsize = 16; |
271 | 4.93k | printMemReference(MI, OpNo, O); |
272 | 4.93k | } |
273 | | |
274 | | static void printf256mem(MCInst *MI, unsigned OpNo, SStream *O) |
275 | 4.46k | { |
276 | 4.46k | SStream_concat0(O, "ymmword ptr "); |
277 | 4.46k | MI->x86opsize = 32; |
278 | 4.46k | printMemReference(MI, OpNo, O); |
279 | 4.46k | } |
280 | | |
281 | | static void printf512mem(MCInst *MI, unsigned OpNo, SStream *O) |
282 | 3.10k | { |
283 | 3.10k | SStream_concat0(O, "zmmword ptr "); |
284 | 3.10k | MI->x86opsize = 64; |
285 | 3.10k | printMemReference(MI, OpNo, O); |
286 | 3.10k | } |
287 | | #endif |
288 | | |
289 | | static const char *getRegisterName(unsigned RegNo); |
290 | | static void printRegName(SStream *OS, unsigned RegNo) |
291 | 921k | { |
292 | 921k | SStream_concat0(OS, getRegisterName(RegNo)); |
293 | 921k | } |
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 | 244k | { |
311 | 244k | if (positive) { |
312 | | // always print this number in positive form |
313 | 209k | 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, |
341 | 0 | "0%" PRIx64 "h", |
342 | 0 | imm); |
343 | 0 | else |
344 | 0 | SStream_concat( |
345 | 0 | O, "%" PRIx64 "h", imm); |
346 | 0 | } else |
347 | 0 | SStream_concat(O, "%" PRIu64, imm); |
348 | 0 | } |
349 | 209k | } else { // Intel syntax |
350 | 209k | if (imm < 0) { |
351 | 3.53k | if (MI->op1_size) { |
352 | 1.22k | switch (MI->op1_size) { |
353 | 1.22k | default: |
354 | 1.22k | break; |
355 | 1.22k | case 1: |
356 | 0 | imm &= 0xff; |
357 | 0 | break; |
358 | 0 | case 2: |
359 | 0 | imm &= 0xffff; |
360 | 0 | break; |
361 | 0 | case 4: |
362 | 0 | imm &= 0xffffffff; |
363 | 0 | break; |
364 | 1.22k | } |
365 | 1.22k | } |
366 | | |
367 | 3.53k | SStream_concat(O, "0x%" PRIx64, imm); |
368 | 206k | } else { |
369 | 206k | if (imm > HEX_THRESHOLD) |
370 | 193k | SStream_concat(O, "0x%" PRIx64, imm); |
371 | 12.7k | else |
372 | 12.7k | SStream_concat(O, "%" PRIu64, imm); |
373 | 206k | } |
374 | 209k | } |
375 | 209k | } else { |
376 | 34.7k | if (MI->csh->syntax == CS_OPT_SYNTAX_MASM) { |
377 | 0 | if (imm < 0) { |
378 | 0 | if (imm == 0x8000000000000000LL) // imm == -imm |
379 | 0 | SStream_concat0(O, "8000000000000000h"); |
380 | 0 | else if (imm < -HEX_THRESHOLD) { |
381 | 0 | if (need_zero_prefix(imm)) |
382 | 0 | SStream_concat(O, |
383 | 0 | "-0%" PRIx64 "h", |
384 | 0 | -imm); |
385 | 0 | else |
386 | 0 | SStream_concat(O, |
387 | 0 | "-%" PRIx64 "h", |
388 | 0 | -imm); |
389 | 0 | } else |
390 | 0 | SStream_concat(O, "-%" PRIu64, -imm); |
391 | 0 | } else { |
392 | 0 | if (imm > HEX_THRESHOLD) { |
393 | 0 | if (need_zero_prefix(imm)) |
394 | 0 | SStream_concat(O, |
395 | 0 | "0%" PRIx64 "h", |
396 | 0 | imm); |
397 | 0 | else |
398 | 0 | SStream_concat( |
399 | 0 | O, "%" PRIx64 "h", imm); |
400 | 0 | } else |
401 | 0 | SStream_concat(O, "%" PRIu64, imm); |
402 | 0 | } |
403 | 34.7k | } else { // Intel syntax |
404 | 34.7k | if (imm < 0) { |
405 | 4.14k | if (imm == 0x8000000000000000LL) // imm == -imm |
406 | 0 | SStream_concat0(O, |
407 | 0 | "0x8000000000000000"); |
408 | 4.14k | else if (imm < -HEX_THRESHOLD) |
409 | 3.62k | SStream_concat(O, "-0x%" PRIx64, -imm); |
410 | 513 | else |
411 | 513 | SStream_concat(O, "-%" PRIu64, -imm); |
412 | | |
413 | 30.5k | } else { |
414 | 30.5k | if (imm > HEX_THRESHOLD) |
415 | 25.3k | SStream_concat(O, "0x%" PRIx64, imm); |
416 | 5.23k | else |
417 | 5.23k | SStream_concat(O, "%" PRIu64, imm); |
418 | 30.5k | } |
419 | 34.7k | } |
420 | 34.7k | } |
421 | 244k | } |
422 | | |
423 | | // local printOperand, without updating public operands |
424 | | static void _printOperand(MCInst *MI, unsigned OpNo, SStream *O) |
425 | 334k | { |
426 | 334k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
427 | 334k | if (MCOperand_isReg(Op)) { |
428 | 334k | printRegName(O, MCOperand_getReg(Op)); |
429 | 334k | } else if (MCOperand_isImm(Op)) { |
430 | 0 | int64_t imm = MCOperand_getImm(Op); |
431 | 0 | printImm(MI, O, imm, MI->csh->imm_unsigned); |
432 | 0 | } |
433 | 334k | } |
434 | | |
435 | | #ifndef CAPSTONE_DIET |
436 | | // copy & normalize access info |
437 | | static void get_op_access(cs_struct *h, unsigned int id, uint8_t *access, |
438 | | uint64_t *eflags) |
439 | 1.66M | { |
440 | 1.66M | #ifndef CAPSTONE_DIET |
441 | 1.66M | uint8_t i; |
442 | 1.66M | const uint8_t *arr = X86_get_op_access(h, id, eflags); |
443 | | |
444 | | // initialize access |
445 | 1.66M | memset(access, 0, CS_X86_MAXIMUM_OPERAND_SIZE * sizeof(access[0])); |
446 | | |
447 | 1.66M | if (!arr) { |
448 | 0 | access[0] = 0; |
449 | 0 | return; |
450 | 0 | } |
451 | | |
452 | | // copy to access but zero out CS_AC_IGNORE |
453 | 4.81M | for (i = 0; arr[i]; i++) { |
454 | 3.15M | if (arr[i] != CS_AC_IGNORE) |
455 | 2.65M | access[i] = arr[i]; |
456 | 494k | else |
457 | 494k | access[i] = 0; |
458 | 3.15M | } |
459 | | |
460 | | // mark the end of array |
461 | 1.66M | access[i] = 0; |
462 | 1.66M | #endif |
463 | 1.66M | } |
464 | | #endif |
465 | | |
466 | | static void printSrcIdx(MCInst *MI, unsigned Op, SStream *O) |
467 | 33.9k | { |
468 | 33.9k | MCOperand *SegReg; |
469 | 33.9k | int reg; |
470 | | |
471 | 33.9k | if (MI->csh->detail_opt) { |
472 | 33.9k | #ifndef CAPSTONE_DIET |
473 | 33.9k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
474 | 33.9k | #endif |
475 | | |
476 | 33.9k | MI->flat_insn->detail->x86 |
477 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
478 | 33.9k | .type = X86_OP_MEM; |
479 | 33.9k | MI->flat_insn->detail->x86 |
480 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
481 | 33.9k | .size = MI->x86opsize; |
482 | 33.9k | MI->flat_insn->detail->x86 |
483 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
484 | 33.9k | .mem.segment = X86_REG_INVALID; |
485 | 33.9k | MI->flat_insn->detail->x86 |
486 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
487 | 33.9k | .mem.base = X86_REG_INVALID; |
488 | 33.9k | MI->flat_insn->detail->x86 |
489 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
490 | 33.9k | .mem.index = X86_REG_INVALID; |
491 | 33.9k | MI->flat_insn->detail->x86 |
492 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
493 | 33.9k | .mem.scale = 1; |
494 | 33.9k | MI->flat_insn->detail->x86 |
495 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
496 | 33.9k | .mem.disp = 0; |
497 | | |
498 | 33.9k | #ifndef CAPSTONE_DIET |
499 | 33.9k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
500 | 33.9k | &MI->flat_insn->detail->x86.eflags); |
501 | 33.9k | MI->flat_insn->detail->x86 |
502 | 33.9k | .operands[MI->flat_insn->detail->x86.op_count] |
503 | 33.9k | .access = access[MI->flat_insn->detail->x86.op_count]; |
504 | 33.9k | #endif |
505 | 33.9k | } |
506 | | |
507 | 33.9k | SegReg = MCInst_getOperand(MI, Op + 1); |
508 | 33.9k | reg = MCOperand_getReg(SegReg); |
509 | | |
510 | | // If this has a segment register, print it. |
511 | 33.9k | if (reg) { |
512 | 1.18k | _printOperand(MI, Op + 1, O); |
513 | 1.18k | if (MI->csh->detail_opt) { |
514 | 1.18k | MI->flat_insn->detail->x86 |
515 | 1.18k | .operands[MI->flat_insn->detail->x86.op_count] |
516 | 1.18k | .mem.segment = X86_register_map(reg); |
517 | 1.18k | } |
518 | 1.18k | SStream_concat0(O, ":"); |
519 | 1.18k | } |
520 | | |
521 | 33.9k | SStream_concat0(O, "["); |
522 | 33.9k | set_mem_access(MI, true); |
523 | 33.9k | printOperand(MI, Op, O); |
524 | 33.9k | SStream_concat0(O, "]"); |
525 | 33.9k | set_mem_access(MI, false); |
526 | 33.9k | } |
527 | | |
528 | | static void printDstIdx(MCInst *MI, unsigned Op, SStream *O) |
529 | 41.1k | { |
530 | 41.1k | if (MI->csh->detail_opt) { |
531 | 41.1k | #ifndef CAPSTONE_DIET |
532 | 41.1k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
533 | 41.1k | #endif |
534 | | |
535 | 41.1k | MI->flat_insn->detail->x86 |
536 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
537 | 41.1k | .type = X86_OP_MEM; |
538 | 41.1k | MI->flat_insn->detail->x86 |
539 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
540 | 41.1k | .size = MI->x86opsize; |
541 | 41.1k | MI->flat_insn->detail->x86 |
542 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
543 | 41.1k | .mem.segment = X86_REG_INVALID; |
544 | 41.1k | MI->flat_insn->detail->x86 |
545 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
546 | 41.1k | .mem.base = X86_REG_INVALID; |
547 | 41.1k | MI->flat_insn->detail->x86 |
548 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
549 | 41.1k | .mem.index = X86_REG_INVALID; |
550 | 41.1k | MI->flat_insn->detail->x86 |
551 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
552 | 41.1k | .mem.scale = 1; |
553 | 41.1k | MI->flat_insn->detail->x86 |
554 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
555 | 41.1k | .mem.disp = 0; |
556 | | |
557 | 41.1k | #ifndef CAPSTONE_DIET |
558 | 41.1k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
559 | 41.1k | &MI->flat_insn->detail->x86.eflags); |
560 | 41.1k | MI->flat_insn->detail->x86 |
561 | 41.1k | .operands[MI->flat_insn->detail->x86.op_count] |
562 | 41.1k | .access = access[MI->flat_insn->detail->x86.op_count]; |
563 | 41.1k | #endif |
564 | 41.1k | } |
565 | | |
566 | | // DI accesses are always ES-based on non-64bit mode |
567 | 41.1k | if (MI->csh->mode != CS_MODE_64) { |
568 | 27.9k | SStream_concat0(O, "es:["); |
569 | 27.9k | if (MI->csh->detail_opt) { |
570 | 27.9k | MI->flat_insn->detail->x86 |
571 | 27.9k | .operands[MI->flat_insn->detail->x86.op_count] |
572 | 27.9k | .mem.segment = X86_REG_ES; |
573 | 27.9k | } |
574 | 27.9k | } else |
575 | 13.2k | SStream_concat0(O, "["); |
576 | | |
577 | 41.1k | set_mem_access(MI, true); |
578 | 41.1k | printOperand(MI, Op, O); |
579 | 41.1k | SStream_concat0(O, "]"); |
580 | 41.1k | set_mem_access(MI, false); |
581 | 41.1k | } |
582 | | |
583 | | static void printSrcIdx8(MCInst *MI, unsigned OpNo, SStream *O) |
584 | 9.91k | { |
585 | 9.91k | SStream_concat0(O, "byte ptr "); |
586 | 9.91k | MI->x86opsize = 1; |
587 | 9.91k | printSrcIdx(MI, OpNo, O); |
588 | 9.91k | } |
589 | | |
590 | | static void printSrcIdx16(MCInst *MI, unsigned OpNo, SStream *O) |
591 | 9.88k | { |
592 | 9.88k | SStream_concat0(O, "word ptr "); |
593 | 9.88k | MI->x86opsize = 2; |
594 | 9.88k | printSrcIdx(MI, OpNo, O); |
595 | 9.88k | } |
596 | | |
597 | | static void printSrcIdx32(MCInst *MI, unsigned OpNo, SStream *O) |
598 | 11.5k | { |
599 | 11.5k | SStream_concat0(O, "dword ptr "); |
600 | 11.5k | MI->x86opsize = 4; |
601 | 11.5k | printSrcIdx(MI, OpNo, O); |
602 | 11.5k | } |
603 | | |
604 | | static void printSrcIdx64(MCInst *MI, unsigned OpNo, SStream *O) |
605 | 2.65k | { |
606 | 2.65k | SStream_concat0(O, "qword ptr "); |
607 | 2.65k | MI->x86opsize = 8; |
608 | 2.65k | printSrcIdx(MI, OpNo, O); |
609 | 2.65k | } |
610 | | |
611 | | static void printDstIdx8(MCInst *MI, unsigned OpNo, SStream *O) |
612 | 13.1k | { |
613 | 13.1k | SStream_concat0(O, "byte ptr "); |
614 | 13.1k | MI->x86opsize = 1; |
615 | 13.1k | printDstIdx(MI, OpNo, O); |
616 | 13.1k | } |
617 | | |
618 | | static void printDstIdx16(MCInst *MI, unsigned OpNo, SStream *O) |
619 | 9.69k | { |
620 | 9.69k | SStream_concat0(O, "word ptr "); |
621 | 9.69k | MI->x86opsize = 2; |
622 | 9.69k | printDstIdx(MI, OpNo, O); |
623 | 9.69k | } |
624 | | |
625 | | static void printDstIdx32(MCInst *MI, unsigned OpNo, SStream *O) |
626 | 15.6k | { |
627 | 15.6k | SStream_concat0(O, "dword ptr "); |
628 | 15.6k | MI->x86opsize = 4; |
629 | 15.6k | printDstIdx(MI, OpNo, O); |
630 | 15.6k | } |
631 | | |
632 | | static void printDstIdx64(MCInst *MI, unsigned OpNo, SStream *O) |
633 | 2.72k | { |
634 | 2.72k | SStream_concat0(O, "qword ptr "); |
635 | 2.72k | MI->x86opsize = 8; |
636 | 2.72k | printDstIdx(MI, OpNo, O); |
637 | 2.72k | } |
638 | | |
639 | | static void printMemOffset(MCInst *MI, unsigned Op, SStream *O) |
640 | 7.34k | { |
641 | 7.34k | MCOperand *DispSpec = MCInst_getOperand(MI, Op); |
642 | 7.34k | MCOperand *SegReg = MCInst_getOperand(MI, Op + 1); |
643 | 7.34k | int reg; |
644 | | |
645 | 7.34k | if (MI->csh->detail_opt) { |
646 | 7.34k | #ifndef CAPSTONE_DIET |
647 | 7.34k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
648 | 7.34k | #endif |
649 | | |
650 | 7.34k | MI->flat_insn->detail->x86 |
651 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
652 | 7.34k | .type = X86_OP_MEM; |
653 | 7.34k | MI->flat_insn->detail->x86 |
654 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
655 | 7.34k | .size = MI->x86opsize; |
656 | 7.34k | MI->flat_insn->detail->x86 |
657 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
658 | 7.34k | .mem.segment = X86_REG_INVALID; |
659 | 7.34k | MI->flat_insn->detail->x86 |
660 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
661 | 7.34k | .mem.base = X86_REG_INVALID; |
662 | 7.34k | MI->flat_insn->detail->x86 |
663 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
664 | 7.34k | .mem.index = X86_REG_INVALID; |
665 | 7.34k | MI->flat_insn->detail->x86 |
666 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
667 | 7.34k | .mem.scale = 1; |
668 | 7.34k | MI->flat_insn->detail->x86 |
669 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
670 | 7.34k | .mem.disp = 0; |
671 | | |
672 | 7.34k | #ifndef CAPSTONE_DIET |
673 | 7.34k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
674 | 7.34k | &MI->flat_insn->detail->x86.eflags); |
675 | 7.34k | MI->flat_insn->detail->x86 |
676 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
677 | 7.34k | .access = access[MI->flat_insn->detail->x86.op_count]; |
678 | 7.34k | #endif |
679 | 7.34k | } |
680 | | |
681 | | // If this has a segment register, print it. |
682 | 7.34k | reg = MCOperand_getReg(SegReg); |
683 | 7.34k | if (reg) { |
684 | 484 | _printOperand(MI, Op + 1, O); |
685 | 484 | SStream_concat0(O, ":"); |
686 | 484 | if (MI->csh->detail_opt) { |
687 | 484 | MI->flat_insn->detail->x86 |
688 | 484 | .operands[MI->flat_insn->detail->x86.op_count] |
689 | 484 | .mem.segment = X86_register_map(reg); |
690 | 484 | } |
691 | 484 | } |
692 | | |
693 | 7.34k | SStream_concat0(O, "["); |
694 | | |
695 | 7.34k | if (MCOperand_isImm(DispSpec)) { |
696 | 7.34k | int64_t imm = MCOperand_getImm(DispSpec); |
697 | 7.34k | if (MI->csh->detail_opt) |
698 | 7.34k | MI->flat_insn->detail->x86 |
699 | 7.34k | .operands[MI->flat_insn->detail->x86.op_count] |
700 | 7.34k | .mem.disp = imm; |
701 | | |
702 | 7.34k | if (imm < 0) |
703 | 1.47k | printImm(MI, O, arch_masks[MI->csh->mode] & imm, true); |
704 | 5.87k | else |
705 | 5.87k | printImm(MI, O, imm, true); |
706 | 7.34k | } |
707 | | |
708 | 7.34k | SStream_concat0(O, "]"); |
709 | | |
710 | 7.34k | if (MI->csh->detail_opt) |
711 | 7.34k | MI->flat_insn->detail->x86.op_count++; |
712 | | |
713 | 7.34k | if (MI->op1_size == 0) |
714 | 7.34k | MI->op1_size = MI->x86opsize; |
715 | 7.34k | } |
716 | | |
717 | | static void printU8Imm(MCInst *MI, unsigned Op, SStream *O) |
718 | 41.5k | { |
719 | 41.5k | uint8_t val = MCOperand_getImm(MCInst_getOperand(MI, Op)) & 0xff; |
720 | | |
721 | 41.5k | printImm(MI, O, val, true); |
722 | | |
723 | 41.5k | if (MI->csh->detail_opt) { |
724 | 41.5k | #ifndef CAPSTONE_DIET |
725 | 41.5k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
726 | 41.5k | #endif |
727 | | |
728 | 41.5k | MI->flat_insn->detail->x86 |
729 | 41.5k | .operands[MI->flat_insn->detail->x86.op_count] |
730 | 41.5k | .type = X86_OP_IMM; |
731 | 41.5k | MI->flat_insn->detail->x86 |
732 | 41.5k | .operands[MI->flat_insn->detail->x86.op_count] |
733 | 41.5k | .imm = val; |
734 | 41.5k | MI->flat_insn->detail->x86 |
735 | 41.5k | .operands[MI->flat_insn->detail->x86.op_count] |
736 | 41.5k | .size = 1; |
737 | | |
738 | 41.5k | #ifndef CAPSTONE_DIET |
739 | 41.5k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
740 | 41.5k | &MI->flat_insn->detail->x86.eflags); |
741 | 41.5k | MI->flat_insn->detail->x86 |
742 | 41.5k | .operands[MI->flat_insn->detail->x86.op_count] |
743 | 41.5k | .access = access[MI->flat_insn->detail->x86.op_count]; |
744 | 41.5k | #endif |
745 | | |
746 | 41.5k | MI->flat_insn->detail->x86.op_count++; |
747 | 41.5k | } |
748 | 41.5k | } |
749 | | |
750 | | static void printMemOffs8(MCInst *MI, unsigned OpNo, SStream *O) |
751 | 4.03k | { |
752 | 4.03k | SStream_concat0(O, "byte ptr "); |
753 | 4.03k | MI->x86opsize = 1; |
754 | 4.03k | printMemOffset(MI, OpNo, O); |
755 | 4.03k | } |
756 | | |
757 | | static void printMemOffs16(MCInst *MI, unsigned OpNo, SStream *O) |
758 | 897 | { |
759 | 897 | SStream_concat0(O, "word ptr "); |
760 | 897 | MI->x86opsize = 2; |
761 | 897 | printMemOffset(MI, OpNo, O); |
762 | 897 | } |
763 | | |
764 | | static void printMemOffs32(MCInst *MI, unsigned OpNo, SStream *O) |
765 | 1.99k | { |
766 | 1.99k | SStream_concat0(O, "dword ptr "); |
767 | 1.99k | MI->x86opsize = 4; |
768 | 1.99k | printMemOffset(MI, OpNo, O); |
769 | 1.99k | } |
770 | | |
771 | | static void printMemOffs64(MCInst *MI, unsigned OpNo, SStream *O) |
772 | 418 | { |
773 | 418 | SStream_concat0(O, "qword ptr "); |
774 | 418 | MI->x86opsize = 8; |
775 | 418 | printMemOffset(MI, OpNo, O); |
776 | 418 | } |
777 | | |
778 | | static void printInstruction(MCInst *MI, SStream *O); |
779 | | |
780 | | void X86_Intel_printInst(MCInst *MI, SStream *O, void *Info) |
781 | 641k | { |
782 | 641k | x86_reg reg, reg2; |
783 | 641k | enum cs_ac_type access1, access2; |
784 | | |
785 | | // printf("opcode = %u\n", MCInst_getOpcode(MI)); |
786 | | |
787 | | // perhaps this instruction does not need printer |
788 | 641k | if (MI->assembly[0]) { |
789 | 0 | strncpy(O->buffer, MI->assembly, sizeof(O->buffer)); |
790 | 0 | return; |
791 | 0 | } |
792 | | |
793 | 641k | X86_lockrep(MI, O); |
794 | 641k | printInstruction(MI, O); |
795 | | |
796 | 641k | reg = X86_insn_reg_intel(MCInst_getOpcode(MI), &access1); |
797 | 641k | if (MI->csh->detail_opt) { |
798 | 641k | #ifndef CAPSTONE_DIET |
799 | 641k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE] = { 0 }; |
800 | 641k | #endif |
801 | | |
802 | | // first op can be embedded in the asm by llvm. |
803 | | // so we have to add the missing register as the first operand |
804 | 641k | if (reg) { |
805 | | // shift all the ops right to leave 1st slot for this new register op |
806 | 64.0k | memmove(&(MI->flat_insn->detail->x86.operands[1]), |
807 | 64.0k | &(MI->flat_insn->detail->x86.operands[0]), |
808 | 64.0k | sizeof(MI->flat_insn->detail->x86.operands[0]) * |
809 | 64.0k | (ARR_SIZE(MI->flat_insn->detail->x86 |
810 | 64.0k | .operands) - |
811 | 64.0k | 1)); |
812 | 64.0k | MI->flat_insn->detail->x86.operands[0].type = |
813 | 64.0k | X86_OP_REG; |
814 | 64.0k | MI->flat_insn->detail->x86.operands[0].reg = reg; |
815 | 64.0k | MI->flat_insn->detail->x86.operands[0].size = |
816 | 64.0k | MI->csh->regsize_map[reg]; |
817 | 64.0k | MI->flat_insn->detail->x86.operands[0].access = access1; |
818 | 64.0k | MI->flat_insn->detail->x86.op_count++; |
819 | 576k | } else { |
820 | 576k | if (X86_insn_reg_intel2(MCInst_getOpcode(MI), ®, |
821 | 576k | &access1, ®2, &access2)) { |
822 | 9.21k | MI->flat_insn->detail->x86.operands[0].type = |
823 | 9.21k | X86_OP_REG; |
824 | 9.21k | MI->flat_insn->detail->x86.operands[0].reg = |
825 | 9.21k | reg; |
826 | 9.21k | MI->flat_insn->detail->x86.operands[0].size = |
827 | 9.21k | MI->csh->regsize_map[reg]; |
828 | 9.21k | MI->flat_insn->detail->x86.operands[0].access = |
829 | 9.21k | access1; |
830 | 9.21k | MI->flat_insn->detail->x86.operands[1].type = |
831 | 9.21k | X86_OP_REG; |
832 | 9.21k | MI->flat_insn->detail->x86.operands[1].reg = |
833 | 9.21k | reg2; |
834 | 9.21k | MI->flat_insn->detail->x86.operands[1].size = |
835 | 9.21k | MI->csh->regsize_map[reg2]; |
836 | 9.21k | MI->flat_insn->detail->x86.operands[1].access = |
837 | 9.21k | access2; |
838 | 9.21k | MI->flat_insn->detail->x86.op_count = 2; |
839 | 9.21k | } |
840 | 576k | } |
841 | | |
842 | 641k | #ifndef CAPSTONE_DIET |
843 | 641k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
844 | 641k | &MI->flat_insn->detail->x86.eflags); |
845 | 641k | MI->flat_insn->detail->x86.operands[0].access = access[0]; |
846 | 641k | MI->flat_insn->detail->x86.operands[1].access = access[1]; |
847 | 641k | #endif |
848 | 641k | } |
849 | | |
850 | 641k | if (MI->op1_size == 0 && reg) |
851 | 43.5k | MI->op1_size = MI->csh->regsize_map[reg]; |
852 | 641k | } |
853 | | |
854 | | /// printPCRelImm - This is used to print an immediate value that ends up |
855 | | /// being encoded as a pc-relative value. |
856 | | static void printPCRelImm(MCInst *MI, unsigned OpNo, SStream *O) |
857 | 42.6k | { |
858 | 42.6k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
859 | 42.6k | if (MCOperand_isImm(Op)) { |
860 | 42.6k | int64_t imm = MCOperand_getImm(Op) + MI->flat_insn->size + |
861 | 42.6k | MI->address; |
862 | 42.6k | uint8_t opsize = X86_immediate_size(MI->Opcode, NULL); |
863 | | |
864 | | // truncate imm for non-64bit |
865 | 42.6k | if (MI->csh->mode != CS_MODE_64) { |
866 | 28.5k | imm = imm & 0xffffffff; |
867 | 28.5k | } |
868 | | |
869 | 42.6k | printImm(MI, O, imm, true); |
870 | | |
871 | 42.6k | if (MI->csh->detail_opt) { |
872 | 42.6k | #ifndef CAPSTONE_DIET |
873 | 42.6k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
874 | 42.6k | #endif |
875 | | |
876 | 42.6k | MI->flat_insn->detail->x86 |
877 | 42.6k | .operands[MI->flat_insn->detail->x86.op_count] |
878 | 42.6k | .type = X86_OP_IMM; |
879 | | // if op_count > 0, then this operand's size is taken from the destination op |
880 | 42.6k | if (MI->flat_insn->detail->x86.op_count > 0) |
881 | 0 | MI->flat_insn->detail->x86 |
882 | 0 | .operands[MI->flat_insn->detail->x86 |
883 | 0 | .op_count] |
884 | 0 | .size = |
885 | 0 | MI->flat_insn->detail->x86.operands[0] |
886 | 0 | .size; |
887 | 42.6k | else if (opsize > 0) |
888 | 1.30k | MI->flat_insn->detail->x86 |
889 | 1.30k | .operands[MI->flat_insn->detail->x86 |
890 | 1.30k | .op_count] |
891 | 1.30k | .size = opsize; |
892 | 41.3k | else |
893 | 41.3k | MI->flat_insn->detail->x86 |
894 | 41.3k | .operands[MI->flat_insn->detail->x86 |
895 | 41.3k | .op_count] |
896 | 41.3k | .size = MI->imm_size; |
897 | 42.6k | MI->flat_insn->detail->x86 |
898 | 42.6k | .operands[MI->flat_insn->detail->x86.op_count] |
899 | 42.6k | .imm = imm; |
900 | | |
901 | 42.6k | #ifndef CAPSTONE_DIET |
902 | 42.6k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
903 | 42.6k | &MI->flat_insn->detail->x86.eflags); |
904 | 42.6k | MI->flat_insn->detail->x86 |
905 | 42.6k | .operands[MI->flat_insn->detail->x86.op_count] |
906 | 42.6k | .access = |
907 | 42.6k | access[MI->flat_insn->detail->x86.op_count]; |
908 | 42.6k | #endif |
909 | | |
910 | 42.6k | MI->flat_insn->detail->x86.op_count++; |
911 | 42.6k | } |
912 | | |
913 | 42.6k | if (MI->op1_size == 0) |
914 | 42.6k | MI->op1_size = MI->imm_size; |
915 | 42.6k | } |
916 | 42.6k | } |
917 | | |
918 | | static void printOperand(MCInst *MI, unsigned OpNo, SStream *O) |
919 | 663k | { |
920 | 663k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
921 | | |
922 | 663k | if (MCOperand_isReg(Op)) { |
923 | 587k | unsigned int reg = MCOperand_getReg(Op); |
924 | | |
925 | 587k | printRegName(O, reg); |
926 | 587k | if (MI->csh->detail_opt) { |
927 | 587k | if (MI->csh->doing_mem) { |
928 | 75.1k | MI->flat_insn->detail->x86 |
929 | 75.1k | .operands[MI->flat_insn->detail->x86 |
930 | 75.1k | .op_count] |
931 | 75.1k | .mem.base = X86_register_map(reg); |
932 | 512k | } else { |
933 | 512k | #ifndef CAPSTONE_DIET |
934 | 512k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
935 | 512k | #endif |
936 | | |
937 | 512k | MI->flat_insn->detail->x86 |
938 | 512k | .operands[MI->flat_insn->detail->x86 |
939 | 512k | .op_count] |
940 | 512k | .type = X86_OP_REG; |
941 | 512k | MI->flat_insn->detail->x86 |
942 | 512k | .operands[MI->flat_insn->detail->x86 |
943 | 512k | .op_count] |
944 | 512k | .reg = X86_register_map(reg); |
945 | 512k | MI->flat_insn->detail->x86 |
946 | 512k | .operands[MI->flat_insn->detail->x86 |
947 | 512k | .op_count] |
948 | 512k | .size = |
949 | 512k | MI->csh->regsize_map[X86_register_map( |
950 | 512k | reg)]; |
951 | | |
952 | 512k | #ifndef CAPSTONE_DIET |
953 | 512k | get_op_access( |
954 | 512k | MI->csh, MCInst_getOpcode(MI), access, |
955 | 512k | &MI->flat_insn->detail->x86.eflags); |
956 | 512k | MI->flat_insn->detail->x86 |
957 | 512k | .operands[MI->flat_insn->detail->x86 |
958 | 512k | .op_count] |
959 | 512k | .access = |
960 | 512k | access[MI->flat_insn->detail->x86 |
961 | 512k | .op_count]; |
962 | 512k | #endif |
963 | | |
964 | 512k | MI->flat_insn->detail->x86.op_count++; |
965 | 512k | } |
966 | 587k | } |
967 | | |
968 | 587k | if (MI->op1_size == 0) |
969 | 299k | MI->op1_size = |
970 | 299k | MI->csh->regsize_map[X86_register_map(reg)]; |
971 | 587k | } else if (MCOperand_isImm(Op)) { |
972 | 76.8k | uint8_t encsize; |
973 | 76.8k | int64_t imm = MCOperand_getImm(Op); |
974 | 76.8k | uint8_t opsize = |
975 | 76.8k | X86_immediate_size(MCInst_getOpcode(MI), &encsize); |
976 | | |
977 | 76.8k | if (opsize == 1) // print 1 byte immediate in positive form |
978 | 33.2k | imm = imm & 0xff; |
979 | | |
980 | | // printf(">>> id = %u\n", MI->flat_insn->id); |
981 | 76.8k | switch (MI->flat_insn->id) { |
982 | 34.7k | default: |
983 | 34.7k | printImm(MI, O, imm, MI->csh->imm_unsigned); |
984 | 34.7k | break; |
985 | | |
986 | 425 | case X86_INS_MOVABS: |
987 | 12.2k | case X86_INS_MOV: |
988 | | // do not print number in negative form |
989 | 12.2k | printImm(MI, O, imm, true); |
990 | 12.2k | break; |
991 | | |
992 | 0 | case X86_INS_IN: |
993 | 0 | case X86_INS_OUT: |
994 | 0 | case X86_INS_INT: |
995 | | // do not print number in negative form |
996 | 0 | imm = imm & 0xff; |
997 | 0 | printImm(MI, O, imm, true); |
998 | 0 | break; |
999 | | |
1000 | 1.28k | case X86_INS_LCALL: |
1001 | 2.84k | case X86_INS_LJMP: |
1002 | 2.84k | case X86_INS_JMP: |
1003 | | // always print address in positive form |
1004 | 2.84k | if (OpNo == 1) { // ptr16 part |
1005 | 1.42k | imm = imm & 0xffff; |
1006 | 1.42k | opsize = 2; |
1007 | 1.42k | } else |
1008 | 1.42k | opsize = 4; |
1009 | 2.84k | printImm(MI, O, imm, true); |
1010 | 2.84k | break; |
1011 | | |
1012 | 7.33k | case X86_INS_AND: |
1013 | 13.1k | case X86_INS_OR: |
1014 | 18.7k | case X86_INS_XOR: |
1015 | | // do not print number in negative form |
1016 | 18.7k | if (imm >= 0 && imm <= HEX_THRESHOLD) |
1017 | 2.01k | printImm(MI, O, imm, true); |
1018 | 16.6k | else { |
1019 | 16.6k | imm = arch_masks[opsize ? opsize : MI->imm_size] & |
1020 | 16.6k | imm; |
1021 | 16.6k | printImm(MI, O, imm, true); |
1022 | 16.6k | } |
1023 | 18.7k | break; |
1024 | | |
1025 | 6.80k | case X86_INS_RET: |
1026 | 8.33k | case X86_INS_RETF: |
1027 | | // RET imm16 |
1028 | 8.33k | if (imm >= 0 && imm <= HEX_THRESHOLD) |
1029 | 740 | printImm(MI, O, imm, true); |
1030 | 7.59k | else { |
1031 | 7.59k | imm = 0xffff & imm; |
1032 | 7.59k | printImm(MI, O, imm, true); |
1033 | 7.59k | } |
1034 | 8.33k | break; |
1035 | 76.8k | } |
1036 | | |
1037 | 76.8k | if (MI->csh->detail_opt) { |
1038 | 76.8k | if (MI->csh->doing_mem) { |
1039 | 0 | MI->flat_insn->detail->x86 |
1040 | 0 | .operands[MI->flat_insn->detail->x86 |
1041 | 0 | .op_count] |
1042 | 0 | .mem.disp = imm; |
1043 | 76.8k | } else { |
1044 | 76.8k | #ifndef CAPSTONE_DIET |
1045 | 76.8k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
1046 | 76.8k | #endif |
1047 | | |
1048 | 76.8k | MI->flat_insn->detail->x86 |
1049 | 76.8k | .operands[MI->flat_insn->detail->x86 |
1050 | 76.8k | .op_count] |
1051 | 76.8k | .type = X86_OP_IMM; |
1052 | 76.8k | if (opsize > 0) { |
1053 | 64.6k | MI->flat_insn->detail->x86 |
1054 | 64.6k | .operands[MI->flat_insn->detail |
1055 | 64.6k | ->x86.op_count] |
1056 | 64.6k | .size = opsize; |
1057 | 64.6k | MI->flat_insn->detail->x86.encoding |
1058 | 64.6k | .imm_size = encsize; |
1059 | 64.6k | } else if (MI->flat_insn->detail->x86.op_count > |
1060 | 12.1k | 0) { |
1061 | 2.52k | if (MI->flat_insn->id != |
1062 | 2.52k | X86_INS_LCALL && |
1063 | 2.52k | MI->flat_insn->id != X86_INS_LJMP) { |
1064 | 2.52k | MI->flat_insn->detail->x86 |
1065 | 2.52k | .operands[MI->flat_insn |
1066 | 2.52k | ->detail |
1067 | 2.52k | ->x86 |
1068 | 2.52k | .op_count] |
1069 | 2.52k | .size = |
1070 | 2.52k | MI->flat_insn->detail |
1071 | 2.52k | ->x86 |
1072 | 2.52k | .operands[0] |
1073 | 2.52k | .size; |
1074 | 2.52k | } else |
1075 | 0 | MI->flat_insn->detail->x86 |
1076 | 0 | .operands[MI->flat_insn |
1077 | 0 | ->detail |
1078 | 0 | ->x86 |
1079 | 0 | .op_count] |
1080 | 0 | .size = MI->imm_size; |
1081 | 2.52k | } else |
1082 | 9.62k | MI->flat_insn->detail->x86 |
1083 | 9.62k | .operands[MI->flat_insn->detail |
1084 | 9.62k | ->x86.op_count] |
1085 | 9.62k | .size = MI->imm_size; |
1086 | 76.8k | MI->flat_insn->detail->x86 |
1087 | 76.8k | .operands[MI->flat_insn->detail->x86 |
1088 | 76.8k | .op_count] |
1089 | 76.8k | .imm = imm; |
1090 | | |
1091 | 76.8k | #ifndef CAPSTONE_DIET |
1092 | 76.8k | get_op_access( |
1093 | 76.8k | MI->csh, MCInst_getOpcode(MI), access, |
1094 | 76.8k | &MI->flat_insn->detail->x86.eflags); |
1095 | 76.8k | MI->flat_insn->detail->x86 |
1096 | 76.8k | .operands[MI->flat_insn->detail->x86 |
1097 | 76.8k | .op_count] |
1098 | 76.8k | .access = |
1099 | 76.8k | access[MI->flat_insn->detail->x86 |
1100 | 76.8k | .op_count]; |
1101 | 76.8k | #endif |
1102 | | |
1103 | 76.8k | MI->flat_insn->detail->x86.op_count++; |
1104 | 76.8k | } |
1105 | 76.8k | } |
1106 | 76.8k | } |
1107 | 663k | } |
1108 | | |
1109 | | static void printMemReference(MCInst *MI, unsigned Op, SStream *O) |
1110 | 265k | { |
1111 | 265k | bool NeedPlus = false; |
1112 | 265k | MCOperand *BaseReg = MCInst_getOperand(MI, Op + X86_AddrBaseReg); |
1113 | 265k | uint64_t ScaleVal = |
1114 | 265k | MCOperand_getImm(MCInst_getOperand(MI, Op + X86_AddrScaleAmt)); |
1115 | 265k | MCOperand *IndexReg = MCInst_getOperand(MI, Op + X86_AddrIndexReg); |
1116 | 265k | MCOperand *DispSpec = MCInst_getOperand(MI, Op + X86_AddrDisp); |
1117 | 265k | MCOperand *SegReg = MCInst_getOperand(MI, Op + X86_AddrSegmentReg); |
1118 | 265k | int reg; |
1119 | | |
1120 | 265k | if (MI->csh->detail_opt) { |
1121 | 265k | #ifndef CAPSTONE_DIET |
1122 | 265k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
1123 | 265k | #endif |
1124 | | |
1125 | 265k | MI->flat_insn->detail->x86 |
1126 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1127 | 265k | .type = X86_OP_MEM; |
1128 | 265k | MI->flat_insn->detail->x86 |
1129 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1130 | 265k | .size = MI->x86opsize; |
1131 | 265k | MI->flat_insn->detail->x86 |
1132 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1133 | 265k | .mem.segment = X86_REG_INVALID; |
1134 | 265k | MI->flat_insn->detail->x86 |
1135 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1136 | 265k | .mem.base = X86_register_map(MCOperand_getReg(BaseReg)); |
1137 | 265k | if (MCOperand_getReg(IndexReg) != X86_EIZ) { |
1138 | 264k | MI->flat_insn->detail->x86 |
1139 | 264k | .operands[MI->flat_insn->detail->x86.op_count] |
1140 | 264k | .mem.index = |
1141 | 264k | X86_register_map(MCOperand_getReg(IndexReg)); |
1142 | 264k | } |
1143 | 265k | MI->flat_insn->detail->x86 |
1144 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1145 | 265k | .mem.scale = (int)ScaleVal; |
1146 | 265k | MI->flat_insn->detail->x86 |
1147 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1148 | 265k | .mem.disp = 0; |
1149 | | |
1150 | 265k | #ifndef CAPSTONE_DIET |
1151 | 265k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, |
1152 | 265k | &MI->flat_insn->detail->x86.eflags); |
1153 | 265k | MI->flat_insn->detail->x86 |
1154 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1155 | 265k | .access = access[MI->flat_insn->detail->x86.op_count]; |
1156 | 265k | #endif |
1157 | 265k | } |
1158 | | |
1159 | | // If this has a segment register, print it. |
1160 | 265k | reg = MCOperand_getReg(SegReg); |
1161 | 265k | if (reg) { |
1162 | 6.83k | _printOperand(MI, Op + X86_AddrSegmentReg, O); |
1163 | 6.83k | if (MI->csh->detail_opt) { |
1164 | 6.83k | MI->flat_insn->detail->x86 |
1165 | 6.83k | .operands[MI->flat_insn->detail->x86.op_count] |
1166 | 6.83k | .mem.segment = X86_register_map(reg); |
1167 | 6.83k | } |
1168 | 6.83k | SStream_concat0(O, ":"); |
1169 | 6.83k | } |
1170 | | |
1171 | 265k | SStream_concat0(O, "["); |
1172 | | |
1173 | 265k | if (MCOperand_getReg(BaseReg)) { |
1174 | 260k | _printOperand(MI, Op + X86_AddrBaseReg, O); |
1175 | 260k | NeedPlus = true; |
1176 | 260k | } |
1177 | | |
1178 | 265k | if (MCOperand_getReg(IndexReg) && |
1179 | 265k | MCOperand_getReg(IndexReg) != X86_EIZ) { |
1180 | 65.5k | if (NeedPlus) |
1181 | 64.7k | SStream_concat0(O, " + "); |
1182 | 65.5k | _printOperand(MI, Op + X86_AddrIndexReg, O); |
1183 | 65.5k | if (ScaleVal != 1) |
1184 | 9.86k | SStream_concat(O, "*%u", ScaleVal); |
1185 | 65.5k | NeedPlus = true; |
1186 | 65.5k | } |
1187 | | |
1188 | 265k | if (MCOperand_isImm(DispSpec)) { |
1189 | 265k | int64_t DispVal = MCOperand_getImm(DispSpec); |
1190 | 265k | if (MI->csh->detail_opt) |
1191 | 265k | MI->flat_insn->detail->x86 |
1192 | 265k | .operands[MI->flat_insn->detail->x86.op_count] |
1193 | 265k | .mem.disp = DispVal; |
1194 | 265k | if (DispVal) { |
1195 | 75.9k | if (NeedPlus) { |
1196 | 71.7k | if (DispVal < 0) { |
1197 | 29.7k | SStream_concat0(O, " - "); |
1198 | 29.7k | printImm(MI, O, -DispVal, true); |
1199 | 42.0k | } else { |
1200 | 42.0k | SStream_concat0(O, " + "); |
1201 | 42.0k | printImm(MI, O, DispVal, true); |
1202 | 42.0k | } |
1203 | 71.7k | } else { |
1204 | | // memory reference to an immediate address |
1205 | 4.20k | if (MI->csh->mode == CS_MODE_64) |
1206 | 494 | MI->op1_size = 8; |
1207 | 4.20k | if (DispVal < 0) { |
1208 | 1.39k | printImm(MI, O, |
1209 | 1.39k | arch_masks[MI->csh->mode] & |
1210 | 1.39k | DispVal, |
1211 | 1.39k | true); |
1212 | 2.81k | } else { |
1213 | 2.81k | printImm(MI, O, DispVal, true); |
1214 | 2.81k | } |
1215 | 4.20k | } |
1216 | | |
1217 | 189k | } else { |
1218 | | // DispVal = 0 |
1219 | 189k | if (!NeedPlus) // [0] |
1220 | 447 | SStream_concat0(O, "0"); |
1221 | 189k | } |
1222 | 265k | } |
1223 | | |
1224 | 265k | SStream_concat0(O, "]"); |
1225 | | |
1226 | 265k | if (MI->csh->detail_opt) |
1227 | 265k | MI->flat_insn->detail->x86.op_count++; |
1228 | | |
1229 | 265k | if (MI->op1_size == 0) |
1230 | 171k | MI->op1_size = MI->x86opsize; |
1231 | 265k | } |
1232 | | |
1233 | | static void printanymem(MCInst *MI, unsigned OpNo, SStream *O) |
1234 | 7.74k | { |
1235 | 7.74k | switch (MI->Opcode) { |
1236 | 494 | default: |
1237 | 494 | break; |
1238 | 766 | case X86_LEA16r: |
1239 | 766 | MI->x86opsize = 2; |
1240 | 766 | break; |
1241 | 922 | case X86_LEA32r: |
1242 | 1.75k | case X86_LEA64_32r: |
1243 | 1.75k | MI->x86opsize = 4; |
1244 | 1.75k | break; |
1245 | 482 | case X86_LEA64r: |
1246 | 482 | MI->x86opsize = 8; |
1247 | 482 | break; |
1248 | 0 | #ifndef CAPSTONE_X86_REDUCE |
1249 | 503 | case X86_BNDCL32rm: |
1250 | 899 | case X86_BNDCN32rm: |
1251 | 1.30k | case X86_BNDCU32rm: |
1252 | 2.03k | case X86_BNDSTXmr: |
1253 | 2.79k | case X86_BNDLDXrm: |
1254 | 3.19k | case X86_BNDCL64rm: |
1255 | 3.67k | case X86_BNDCN64rm: |
1256 | 4.24k | case X86_BNDCU64rm: |
1257 | 4.24k | MI->x86opsize = 16; |
1258 | 4.24k | break; |
1259 | 7.74k | #endif |
1260 | 7.74k | } |
1261 | | |
1262 | 7.74k | printMemReference(MI, OpNo, O); |
1263 | 7.74k | } |
1264 | | |
1265 | | #ifdef CAPSTONE_X86_REDUCE |
1266 | | #include "X86GenAsmWriter1_reduce.inc" |
1267 | | #else |
1268 | | #include "X86GenAsmWriter1.inc" |
1269 | | #endif |
1270 | | |
1271 | | #include "X86GenRegisterName1.inc" |
1272 | | |
1273 | | #endif |