/src/capstonenext/arch/X86/X86ATTInstPrinter.c
Line | Count | Source (jump to first uncovered line) |
1 | | //===-- X86ATTInstPrinter.cpp - AT&T 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 AT&T-style |
11 | | // assembly. |
12 | | // |
13 | | //===----------------------------------------------------------------------===// |
14 | | |
15 | | /* Capstone Disassembly Engine */ |
16 | | /* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2019 */ |
17 | | |
18 | | // this code is only relevant when DIET mode is disable |
19 | | #if defined(CAPSTONE_HAS_X86) && !defined(CAPSTONE_DIET) && !defined(CAPSTONE_X86_ATT_DISABLE) |
20 | | |
21 | | #ifdef _MSC_VER |
22 | | #pragma warning(disable:4996) // disable MSVC's warning on strncpy() |
23 | | #pragma warning(disable:28719) // disable MSVC's warning on strncpy() |
24 | | #endif |
25 | | |
26 | | #if !defined(CAPSTONE_HAS_OSXKERNEL) |
27 | | #include <ctype.h> |
28 | | #endif |
29 | | #include <capstone/platform.h> |
30 | | |
31 | | #if defined(CAPSTONE_HAS_OSXKERNEL) |
32 | | #include <Availability.h> |
33 | | #include <libkern/libkern.h> |
34 | | #else |
35 | | #include <stdio.h> |
36 | | #include <stdlib.h> |
37 | | #endif |
38 | | |
39 | | #include <string.h> |
40 | | |
41 | | #include "../../utils.h" |
42 | | #include "../../MCInst.h" |
43 | | #include "../../SStream.h" |
44 | | #include "../../MCRegisterInfo.h" |
45 | | #include "X86Mapping.h" |
46 | | #include "X86BaseInfo.h" |
47 | | #include "X86InstPrinterCommon.h" |
48 | | |
49 | | #define GET_INSTRINFO_ENUM |
50 | | #ifdef CAPSTONE_X86_REDUCE |
51 | | #include "X86GenInstrInfo_reduce.inc" |
52 | | #else |
53 | | #include "X86GenInstrInfo.inc" |
54 | | #endif |
55 | | |
56 | | #define GET_REGINFO_ENUM |
57 | | #include "X86GenRegisterInfo.inc" |
58 | | |
59 | | static void printMemReference(MCInst *MI, unsigned Op, SStream *O); |
60 | | static void printOperand(MCInst *MI, unsigned OpNo, SStream *O); |
61 | | |
62 | | |
63 | | static void set_mem_access(MCInst *MI, bool status) |
64 | 281k | { |
65 | 281k | if (MI->csh->detail_opt != CS_OPT_ON) |
66 | 0 | return; |
67 | | |
68 | 281k | MI->csh->doing_mem = status; |
69 | 281k | if (!status) |
70 | | // done, create the next operand slot |
71 | 140k | MI->flat_insn->detail->x86.op_count++; |
72 | 281k | } |
73 | | |
74 | | static void printopaquemem(MCInst *MI, unsigned OpNo, SStream *O) |
75 | 17.1k | { |
76 | 17.1k | switch(MI->csh->mode) { |
77 | 6.26k | case CS_MODE_16: |
78 | 6.26k | switch(MI->flat_insn->id) { |
79 | 2.48k | default: |
80 | 2.48k | MI->x86opsize = 2; |
81 | 2.48k | break; |
82 | 983 | case X86_INS_LJMP: |
83 | 1.62k | case X86_INS_LCALL: |
84 | 1.62k | MI->x86opsize = 4; |
85 | 1.62k | break; |
86 | 565 | case X86_INS_SGDT: |
87 | 1.01k | case X86_INS_SIDT: |
88 | 1.59k | case X86_INS_LGDT: |
89 | 2.15k | case X86_INS_LIDT: |
90 | 2.15k | MI->x86opsize = 6; |
91 | 2.15k | break; |
92 | 6.26k | } |
93 | 6.26k | break; |
94 | 6.26k | case CS_MODE_32: |
95 | 6.03k | switch(MI->flat_insn->id) { |
96 | 1.94k | default: |
97 | 1.94k | MI->x86opsize = 4; |
98 | 1.94k | break; |
99 | 187 | case X86_INS_LJMP: |
100 | 977 | case X86_INS_JMP: |
101 | 1.51k | case X86_INS_LCALL: |
102 | 2.05k | case X86_INS_SGDT: |
103 | 2.71k | case X86_INS_SIDT: |
104 | 3.28k | case X86_INS_LGDT: |
105 | 4.09k | case X86_INS_LIDT: |
106 | 4.09k | MI->x86opsize = 6; |
107 | 4.09k | break; |
108 | 6.03k | } |
109 | 6.03k | break; |
110 | 6.03k | case CS_MODE_64: |
111 | 4.84k | switch(MI->flat_insn->id) { |
112 | 895 | default: |
113 | 895 | MI->x86opsize = 8; |
114 | 895 | break; |
115 | 1.40k | case X86_INS_LJMP: |
116 | 2.04k | case X86_INS_LCALL: |
117 | 2.46k | case X86_INS_SGDT: |
118 | 2.90k | case X86_INS_SIDT: |
119 | 3.26k | case X86_INS_LGDT: |
120 | 3.95k | case X86_INS_LIDT: |
121 | 3.95k | MI->x86opsize = 10; |
122 | 3.95k | break; |
123 | 4.84k | } |
124 | 4.84k | break; |
125 | 4.84k | default: // never reach |
126 | 0 | break; |
127 | 17.1k | } |
128 | | |
129 | 17.1k | printMemReference(MI, OpNo, O); |
130 | 17.1k | } |
131 | | |
132 | | static void printi8mem(MCInst *MI, unsigned OpNo, SStream *O) |
133 | 268k | { |
134 | 268k | MI->x86opsize = 1; |
135 | 268k | printMemReference(MI, OpNo, O); |
136 | 268k | } |
137 | | |
138 | | static void printi16mem(MCInst *MI, unsigned OpNo, SStream *O) |
139 | 78.3k | { |
140 | 78.3k | MI->x86opsize = 2; |
141 | | |
142 | 78.3k | printMemReference(MI, OpNo, O); |
143 | 78.3k | } |
144 | | |
145 | | static void printi32mem(MCInst *MI, unsigned OpNo, SStream *O) |
146 | 86.9k | { |
147 | 86.9k | MI->x86opsize = 4; |
148 | | |
149 | 86.9k | printMemReference(MI, OpNo, O); |
150 | 86.9k | } |
151 | | |
152 | | static void printi64mem(MCInst *MI, unsigned OpNo, SStream *O) |
153 | 30.7k | { |
154 | 30.7k | MI->x86opsize = 8; |
155 | 30.7k | printMemReference(MI, OpNo, O); |
156 | 30.7k | } |
157 | | |
158 | | static void printi128mem(MCInst *MI, unsigned OpNo, SStream *O) |
159 | 9.04k | { |
160 | 9.04k | MI->x86opsize = 16; |
161 | 9.04k | printMemReference(MI, OpNo, O); |
162 | 9.04k | } |
163 | | |
164 | | static void printi512mem(MCInst *MI, unsigned OpNo, SStream *O) |
165 | 4.40k | { |
166 | 4.40k | MI->x86opsize = 64; |
167 | 4.40k | printMemReference(MI, OpNo, O); |
168 | 4.40k | } |
169 | | |
170 | | #ifndef CAPSTONE_X86_REDUCE |
171 | | static void printi256mem(MCInst *MI, unsigned OpNo, SStream *O) |
172 | 5.54k | { |
173 | 5.54k | MI->x86opsize = 32; |
174 | 5.54k | printMemReference(MI, OpNo, O); |
175 | 5.54k | } |
176 | | |
177 | | static void printf32mem(MCInst *MI, unsigned OpNo, SStream *O) |
178 | 11.1k | { |
179 | 11.1k | switch(MCInst_getOpcode(MI)) { |
180 | 7.08k | default: |
181 | 7.08k | MI->x86opsize = 4; |
182 | 7.08k | break; |
183 | 2.77k | case X86_FSTENVm: |
184 | 4.05k | case X86_FLDENVm: |
185 | | // TODO: fix this in tablegen instead |
186 | 4.05k | switch(MI->csh->mode) { |
187 | 0 | default: // never reach |
188 | 0 | break; |
189 | 2.42k | case CS_MODE_16: |
190 | 2.42k | MI->x86opsize = 14; |
191 | 2.42k | break; |
192 | 678 | case CS_MODE_32: |
193 | 1.63k | case CS_MODE_64: |
194 | 1.63k | MI->x86opsize = 28; |
195 | 1.63k | break; |
196 | 4.05k | } |
197 | 4.05k | break; |
198 | 11.1k | } |
199 | | |
200 | 11.1k | printMemReference(MI, OpNo, O); |
201 | 11.1k | } |
202 | | |
203 | | static void printf64mem(MCInst *MI, unsigned OpNo, SStream *O) |
204 | 9.64k | { |
205 | 9.64k | MI->x86opsize = 8; |
206 | 9.64k | printMemReference(MI, OpNo, O); |
207 | 9.64k | } |
208 | | |
209 | | static void printf80mem(MCInst *MI, unsigned OpNo, SStream *O) |
210 | 1.08k | { |
211 | 1.08k | MI->x86opsize = 10; |
212 | 1.08k | printMemReference(MI, OpNo, O); |
213 | 1.08k | } |
214 | | |
215 | | static void printf128mem(MCInst *MI, unsigned OpNo, SStream *O) |
216 | 7.58k | { |
217 | 7.58k | MI->x86opsize = 16; |
218 | 7.58k | printMemReference(MI, OpNo, O); |
219 | 7.58k | } |
220 | | |
221 | | static void printf256mem(MCInst *MI, unsigned OpNo, SStream *O) |
222 | 4.30k | { |
223 | 4.30k | MI->x86opsize = 32; |
224 | 4.30k | printMemReference(MI, OpNo, O); |
225 | 4.30k | } |
226 | | |
227 | | static void printf512mem(MCInst *MI, unsigned OpNo, SStream *O) |
228 | 4.08k | { |
229 | 4.08k | MI->x86opsize = 64; |
230 | 4.08k | printMemReference(MI, OpNo, O); |
231 | 4.08k | } |
232 | | |
233 | | #endif |
234 | | |
235 | | static void printRegName(SStream *OS, unsigned RegNo); |
236 | | |
237 | | // local printOperand, without updating public operands |
238 | | static void _printOperand(MCInst *MI, unsigned OpNo, SStream *O) |
239 | 751k | { |
240 | 751k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
241 | 751k | if (MCOperand_isReg(Op)) { |
242 | 751k | printRegName(O, MCOperand_getReg(Op)); |
243 | 751k | } else if (MCOperand_isImm(Op)) { |
244 | 0 | uint8_t encsize; |
245 | 0 | uint8_t opsize = X86_immediate_size(MCInst_getOpcode(MI), &encsize); |
246 | | |
247 | | // Print X86 immediates as signed values. |
248 | 0 | int64_t imm = MCOperand_getImm(Op); |
249 | 0 | if (imm < 0) { |
250 | 0 | if (MI->csh->imm_unsigned) { |
251 | 0 | if (opsize) { |
252 | 0 | switch(opsize) { |
253 | 0 | default: |
254 | 0 | break; |
255 | 0 | case 1: |
256 | 0 | imm &= 0xff; |
257 | 0 | break; |
258 | 0 | case 2: |
259 | 0 | imm &= 0xffff; |
260 | 0 | break; |
261 | 0 | case 4: |
262 | 0 | imm &= 0xffffffff; |
263 | 0 | break; |
264 | 0 | } |
265 | 0 | } |
266 | | |
267 | 0 | SStream_concat(O, "$0x%"PRIx64, imm); |
268 | 0 | } else { |
269 | 0 | if (imm < -HEX_THRESHOLD) |
270 | 0 | SStream_concat(O, "$-0x%"PRIx64, -imm); |
271 | 0 | else |
272 | 0 | SStream_concat(O, "$-%"PRIu64, -imm); |
273 | 0 | } |
274 | 0 | } else { |
275 | 0 | if (imm > HEX_THRESHOLD) |
276 | 0 | SStream_concat(O, "$0x%"PRIx64, imm); |
277 | 0 | else |
278 | 0 | SStream_concat(O, "$%"PRIu64, imm); |
279 | 0 | } |
280 | 0 | } |
281 | 751k | } |
282 | | |
283 | | // convert Intel access info to AT&T access info |
284 | | static void get_op_access(cs_struct *h, unsigned int id, uint8_t *access, uint64_t *eflags) |
285 | 1.17M | { |
286 | 1.17M | uint8_t count, i; |
287 | 1.17M | const uint8_t *arr = X86_get_op_access(h, id, eflags); |
288 | | |
289 | | // initialize access |
290 | 1.17M | memset(access, 0, CS_X86_MAXIMUM_OPERAND_SIZE * sizeof(access[0])); |
291 | 1.17M | if (!arr) { |
292 | 0 | return; |
293 | 0 | } |
294 | | |
295 | | // find the non-zero last entry |
296 | 3.35M | for(count = 0; arr[count]; count++); |
297 | | |
298 | 1.17M | if (count == 0) |
299 | 66.9k | return; |
300 | | |
301 | | // copy in reverse order this access array from Intel syntax -> AT&T syntax |
302 | 1.11M | count--; |
303 | 3.29M | for(i = 0; i <= count && ((count - i) < CS_X86_MAXIMUM_OPERAND_SIZE) && i < CS_X86_MAXIMUM_OPERAND_SIZE; i++) { |
304 | 2.18M | if (arr[count - i] != CS_AC_IGNORE) |
305 | 1.88M | access[i] = arr[count - i]; |
306 | 296k | else |
307 | 296k | access[i] = 0; |
308 | 2.18M | } |
309 | 1.11M | } |
310 | | |
311 | | static void printSrcIdx(MCInst *MI, unsigned Op, SStream *O) |
312 | 66.8k | { |
313 | 66.8k | MCOperand *SegReg; |
314 | 66.8k | int reg; |
315 | | |
316 | 66.8k | if (MI->csh->detail_opt) { |
317 | 66.8k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
318 | | |
319 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
320 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
321 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
322 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_REG_INVALID; |
323 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_REG_INVALID; |
324 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
325 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
326 | | |
327 | 66.8k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
328 | 66.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
329 | 66.8k | } |
330 | | |
331 | 66.8k | SegReg = MCInst_getOperand(MI, Op+1); |
332 | 66.8k | reg = MCOperand_getReg(SegReg); |
333 | | // If this has a segment register, print it. |
334 | 66.8k | if (reg) { |
335 | 1.97k | _printOperand(MI, Op + 1, O); |
336 | 1.97k | SStream_concat0(O, ":"); |
337 | | |
338 | 1.97k | if (MI->csh->detail_opt) { |
339 | 1.97k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_register_map(reg); |
340 | 1.97k | } |
341 | 1.97k | } |
342 | | |
343 | 66.8k | SStream_concat0(O, "("); |
344 | 66.8k | set_mem_access(MI, true); |
345 | | |
346 | 66.8k | printOperand(MI, Op, O); |
347 | | |
348 | 66.8k | SStream_concat0(O, ")"); |
349 | 66.8k | set_mem_access(MI, false); |
350 | 66.8k | } |
351 | | |
352 | | static void printDstIdx(MCInst *MI, unsigned Op, SStream *O) |
353 | 73.9k | { |
354 | 73.9k | if (MI->csh->detail_opt) { |
355 | 73.9k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
356 | | |
357 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
358 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
359 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
360 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_REG_INVALID; |
361 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_REG_INVALID; |
362 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
363 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
364 | | |
365 | 73.9k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
366 | 73.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
367 | 73.9k | } |
368 | | |
369 | | // DI accesses are always ES-based on non-64bit mode |
370 | 73.9k | if (MI->csh->mode != CS_MODE_64) { |
371 | 43.3k | SStream_concat0(O, "%es:("); |
372 | 43.3k | if (MI->csh->detail_opt) { |
373 | 43.3k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_ES; |
374 | 43.3k | } |
375 | 43.3k | } else |
376 | 30.5k | SStream_concat0(O, "("); |
377 | | |
378 | 73.9k | set_mem_access(MI, true); |
379 | | |
380 | 73.9k | printOperand(MI, Op, O); |
381 | | |
382 | 73.9k | SStream_concat0(O, ")"); |
383 | 73.9k | set_mem_access(MI, false); |
384 | 73.9k | } |
385 | | |
386 | | static void printSrcIdx8(MCInst *MI, unsigned OpNo, SStream *O) |
387 | 26.3k | { |
388 | 26.3k | MI->x86opsize = 1; |
389 | 26.3k | printSrcIdx(MI, OpNo, O); |
390 | 26.3k | } |
391 | | |
392 | | static void printSrcIdx16(MCInst *MI, unsigned OpNo, SStream *O) |
393 | 19.2k | { |
394 | 19.2k | MI->x86opsize = 2; |
395 | 19.2k | printSrcIdx(MI, OpNo, O); |
396 | 19.2k | } |
397 | | |
398 | | static void printSrcIdx32(MCInst *MI, unsigned OpNo, SStream *O) |
399 | 17.5k | { |
400 | 17.5k | MI->x86opsize = 4; |
401 | 17.5k | printSrcIdx(MI, OpNo, O); |
402 | 17.5k | } |
403 | | |
404 | | static void printSrcIdx64(MCInst *MI, unsigned OpNo, SStream *O) |
405 | 3.80k | { |
406 | 3.80k | MI->x86opsize = 8; |
407 | 3.80k | printSrcIdx(MI, OpNo, O); |
408 | 3.80k | } |
409 | | |
410 | | static void printDstIdx8(MCInst *MI, unsigned OpNo, SStream *O) |
411 | 30.7k | { |
412 | 30.7k | MI->x86opsize = 1; |
413 | 30.7k | printDstIdx(MI, OpNo, O); |
414 | 30.7k | } |
415 | | |
416 | | static void printDstIdx16(MCInst *MI, unsigned OpNo, SStream *O) |
417 | 19.4k | { |
418 | 19.4k | MI->x86opsize = 2; |
419 | 19.4k | printDstIdx(MI, OpNo, O); |
420 | 19.4k | } |
421 | | |
422 | | static void printDstIdx32(MCInst *MI, unsigned OpNo, SStream *O) |
423 | 18.2k | { |
424 | 18.2k | MI->x86opsize = 4; |
425 | 18.2k | printDstIdx(MI, OpNo, O); |
426 | 18.2k | } |
427 | | |
428 | | static void printDstIdx64(MCInst *MI, unsigned OpNo, SStream *O) |
429 | 5.52k | { |
430 | 5.52k | MI->x86opsize = 8; |
431 | 5.52k | printDstIdx(MI, OpNo, O); |
432 | 5.52k | } |
433 | | |
434 | | static void printMemOffset(MCInst *MI, unsigned Op, SStream *O) |
435 | 14.1k | { |
436 | 14.1k | MCOperand *DispSpec = MCInst_getOperand(MI, Op); |
437 | 14.1k | MCOperand *SegReg = MCInst_getOperand(MI, Op+1); |
438 | 14.1k | int reg; |
439 | | |
440 | 14.1k | if (MI->csh->detail_opt) { |
441 | 14.1k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
442 | | |
443 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
444 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
445 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
446 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_REG_INVALID; |
447 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_REG_INVALID; |
448 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
449 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
450 | | |
451 | 14.1k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
452 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
453 | 14.1k | } |
454 | | |
455 | | // If this has a segment register, print it. |
456 | 14.1k | reg = MCOperand_getReg(SegReg); |
457 | 14.1k | if (reg) { |
458 | 918 | _printOperand(MI, Op + 1, O); |
459 | 918 | SStream_concat0(O, ":"); |
460 | | |
461 | 918 | if (MI->csh->detail_opt) { |
462 | 918 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_register_map(reg); |
463 | 918 | } |
464 | 918 | } |
465 | | |
466 | 14.1k | if (MCOperand_isImm(DispSpec)) { |
467 | 14.1k | int64_t imm = MCOperand_getImm(DispSpec); |
468 | 14.1k | if (MI->csh->detail_opt) |
469 | 14.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = imm; |
470 | 14.1k | if (imm < 0) { |
471 | 2.39k | SStream_concat(O, "0x%"PRIx64, arch_masks[MI->csh->mode] & imm); |
472 | 11.7k | } else { |
473 | 11.7k | if (imm > HEX_THRESHOLD) |
474 | 11.2k | SStream_concat(O, "0x%"PRIx64, imm); |
475 | 461 | else |
476 | 461 | SStream_concat(O, "%"PRIu64, imm); |
477 | 11.7k | } |
478 | 14.1k | } |
479 | | |
480 | 14.1k | if (MI->csh->detail_opt) |
481 | 14.1k | MI->flat_insn->detail->x86.op_count++; |
482 | 14.1k | } |
483 | | |
484 | | static void printU8Imm(MCInst *MI, unsigned Op, SStream *O) |
485 | 62.1k | { |
486 | 62.1k | uint8_t val = MCOperand_getImm(MCInst_getOperand(MI, Op)) & 0xff; |
487 | | |
488 | 62.1k | if (val > HEX_THRESHOLD) |
489 | 54.1k | SStream_concat(O, "$0x%x", val); |
490 | 7.94k | else |
491 | 7.94k | SStream_concat(O, "$%u", val); |
492 | | |
493 | 62.1k | if (MI->csh->detail_opt) { |
494 | 62.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM; |
495 | 62.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = val; |
496 | 62.1k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = 1; |
497 | 62.1k | MI->flat_insn->detail->x86.op_count++; |
498 | 62.1k | } |
499 | 62.1k | } |
500 | | |
501 | | static void printMemOffs8(MCInst *MI, unsigned OpNo, SStream *O) |
502 | 7.55k | { |
503 | 7.55k | MI->x86opsize = 1; |
504 | 7.55k | printMemOffset(MI, OpNo, O); |
505 | 7.55k | } |
506 | | |
507 | | static void printMemOffs16(MCInst *MI, unsigned OpNo, SStream *O) |
508 | 2.76k | { |
509 | 2.76k | MI->x86opsize = 2; |
510 | 2.76k | printMemOffset(MI, OpNo, O); |
511 | 2.76k | } |
512 | | |
513 | | static void printMemOffs32(MCInst *MI, unsigned OpNo, SStream *O) |
514 | 3.28k | { |
515 | 3.28k | MI->x86opsize = 4; |
516 | 3.28k | printMemOffset(MI, OpNo, O); |
517 | 3.28k | } |
518 | | |
519 | | static void printMemOffs64(MCInst *MI, unsigned OpNo, SStream *O) |
520 | 533 | { |
521 | 533 | MI->x86opsize = 8; |
522 | 533 | printMemOffset(MI, OpNo, O); |
523 | 533 | } |
524 | | |
525 | | /// printPCRelImm - This is used to print an immediate value that ends up |
526 | | /// being encoded as a pc-relative value (e.g. for jumps and calls). These |
527 | | /// print slightly differently than normal immediates. For example, a $ is not |
528 | | /// emitted. |
529 | | static void printPCRelImm(MCInst *MI, unsigned OpNo, SStream *O) |
530 | 96.0k | { |
531 | 96.0k | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
532 | 96.0k | if (MCOperand_isImm(Op)) { |
533 | 96.0k | int64_t imm = MCOperand_getImm(Op) + MI->flat_insn->size + MI->address; |
534 | | |
535 | | // truncate imm for non-64bit |
536 | 96.0k | if (MI->csh->mode != CS_MODE_64) { |
537 | 64.5k | imm = imm & 0xffffffff; |
538 | 64.5k | } |
539 | | |
540 | 96.0k | if (imm < 0) { |
541 | 1.81k | SStream_concat(O, "0x%"PRIx64, imm); |
542 | 94.1k | } else { |
543 | 94.1k | if (imm > HEX_THRESHOLD) |
544 | 94.1k | SStream_concat(O, "0x%"PRIx64, imm); |
545 | 21 | else |
546 | 21 | SStream_concat(O, "%"PRIu64, imm); |
547 | 94.1k | } |
548 | 96.0k | if (MI->csh->detail_opt) { |
549 | 96.0k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM; |
550 | 96.0k | MI->has_imm = true; |
551 | 96.0k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = imm; |
552 | 96.0k | MI->flat_insn->detail->x86.op_count++; |
553 | 96.0k | } |
554 | 96.0k | } |
555 | 96.0k | } |
556 | | |
557 | | static void printOperand(MCInst *MI, unsigned OpNo, SStream *O) |
558 | 1.28M | { |
559 | 1.28M | MCOperand *Op = MCInst_getOperand(MI, OpNo); |
560 | 1.28M | if (MCOperand_isReg(Op)) { |
561 | 1.11M | unsigned int reg = MCOperand_getReg(Op); |
562 | 1.11M | printRegName(O, reg); |
563 | 1.11M | if (MI->csh->detail_opt) { |
564 | 1.11M | if (MI->csh->doing_mem) { |
565 | 140k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_register_map(reg); |
566 | 973k | } else { |
567 | 973k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
568 | | |
569 | 973k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_REG; |
570 | 973k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].reg = X86_register_map(reg); |
571 | 973k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->csh->regsize_map[X86_register_map(reg)]; |
572 | | |
573 | 973k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
574 | 973k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
575 | | |
576 | 973k | MI->flat_insn->detail->x86.op_count++; |
577 | 973k | } |
578 | 1.11M | } |
579 | 1.11M | } else if (MCOperand_isImm(Op)) { |
580 | | // Print X86 immediates as signed values. |
581 | 173k | uint8_t encsize; |
582 | 173k | int64_t imm = MCOperand_getImm(Op); |
583 | 173k | uint8_t opsize = X86_immediate_size(MCInst_getOpcode(MI), &encsize); |
584 | | |
585 | 173k | if (opsize == 1) // print 1 byte immediate in positive form |
586 | 75.5k | imm = imm & 0xff; |
587 | | |
588 | 173k | switch(MI->flat_insn->id) { |
589 | 78.6k | default: |
590 | 78.6k | if (imm >= 0) { |
591 | 70.9k | if (imm > HEX_THRESHOLD) |
592 | 62.0k | SStream_concat(O, "$0x%"PRIx64, imm); |
593 | 8.89k | else |
594 | 8.89k | SStream_concat(O, "$%"PRIu64, imm); |
595 | 70.9k | } else { |
596 | 7.64k | if (MI->csh->imm_unsigned) { |
597 | 0 | if (opsize) { |
598 | 0 | switch(opsize) { |
599 | 0 | default: |
600 | 0 | break; |
601 | 0 | case 1: |
602 | 0 | imm &= 0xff; |
603 | 0 | break; |
604 | 0 | case 2: |
605 | 0 | imm &= 0xffff; |
606 | 0 | break; |
607 | 0 | case 4: |
608 | 0 | imm &= 0xffffffff; |
609 | 0 | break; |
610 | 0 | } |
611 | 0 | } |
612 | | |
613 | 0 | SStream_concat(O, "$0x%"PRIx64, imm); |
614 | 7.64k | } else { |
615 | 7.64k | if (imm == 0x8000000000000000LL) // imm == -imm |
616 | 0 | SStream_concat0(O, "$0x8000000000000000"); |
617 | 7.64k | else if (imm < -HEX_THRESHOLD) |
618 | 6.64k | SStream_concat(O, "$-0x%"PRIx64, -imm); |
619 | 1.00k | else |
620 | 1.00k | SStream_concat(O, "$-%"PRIu64, -imm); |
621 | 7.64k | } |
622 | 7.64k | } |
623 | 78.6k | break; |
624 | | |
625 | 78.6k | case X86_INS_MOVABS: |
626 | 32.7k | case X86_INS_MOV: |
627 | | // do not print number in negative form |
628 | 32.7k | if (imm > HEX_THRESHOLD) |
629 | 30.4k | SStream_concat(O, "$0x%"PRIx64, imm); |
630 | 2.25k | else |
631 | 2.25k | SStream_concat(O, "$%"PRIu64, imm); |
632 | 32.7k | break; |
633 | | |
634 | 0 | case X86_INS_IN: |
635 | 0 | case X86_INS_OUT: |
636 | 0 | case X86_INS_INT: |
637 | | // do not print number in negative form |
638 | 0 | imm = imm & 0xff; |
639 | 0 | if (imm >= 0 && imm <= HEX_THRESHOLD) |
640 | 0 | SStream_concat(O, "$%u", imm); |
641 | 0 | else { |
642 | 0 | SStream_concat(O, "$0x%x", imm); |
643 | 0 | } |
644 | 0 | break; |
645 | | |
646 | 3.42k | case X86_INS_LCALL: |
647 | 5.67k | case X86_INS_LJMP: |
648 | 5.67k | case X86_INS_JMP: |
649 | | // always print address in positive form |
650 | 5.67k | if (OpNo == 1) { // selector is ptr16 |
651 | 2.83k | imm = imm & 0xffff; |
652 | 2.83k | opsize = 2; |
653 | 2.83k | } else |
654 | 2.83k | opsize = 4; |
655 | 5.67k | SStream_concat(O, "$0x%"PRIx64, imm); |
656 | 5.67k | break; |
657 | | |
658 | 14.9k | case X86_INS_AND: |
659 | 30.9k | case X86_INS_OR: |
660 | 41.2k | case X86_INS_XOR: |
661 | | // do not print number in negative form |
662 | 41.2k | if (imm >= 0 && imm <= HEX_THRESHOLD) |
663 | 3.78k | SStream_concat(O, "$%u", imm); |
664 | 37.4k | else { |
665 | 37.4k | imm = arch_masks[opsize? opsize : MI->imm_size] & imm; |
666 | 37.4k | SStream_concat(O, "$0x%"PRIx64, imm); |
667 | 37.4k | } |
668 | 41.2k | break; |
669 | | |
670 | 13.5k | case X86_INS_RET: |
671 | 15.6k | case X86_INS_RETF: |
672 | | // RET imm16 |
673 | 15.6k | if (imm >= 0 && imm <= HEX_THRESHOLD) |
674 | 934 | SStream_concat(O, "$%u", imm); |
675 | 14.7k | else { |
676 | 14.7k | imm = 0xffff & imm; |
677 | 14.7k | SStream_concat(O, "$0x%x", imm); |
678 | 14.7k | } |
679 | 15.6k | break; |
680 | 173k | } |
681 | | |
682 | 173k | if (MI->csh->detail_opt) { |
683 | 173k | if (MI->csh->doing_mem) { |
684 | 0 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
685 | 0 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = imm; |
686 | 173k | } else { |
687 | 173k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM; |
688 | 173k | MI->has_imm = true; |
689 | 173k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = imm; |
690 | | |
691 | 173k | if (opsize > 0) { |
692 | 147k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = opsize; |
693 | 147k | MI->flat_insn->detail->x86.encoding.imm_size = encsize; |
694 | 147k | } else if (MI->op1_size > 0) |
695 | 0 | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->op1_size; |
696 | 26.8k | else |
697 | 26.8k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->imm_size; |
698 | | |
699 | 173k | MI->flat_insn->detail->x86.op_count++; |
700 | 173k | } |
701 | 173k | } |
702 | 173k | } |
703 | 1.28M | } |
704 | | |
705 | | static void printMemReference(MCInst *MI, unsigned Op, SStream *O) |
706 | 547k | { |
707 | 547k | MCOperand *BaseReg = MCInst_getOperand(MI, Op + X86_AddrBaseReg); |
708 | 547k | MCOperand *IndexReg = MCInst_getOperand(MI, Op + X86_AddrIndexReg); |
709 | 547k | MCOperand *DispSpec = MCInst_getOperand(MI, Op + X86_AddrDisp); |
710 | 547k | MCOperand *SegReg = MCInst_getOperand(MI, Op + X86_AddrSegmentReg); |
711 | 547k | uint64_t ScaleVal; |
712 | 547k | int segreg; |
713 | 547k | int64_t DispVal = 1; |
714 | | |
715 | 547k | if (MI->csh->detail_opt) { |
716 | 547k | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE]; |
717 | | |
718 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_MEM; |
719 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->x86opsize; |
720 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_REG_INVALID; |
721 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.base = X86_register_map(MCOperand_getReg(BaseReg)); |
722 | 547k | if (MCOperand_getReg(IndexReg) != X86_EIZ) { |
723 | 545k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.index = X86_register_map(MCOperand_getReg(IndexReg)); |
724 | 545k | } |
725 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = 1; |
726 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = 0; |
727 | | |
728 | 547k | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
729 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].access = access[MI->flat_insn->detail->x86.op_count]; |
730 | 547k | } |
731 | | |
732 | | // If this has a segment register, print it. |
733 | 547k | segreg = MCOperand_getReg(SegReg); |
734 | 547k | if (segreg) { |
735 | 14.9k | _printOperand(MI, Op + X86_AddrSegmentReg, O); |
736 | 14.9k | SStream_concat0(O, ":"); |
737 | | |
738 | 14.9k | if (MI->csh->detail_opt) { |
739 | 14.9k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.segment = X86_register_map(segreg); |
740 | 14.9k | } |
741 | 14.9k | } |
742 | | |
743 | 547k | if (MCOperand_isImm(DispSpec)) { |
744 | 547k | DispVal = MCOperand_getImm(DispSpec); |
745 | 547k | if (MI->csh->detail_opt) |
746 | 547k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.disp = DispVal; |
747 | 547k | if (DispVal) { |
748 | 174k | if (MCOperand_getReg(IndexReg) || MCOperand_getReg(BaseReg)) { |
749 | 163k | printInt64(O, DispVal); |
750 | 163k | } else { |
751 | | // only immediate as address of memory |
752 | 10.5k | if (DispVal < 0) { |
753 | 3.86k | SStream_concat(O, "0x%"PRIx64, arch_masks[MI->csh->mode] & DispVal); |
754 | 6.67k | } else { |
755 | 6.67k | if (DispVal > HEX_THRESHOLD) |
756 | 6.11k | SStream_concat(O, "0x%"PRIx64, DispVal); |
757 | 556 | else |
758 | 556 | SStream_concat(O, "%"PRIu64, DispVal); |
759 | 6.67k | } |
760 | 10.5k | } |
761 | 174k | } |
762 | 547k | } |
763 | | |
764 | 547k | if (MCOperand_getReg(IndexReg) || MCOperand_getReg(BaseReg)) { |
765 | 535k | SStream_concat0(O, "("); |
766 | | |
767 | 535k | if (MCOperand_getReg(BaseReg)) |
768 | 534k | _printOperand(MI, Op + X86_AddrBaseReg, O); |
769 | | |
770 | 535k | if (MCOperand_getReg(IndexReg) && MCOperand_getReg(IndexReg) != X86_EIZ) { |
771 | 199k | SStream_concat0(O, ", "); |
772 | 199k | _printOperand(MI, Op + X86_AddrIndexReg, O); |
773 | 199k | ScaleVal = MCOperand_getImm(MCInst_getOperand(MI, Op + X86_AddrScaleAmt)); |
774 | 199k | if (MI->csh->detail_opt) |
775 | 199k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].mem.scale = (int)ScaleVal; |
776 | 199k | if (ScaleVal != 1) { |
777 | 13.3k | SStream_concat(O, ", %u", ScaleVal); |
778 | 13.3k | } |
779 | 199k | } |
780 | | |
781 | 535k | SStream_concat0(O, ")"); |
782 | 535k | } else { |
783 | 11.1k | if (!DispVal) |
784 | 583 | SStream_concat0(O, "0"); |
785 | 11.1k | } |
786 | | |
787 | 547k | if (MI->csh->detail_opt) |
788 | 547k | MI->flat_insn->detail->x86.op_count++; |
789 | 547k | } |
790 | | |
791 | | static void printanymem(MCInst *MI, unsigned OpNo, SStream *O) |
792 | 8.54k | { |
793 | 8.54k | switch(MI->Opcode) { |
794 | 370 | default: break; |
795 | 1.26k | case X86_LEA16r: |
796 | 1.26k | MI->x86opsize = 2; |
797 | 1.26k | break; |
798 | 706 | case X86_LEA32r: |
799 | 1.57k | case X86_LEA64_32r: |
800 | 1.57k | MI->x86opsize = 4; |
801 | 1.57k | break; |
802 | 350 | case X86_LEA64r: |
803 | 350 | MI->x86opsize = 8; |
804 | 350 | break; |
805 | 0 | #ifndef CAPSTONE_X86_REDUCE |
806 | 819 | case X86_BNDCL32rm: |
807 | 1.46k | case X86_BNDCN32rm: |
808 | 1.97k | case X86_BNDCU32rm: |
809 | 2.65k | case X86_BNDSTXmr: |
810 | 3.81k | case X86_BNDLDXrm: |
811 | 4.26k | case X86_BNDCL64rm: |
812 | 4.46k | case X86_BNDCN64rm: |
813 | 4.99k | case X86_BNDCU64rm: |
814 | 4.99k | MI->x86opsize = 16; |
815 | 4.99k | break; |
816 | 8.54k | #endif |
817 | 8.54k | } |
818 | | |
819 | 8.54k | printMemReference(MI, OpNo, O); |
820 | 8.54k | } |
821 | | |
822 | | #include "X86InstPrinter.h" |
823 | | |
824 | | // Include the auto-generated portion of the assembly writer. |
825 | | #ifdef CAPSTONE_X86_REDUCE |
826 | | #include "X86GenAsmWriter_reduce.inc" |
827 | | #else |
828 | | #include "X86GenAsmWriter.inc" |
829 | | #endif |
830 | | |
831 | | #include "X86GenRegisterName.inc" |
832 | | |
833 | | static void printRegName(SStream *OS, unsigned RegNo) |
834 | 1.86M | { |
835 | 1.86M | SStream_concat(OS, "%%%s", getRegisterName(RegNo)); |
836 | 1.86M | } |
837 | | |
838 | | void X86_ATT_printInst(MCInst *MI, SStream *OS, void *info) |
839 | 1.39M | { |
840 | 1.39M | x86_reg reg, reg2; |
841 | 1.39M | enum cs_ac_type access1, access2; |
842 | 1.39M | int i; |
843 | | |
844 | | // perhaps this instruction does not need printer |
845 | 1.39M | if (MI->assembly[0]) { |
846 | 0 | strncpy(OS->buffer, MI->assembly, sizeof(OS->buffer)); |
847 | 0 | return; |
848 | 0 | } |
849 | | |
850 | | // Output CALLpcrel32 as "callq" in 64-bit mode. |
851 | | // In Intel annotation it's always emitted as "call". |
852 | | // |
853 | | // TODO: Probably this hack should be redesigned via InstAlias in |
854 | | // InstrInfo.td as soon as Requires clause is supported properly |
855 | | // for InstAlias. |
856 | 1.39M | if (MI->csh->mode == CS_MODE_64 && MCInst_getOpcode(MI) == X86_CALLpcrel32) { |
857 | 0 | SStream_concat0(OS, "callq\t"); |
858 | 0 | MCInst_setOpcodePub(MI, X86_INS_CALL); |
859 | 0 | printPCRelImm(MI, 0, OS); |
860 | 0 | return; |
861 | 0 | } |
862 | | |
863 | 1.39M | X86_lockrep(MI, OS); |
864 | 1.39M | printInstruction(MI, OS); |
865 | | |
866 | 1.39M | if (MI->has_imm) { |
867 | | // if op_count > 1, then this operand's size is taken from the destination op |
868 | 264k | if (MI->flat_insn->detail->x86.op_count > 1) { |
869 | 145k | if (MI->flat_insn->id != X86_INS_LCALL && MI->flat_insn->id != X86_INS_LJMP && MI->flat_insn->id != X86_INS_JMP) { |
870 | 435k | for (i = 0; i < MI->flat_insn->detail->x86.op_count; i++) { |
871 | 292k | if (MI->flat_insn->detail->x86.operands[i].type == X86_OP_IMM) |
872 | 145k | MI->flat_insn->detail->x86.operands[i].size = |
873 | 145k | MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count - 1].size; |
874 | 292k | } |
875 | 142k | } |
876 | 145k | } else |
877 | 118k | MI->flat_insn->detail->x86.operands[0].size = MI->imm_size; |
878 | 264k | } |
879 | | |
880 | 1.39M | if (MI->csh->detail_opt) { |
881 | 1.39M | uint8_t access[CS_X86_MAXIMUM_OPERAND_SIZE] = {0}; |
882 | | |
883 | | // some instructions need to supply immediate 1 in the first op |
884 | 1.39M | switch(MCInst_getOpcode(MI)) { |
885 | 1.32M | default: |
886 | 1.32M | break; |
887 | 1.32M | case X86_SHL8r1: |
888 | 2.51k | case X86_SHL16r1: |
889 | 4.64k | case X86_SHL32r1: |
890 | 5.82k | case X86_SHL64r1: |
891 | 6.26k | case X86_SAL8r1: |
892 | 8.05k | case X86_SAL16r1: |
893 | 10.9k | case X86_SAL32r1: |
894 | 11.5k | case X86_SAL64r1: |
895 | 11.9k | case X86_SHR8r1: |
896 | 13.2k | case X86_SHR16r1: |
897 | 15.2k | case X86_SHR32r1: |
898 | 17.8k | case X86_SHR64r1: |
899 | 18.4k | case X86_SAR8r1: |
900 | 19.1k | case X86_SAR16r1: |
901 | 20.0k | case X86_SAR32r1: |
902 | 20.5k | case X86_SAR64r1: |
903 | 23.9k | case X86_RCL8r1: |
904 | 26.3k | case X86_RCL16r1: |
905 | 28.8k | case X86_RCL32r1: |
906 | 30.1k | case X86_RCL64r1: |
907 | 30.4k | case X86_RCR8r1: |
908 | 31.1k | case X86_RCR16r1: |
909 | 32.1k | case X86_RCR32r1: |
910 | 32.6k | case X86_RCR64r1: |
911 | 33.4k | case X86_ROL8r1: |
912 | 34.2k | case X86_ROL16r1: |
913 | 34.9k | case X86_ROL32r1: |
914 | 35.8k | case X86_ROL64r1: |
915 | 36.4k | case X86_ROR8r1: |
916 | 38.3k | case X86_ROR16r1: |
917 | 39.7k | case X86_ROR32r1: |
918 | 40.5k | case X86_ROR64r1: |
919 | 41.5k | case X86_SHL8m1: |
920 | 42.7k | case X86_SHL16m1: |
921 | 43.8k | case X86_SHL32m1: |
922 | 44.6k | case X86_SHL64m1: |
923 | 45.3k | case X86_SAL8m1: |
924 | 46.0k | case X86_SAL16m1: |
925 | 46.9k | case X86_SAL32m1: |
926 | 47.2k | case X86_SAL64m1: |
927 | 48.0k | case X86_SHR8m1: |
928 | 48.6k | case X86_SHR16m1: |
929 | 50.1k | case X86_SHR32m1: |
930 | 51.1k | case X86_SHR64m1: |
931 | 51.7k | case X86_SAR8m1: |
932 | 52.2k | case X86_SAR16m1: |
933 | 53.3k | case X86_SAR32m1: |
934 | 54.3k | case X86_SAR64m1: |
935 | 55.1k | case X86_RCL8m1: |
936 | 56.0k | case X86_RCL16m1: |
937 | 56.7k | case X86_RCL32m1: |
938 | 57.1k | case X86_RCL64m1: |
939 | 57.4k | case X86_RCR8m1: |
940 | 58.4k | case X86_RCR16m1: |
941 | 59.6k | case X86_RCR32m1: |
942 | 60.7k | case X86_RCR64m1: |
943 | 62.1k | case X86_ROL8m1: |
944 | 62.8k | case X86_ROL16m1: |
945 | 65.3k | case X86_ROL32m1: |
946 | 65.8k | case X86_ROL64m1: |
947 | 66.6k | case X86_ROR8m1: |
948 | 67.9k | case X86_ROR16m1: |
949 | 70.1k | case X86_ROR32m1: |
950 | 71.2k | case X86_ROR64m1: |
951 | | // shift all the ops right to leave 1st slot for this new register op |
952 | 71.2k | memmove(&(MI->flat_insn->detail->x86.operands[1]), &(MI->flat_insn->detail->x86.operands[0]), |
953 | 71.2k | sizeof(MI->flat_insn->detail->x86.operands[0]) * (ARR_SIZE(MI->flat_insn->detail->x86.operands) - 1)); |
954 | 71.2k | MI->flat_insn->detail->x86.operands[0].type = X86_OP_IMM; |
955 | 71.2k | MI->flat_insn->detail->x86.operands[0].imm = 1; |
956 | 71.2k | MI->flat_insn->detail->x86.operands[0].size = 1; |
957 | 71.2k | MI->flat_insn->detail->x86.op_count++; |
958 | 1.39M | } |
959 | | |
960 | | // special instruction needs to supply register op |
961 | | // first op can be embedded in the asm by llvm. |
962 | | // so we have to add the missing register as the first operand |
963 | | |
964 | | //printf(">>> opcode = %u\n", MCInst_getOpcode(MI)); |
965 | | |
966 | 1.39M | reg = X86_insn_reg_att(MCInst_getOpcode(MI), &access1); |
967 | 1.39M | if (reg) { |
968 | | // shift all the ops right to leave 1st slot for this new register op |
969 | 80.2k | memmove(&(MI->flat_insn->detail->x86.operands[1]), &(MI->flat_insn->detail->x86.operands[0]), |
970 | 80.2k | sizeof(MI->flat_insn->detail->x86.operands[0]) * (ARR_SIZE(MI->flat_insn->detail->x86.operands) - 1)); |
971 | 80.2k | MI->flat_insn->detail->x86.operands[0].type = X86_OP_REG; |
972 | 80.2k | MI->flat_insn->detail->x86.operands[0].reg = reg; |
973 | 80.2k | MI->flat_insn->detail->x86.operands[0].size = MI->csh->regsize_map[reg]; |
974 | 80.2k | MI->flat_insn->detail->x86.operands[0].access = access1; |
975 | | |
976 | 80.2k | MI->flat_insn->detail->x86.op_count++; |
977 | 1.31M | } else { |
978 | 1.31M | if (X86_insn_reg_att2(MCInst_getOpcode(MI), ®, &access1, ®2, &access2)) { |
979 | | |
980 | 28.3k | MI->flat_insn->detail->x86.operands[0].type = X86_OP_REG; |
981 | 28.3k | MI->flat_insn->detail->x86.operands[0].reg = reg; |
982 | 28.3k | MI->flat_insn->detail->x86.operands[0].size = MI->csh->regsize_map[reg]; |
983 | 28.3k | MI->flat_insn->detail->x86.operands[0].access = access1; |
984 | 28.3k | MI->flat_insn->detail->x86.operands[1].type = X86_OP_REG; |
985 | 28.3k | MI->flat_insn->detail->x86.operands[1].reg = reg2; |
986 | 28.3k | MI->flat_insn->detail->x86.operands[1].size = MI->csh->regsize_map[reg2]; |
987 | 28.3k | MI->flat_insn->detail->x86.operands[1].access = access2; |
988 | 28.3k | MI->flat_insn->detail->x86.op_count = 2; |
989 | 28.3k | } |
990 | 1.31M | } |
991 | | |
992 | 1.39M | #ifndef CAPSTONE_DIET |
993 | 1.39M | get_op_access(MI->csh, MCInst_getOpcode(MI), access, &MI->flat_insn->detail->x86.eflags); |
994 | 1.39M | MI->flat_insn->detail->x86.operands[0].access = access[0]; |
995 | 1.39M | MI->flat_insn->detail->x86.operands[1].access = access[1]; |
996 | 1.39M | #endif |
997 | 1.39M | } |
998 | 1.39M | } |
999 | | |
1000 | | #endif |