Coverage Report

Created: 2026-05-30 06:22

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonev5/arch/X86/X86Mapping.c
Line
Count
Source
1
/* Capstone Disassembly Engine */
2
/* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2019 */
3
4
#ifdef CAPSTONE_HAS_X86
5
6
#if defined(CAPSTONE_HAS_OSXKERNEL)
7
#include <Availability.h>
8
#endif
9
10
#include <string.h>
11
#ifndef CAPSTONE_HAS_OSXKERNEL
12
#include <stdlib.h>
13
#endif
14
15
#include "X86Mapping.h"
16
#include "X86DisassemblerDecoder.h"
17
18
#include "../../utils.h"
19
20
21
const uint64_t arch_masks[9] = {
22
  0, 0xff,
23
  0xffff, // 16bit
24
  0,
25
  0xffffffff, // 32bit
26
  0, 0, 0,
27
  0xffffffffffffffffLL  // 64bit
28
};
29
30
static const x86_reg sib_base_map[] = {
31
  X86_REG_INVALID,
32
#define ENTRY(x) X86_REG_##x,
33
  ALL_SIB_BASES
34
#undef ENTRY
35
};
36
37
// Fill-ins to make the compiler happy.  These constants are never actually
38
// assigned; they are just filler to make an automatically-generated switch
39
// statement work.
40
enum {
41
  X86_REG_BX_SI = 500,
42
  X86_REG_BX_DI = 501,
43
  X86_REG_BP_SI = 502,
44
  X86_REG_BP_DI = 503,
45
  X86_REG_sib   = 504,
46
  X86_REG_sib64 = 505
47
};
48
49
static const x86_reg sib_index_map[] = {
50
  X86_REG_INVALID,
51
#define ENTRY(x) X86_REG_##x,
52
  ALL_EA_BASES
53
  REGS_XMM
54
  REGS_YMM
55
  REGS_ZMM
56
#undef ENTRY
57
};
58
59
static const x86_reg segment_map[] = {
60
  X86_REG_INVALID,
61
  X86_REG_CS,
62
  X86_REG_SS,
63
  X86_REG_DS,
64
  X86_REG_ES,
65
  X86_REG_FS,
66
  X86_REG_GS,
67
};
68
69
x86_reg x86_map_sib_base(int r)
70
1.53M
{
71
1.53M
  return sib_base_map[r];
72
1.53M
}
73
74
x86_reg x86_map_sib_index(int r)
75
1.53M
{
76
1.53M
  return sib_index_map[r];
77
1.53M
}
78
79
x86_reg x86_map_segment(int r)
80
0
{
81
0
  return segment_map[r];
82
0
}
83
84
#ifndef CAPSTONE_DIET
85
static const name_map reg_name_maps[] = {
86
  { X86_REG_INVALID, NULL },
87
88
  { X86_REG_AH, "ah" },
89
  { X86_REG_AL, "al" },
90
  { X86_REG_AX, "ax" },
91
  { X86_REG_BH, "bh" },
92
  { X86_REG_BL, "bl" },
93
  { X86_REG_BP, "bp" },
94
  { X86_REG_BPL, "bpl" },
95
  { X86_REG_BX, "bx" },
96
  { X86_REG_CH, "ch" },
97
  { X86_REG_CL, "cl" },
98
  { X86_REG_CS, "cs" },
99
  { X86_REG_CX, "cx" },
100
  { X86_REG_DH, "dh" },
101
  { X86_REG_DI, "di" },
102
  { X86_REG_DIL, "dil" },
103
  { X86_REG_DL, "dl" },
104
  { X86_REG_DS, "ds" },
105
  { X86_REG_DX, "dx" },
106
  { X86_REG_EAX, "eax" },
107
  { X86_REG_EBP, "ebp" },
108
  { X86_REG_EBX, "ebx" },
109
  { X86_REG_ECX, "ecx" },
110
  { X86_REG_EDI, "edi" },
111
  { X86_REG_EDX, "edx" },
112
  { X86_REG_EFLAGS, "flags" },
113
  { X86_REG_EIP, "eip" },
114
  { X86_REG_EIZ, "eiz" },
115
  { X86_REG_ES, "es" },
116
  { X86_REG_ESI, "esi" },
117
  { X86_REG_ESP, "esp" },
118
  { X86_REG_FPSW, "fpsw" },
119
  { X86_REG_FS, "fs" },
120
  { X86_REG_GS, "gs" },
121
  { X86_REG_IP, "ip" },
122
  { X86_REG_RAX, "rax" },
123
  { X86_REG_RBP, "rbp" },
124
  { X86_REG_RBX, "rbx" },
125
  { X86_REG_RCX, "rcx" },
126
  { X86_REG_RDI, "rdi" },
127
  { X86_REG_RDX, "rdx" },
128
  { X86_REG_RIP, "rip" },
129
  { X86_REG_RIZ, "riz" },
130
  { X86_REG_RSI, "rsi" },
131
  { X86_REG_RSP, "rsp" },
132
  { X86_REG_SI, "si" },
133
  { X86_REG_SIL, "sil" },
134
  { X86_REG_SP, "sp" },
135
  { X86_REG_SPL, "spl" },
136
  { X86_REG_SS, "ss" },
137
  { X86_REG_CR0, "cr0" },
138
  { X86_REG_CR1, "cr1" },
139
  { X86_REG_CR2, "cr2" },
140
  { X86_REG_CR3, "cr3" },
141
  { X86_REG_CR4, "cr4" },
142
  { X86_REG_CR5, "cr5" },
143
  { X86_REG_CR6, "cr6" },
144
  { X86_REG_CR7, "cr7" },
145
  { X86_REG_CR8, "cr8" },
146
  { X86_REG_CR9, "cr9" },
147
  { X86_REG_CR10, "cr10" },
148
  { X86_REG_CR11, "cr11" },
149
  { X86_REG_CR12, "cr12" },
150
  { X86_REG_CR13, "cr13" },
151
  { X86_REG_CR14, "cr14" },
152
  { X86_REG_CR15, "cr15" },
153
  { X86_REG_DR0, "dr0" },
154
  { X86_REG_DR1, "dr1" },
155
  { X86_REG_DR2, "dr2" },
156
  { X86_REG_DR3, "dr3" },
157
  { X86_REG_DR4, "dr4" },
158
  { X86_REG_DR5, "dr5" },
159
  { X86_REG_DR6, "dr6" },
160
  { X86_REG_DR7, "dr7" },
161
  { X86_REG_DR8, "dr8" },
162
  { X86_REG_DR9, "dr9" },
163
  { X86_REG_DR10, "dr10" },
164
  { X86_REG_DR11, "dr11" },
165
  { X86_REG_DR12, "dr12" },
166
  { X86_REG_DR13, "dr13" },
167
  { X86_REG_DR14, "dr14" },
168
  { X86_REG_DR15, "dr15" },
169
  { X86_REG_FP0, "fp0" },
170
  { X86_REG_FP1, "fp1" },
171
  { X86_REG_FP2, "fp2" },
172
  { X86_REG_FP3, "fp3" },
173
  { X86_REG_FP4, "fp4" },
174
  { X86_REG_FP5, "fp5" },
175
  { X86_REG_FP6, "fp6" },
176
  { X86_REG_FP7, "fp7" },
177
  { X86_REG_K0, "k0" },
178
  { X86_REG_K1, "k1" },
179
  { X86_REG_K2, "k2" },
180
  { X86_REG_K3, "k3" },
181
  { X86_REG_K4, "k4" },
182
  { X86_REG_K5, "k5" },
183
  { X86_REG_K6, "k6" },
184
  { X86_REG_K7, "k7" },
185
  { X86_REG_MM0, "mm0" },
186
  { X86_REG_MM1, "mm1" },
187
  { X86_REG_MM2, "mm2" },
188
  { X86_REG_MM3, "mm3" },
189
  { X86_REG_MM4, "mm4" },
190
  { X86_REG_MM5, "mm5" },
191
  { X86_REG_MM6, "mm6" },
192
  { X86_REG_MM7, "mm7" },
193
  { X86_REG_R8, "r8" },
194
  { X86_REG_R9, "r9" },
195
  { X86_REG_R10, "r10" },
196
  { X86_REG_R11, "r11" },
197
  { X86_REG_R12, "r12" },
198
  { X86_REG_R13, "r13" },
199
  { X86_REG_R14, "r14" },
200
  { X86_REG_R15, "r15" },
201
  { X86_REG_ST0, "st(0)" },
202
  { X86_REG_ST1, "st(1)" },
203
  { X86_REG_ST2, "st(2)" },
204
  { X86_REG_ST3, "st(3)" },
205
  { X86_REG_ST4, "st(4)" },
206
  { X86_REG_ST5, "st(5)" },
207
  { X86_REG_ST6, "st(6)" },
208
  { X86_REG_ST7, "st(7)" },
209
  { X86_REG_XMM0, "xmm0" },
210
  { X86_REG_XMM1, "xmm1" },
211
  { X86_REG_XMM2, "xmm2" },
212
  { X86_REG_XMM3, "xmm3" },
213
  { X86_REG_XMM4, "xmm4" },
214
  { X86_REG_XMM5, "xmm5" },
215
  { X86_REG_XMM6, "xmm6" },
216
  { X86_REG_XMM7, "xmm7" },
217
  { X86_REG_XMM8, "xmm8" },
218
  { X86_REG_XMM9, "xmm9" },
219
  { X86_REG_XMM10, "xmm10" },
220
  { X86_REG_XMM11, "xmm11" },
221
  { X86_REG_XMM12, "xmm12" },
222
  { X86_REG_XMM13, "xmm13" },
223
  { X86_REG_XMM14, "xmm14" },
224
  { X86_REG_XMM15, "xmm15" },
225
  { X86_REG_XMM16, "xmm16" },
226
  { X86_REG_XMM17, "xmm17" },
227
  { X86_REG_XMM18, "xmm18" },
228
  { X86_REG_XMM19, "xmm19" },
229
  { X86_REG_XMM20, "xmm20" },
230
  { X86_REG_XMM21, "xmm21" },
231
  { X86_REG_XMM22, "xmm22" },
232
  { X86_REG_XMM23, "xmm23" },
233
  { X86_REG_XMM24, "xmm24" },
234
  { X86_REG_XMM25, "xmm25" },
235
  { X86_REG_XMM26, "xmm26" },
236
  { X86_REG_XMM27, "xmm27" },
237
  { X86_REG_XMM28, "xmm28" },
238
  { X86_REG_XMM29, "xmm29" },
239
  { X86_REG_XMM30, "xmm30" },
240
  { X86_REG_XMM31, "xmm31" },
241
  { X86_REG_YMM0, "ymm0" },
242
  { X86_REG_YMM1, "ymm1" },
243
  { X86_REG_YMM2, "ymm2" },
244
  { X86_REG_YMM3, "ymm3" },
245
  { X86_REG_YMM4, "ymm4" },
246
  { X86_REG_YMM5, "ymm5" },
247
  { X86_REG_YMM6, "ymm6" },
248
  { X86_REG_YMM7, "ymm7" },
249
  { X86_REG_YMM8, "ymm8" },
250
  { X86_REG_YMM9, "ymm9" },
251
  { X86_REG_YMM10, "ymm10" },
252
  { X86_REG_YMM11, "ymm11" },
253
  { X86_REG_YMM12, "ymm12" },
254
  { X86_REG_YMM13, "ymm13" },
255
  { X86_REG_YMM14, "ymm14" },
256
  { X86_REG_YMM15, "ymm15" },
257
  { X86_REG_YMM16, "ymm16" },
258
  { X86_REG_YMM17, "ymm17" },
259
  { X86_REG_YMM18, "ymm18" },
260
  { X86_REG_YMM19, "ymm19" },
261
  { X86_REG_YMM20, "ymm20" },
262
  { X86_REG_YMM21, "ymm21" },
263
  { X86_REG_YMM22, "ymm22" },
264
  { X86_REG_YMM23, "ymm23" },
265
  { X86_REG_YMM24, "ymm24" },
266
  { X86_REG_YMM25, "ymm25" },
267
  { X86_REG_YMM26, "ymm26" },
268
  { X86_REG_YMM27, "ymm27" },
269
  { X86_REG_YMM28, "ymm28" },
270
  { X86_REG_YMM29, "ymm29" },
271
  { X86_REG_YMM30, "ymm30" },
272
  { X86_REG_YMM31, "ymm31" },
273
  { X86_REG_ZMM0, "zmm0" },
274
  { X86_REG_ZMM1, "zmm1" },
275
  { X86_REG_ZMM2, "zmm2" },
276
  { X86_REG_ZMM3, "zmm3" },
277
  { X86_REG_ZMM4, "zmm4" },
278
  { X86_REG_ZMM5, "zmm5" },
279
  { X86_REG_ZMM6, "zmm6" },
280
  { X86_REG_ZMM7, "zmm7" },
281
  { X86_REG_ZMM8, "zmm8" },
282
  { X86_REG_ZMM9, "zmm9" },
283
  { X86_REG_ZMM10, "zmm10" },
284
  { X86_REG_ZMM11, "zmm11" },
285
  { X86_REG_ZMM12, "zmm12" },
286
  { X86_REG_ZMM13, "zmm13" },
287
  { X86_REG_ZMM14, "zmm14" },
288
  { X86_REG_ZMM15, "zmm15" },
289
  { X86_REG_ZMM16, "zmm16" },
290
  { X86_REG_ZMM17, "zmm17" },
291
  { X86_REG_ZMM18, "zmm18" },
292
  { X86_REG_ZMM19, "zmm19" },
293
  { X86_REG_ZMM20, "zmm20" },
294
  { X86_REG_ZMM21, "zmm21" },
295
  { X86_REG_ZMM22, "zmm22" },
296
  { X86_REG_ZMM23, "zmm23" },
297
  { X86_REG_ZMM24, "zmm24" },
298
  { X86_REG_ZMM25, "zmm25" },
299
  { X86_REG_ZMM26, "zmm26" },
300
  { X86_REG_ZMM27, "zmm27" },
301
  { X86_REG_ZMM28, "zmm28" },
302
  { X86_REG_ZMM29, "zmm29" },
303
  { X86_REG_ZMM30, "zmm30" },
304
  { X86_REG_ZMM31, "zmm31" },
305
  { X86_REG_R8B, "r8b" },
306
  { X86_REG_R9B, "r9b" },
307
  { X86_REG_R10B, "r10b" },
308
  { X86_REG_R11B, "r11b" },
309
  { X86_REG_R12B, "r12b" },
310
  { X86_REG_R13B, "r13b" },
311
  { X86_REG_R14B, "r14b" },
312
  { X86_REG_R15B, "r15b" },
313
  { X86_REG_R8D, "r8d" },
314
  { X86_REG_R9D, "r9d" },
315
  { X86_REG_R10D, "r10d" },
316
  { X86_REG_R11D, "r11d" },
317
  { X86_REG_R12D, "r12d" },
318
  { X86_REG_R13D, "r13d" },
319
  { X86_REG_R14D, "r14d" },
320
  { X86_REG_R15D, "r15d" },
321
  { X86_REG_R8W, "r8w" },
322
  { X86_REG_R9W, "r9w" },
323
  { X86_REG_R10W, "r10w" },
324
  { X86_REG_R11W, "r11w" },
325
  { X86_REG_R12W, "r12w" },
326
  { X86_REG_R13W, "r13w" },
327
  { X86_REG_R14W, "r14w" },
328
  { X86_REG_R15W, "r15w" },
329
330
  { X86_REG_BND0, "bnd0" },
331
  { X86_REG_BND1, "bnd1" },
332
  { X86_REG_BND2, "bnd2" },
333
  { X86_REG_BND3, "bnd3" },
334
};
335
#endif
336
337
// register size in non-64bit mode
338
const uint8_t regsize_map_32 [] = {
339
  0,  //  { X86_REG_INVALID, NULL },
340
  1,  // { X86_REG_AH, "ah" },
341
  1,  // { X86_REG_AL, "al" },
342
  2,  // { X86_REG_AX, "ax" },
343
  1,  // { X86_REG_BH, "bh" },
344
  1,  // { X86_REG_BL, "bl" },
345
  2,  // { X86_REG_BP, "bp" },
346
  1,  // { X86_REG_BPL, "bpl" },
347
  2,  // { X86_REG_BX, "bx" },
348
  1,  // { X86_REG_CH, "ch" },
349
  1,  // { X86_REG_CL, "cl" },
350
  2,  // { X86_REG_CS, "cs" },
351
  2,  // { X86_REG_CX, "cx" },
352
  1,  // { X86_REG_DH, "dh" },
353
  2,  // { X86_REG_DI, "di" },
354
  1,  // { X86_REG_DIL, "dil" },
355
  1,  // { X86_REG_DL, "dl" },
356
  2,  // { X86_REG_DS, "ds" },
357
  2,  // { X86_REG_DX, "dx" },
358
  4,  // { X86_REG_EAX, "eax" },
359
  4,  // { X86_REG_EBP, "ebp" },
360
  4,  // { X86_REG_EBX, "ebx" },
361
  4,  // { X86_REG_ECX, "ecx" },
362
  4,  // { X86_REG_EDI, "edi" },
363
  4,  // { X86_REG_EDX, "edx" },
364
  4,  // { X86_REG_EFLAGS, "flags" },
365
  4,  // { X86_REG_EIP, "eip" },
366
  4,  // { X86_REG_EIZ, "eiz" },
367
  2,  // { X86_REG_ES, "es" },
368
  4,  // { X86_REG_ESI, "esi" },
369
  4,  // { X86_REG_ESP, "esp" },
370
  10, // { X86_REG_FPSW, "fpsw" },
371
  2,  // { X86_REG_FS, "fs" },
372
  2,  // { X86_REG_GS, "gs" },
373
  2,  // { X86_REG_IP, "ip" },
374
  8,  // { X86_REG_RAX, "rax" },
375
  8,  // { X86_REG_RBP, "rbp" },
376
  8,  // { X86_REG_RBX, "rbx" },
377
  8,  // { X86_REG_RCX, "rcx" },
378
  8,  // { X86_REG_RDI, "rdi" },
379
  8,  // { X86_REG_RDX, "rdx" },
380
  8,  // { X86_REG_RIP, "rip" },
381
  8,  // { X86_REG_RIZ, "riz" },
382
  8,  // { X86_REG_RSI, "rsi" },
383
  8,  // { X86_REG_RSP, "rsp" },
384
  2,  // { X86_REG_SI, "si" },
385
  1,  // { X86_REG_SIL, "sil" },
386
  2,  // { X86_REG_SP, "sp" },
387
  1,  // { X86_REG_SPL, "spl" },
388
  2,  // { X86_REG_SS, "ss" },
389
  4,  // { X86_REG_CR0, "cr0" },
390
  4,  // { X86_REG_CR1, "cr1" },
391
  4,  // { X86_REG_CR2, "cr2" },
392
  4,  // { X86_REG_CR3, "cr3" },
393
  4,  // { X86_REG_CR4, "cr4" },
394
  8,  // { X86_REG_CR5, "cr5" },
395
  8,  // { X86_REG_CR6, "cr6" },
396
  8,  // { X86_REG_CR7, "cr7" },
397
  8,  // { X86_REG_CR8, "cr8" },
398
  8,  // { X86_REG_CR9, "cr9" },
399
  8,  // { X86_REG_CR10, "cr10" },
400
  8,  // { X86_REG_CR11, "cr11" },
401
  8,  // { X86_REG_CR12, "cr12" },
402
  8,  // { X86_REG_CR13, "cr13" },
403
  8,  // { X86_REG_CR14, "cr14" },
404
  8,  // { X86_REG_CR15, "cr15" },
405
  4,  // { X86_REG_DR0, "dr0" },
406
  4,  // { X86_REG_DR1, "dr1" },
407
  4,  // { X86_REG_DR2, "dr2" },
408
  4,  // { X86_REG_DR3, "dr3" },
409
  4,  // { X86_REG_DR4, "dr4" },
410
  4,  // { X86_REG_DR5, "dr5" },
411
  4,  // { X86_REG_DR6, "dr6" },
412
  4,  // { X86_REG_DR7, "dr7" },
413
  4,  // { X86_REG_DR8, "dr8" },
414
  4,  // { X86_REG_DR9, "dr9" },
415
  4,  // { X86_REG_DR10, "dr10" },
416
  4,  // { X86_REG_DR11, "dr11" },
417
  4,  // { X86_REG_DR12, "dr12" },
418
  4,  // { X86_REG_DR13, "dr13" },
419
  4,  // { X86_REG_DR14, "dr14" },
420
  4,  // { X86_REG_DR15, "dr15" },
421
  10, // { X86_REG_FP0, "fp0" },
422
  10, // { X86_REG_FP1, "fp1" },
423
  10, // { X86_REG_FP2, "fp2" },
424
  10, // { X86_REG_FP3, "fp3" },
425
  10, // { X86_REG_FP4, "fp4" },
426
  10, // { X86_REG_FP5, "fp5" },
427
  10, // { X86_REG_FP6, "fp6" },
428
  10, // { X86_REG_FP7, "fp7" },
429
  2,  // { X86_REG_K0, "k0" },
430
  2,  // { X86_REG_K1, "k1" },
431
  2,  // { X86_REG_K2, "k2" },
432
  2,  // { X86_REG_K3, "k3" },
433
  2,  // { X86_REG_K4, "k4" },
434
  2,  // { X86_REG_K5, "k5" },
435
  2,  // { X86_REG_K6, "k6" },
436
  2,  // { X86_REG_K7, "k7" },
437
  8,  // { X86_REG_MM0, "mm0" },
438
  8,  // { X86_REG_MM1, "mm1" },
439
  8,  // { X86_REG_MM2, "mm2" },
440
  8,  // { X86_REG_MM3, "mm3" },
441
  8,  // { X86_REG_MM4, "mm4" },
442
  8,  // { X86_REG_MM5, "mm5" },
443
  8,  // { X86_REG_MM6, "mm6" },
444
  8,  // { X86_REG_MM7, "mm7" },
445
  8,  // { X86_REG_R8, "r8" },
446
  8,  // { X86_REG_R9, "r9" },
447
  8,  // { X86_REG_R10, "r10" },
448
  8,  // { X86_REG_R11, "r11" },
449
  8,  // { X86_REG_R12, "r12" },
450
  8,  // { X86_REG_R13, "r13" },
451
  8,  // { X86_REG_R14, "r14" },
452
  8,  // { X86_REG_R15, "r15" },
453
  10, // { X86_REG_ST0, "st0" },
454
  10, // { X86_REG_ST1, "st1" },
455
  10, // { X86_REG_ST2, "st2" },
456
  10, // { X86_REG_ST3, "st3" },
457
  10, // { X86_REG_ST4, "st4" },
458
  10, // { X86_REG_ST5, "st5" },
459
  10, // { X86_REG_ST6, "st6" },
460
  10, // { X86_REG_ST7, "st7" },
461
  16, // { X86_REG_XMM0, "xmm0" },
462
  16, // { X86_REG_XMM1, "xmm1" },
463
  16, // { X86_REG_XMM2, "xmm2" },
464
  16, // { X86_REG_XMM3, "xmm3" },
465
  16, // { X86_REG_XMM4, "xmm4" },
466
  16, // { X86_REG_XMM5, "xmm5" },
467
  16, // { X86_REG_XMM6, "xmm6" },
468
  16, // { X86_REG_XMM7, "xmm7" },
469
  16, // { X86_REG_XMM8, "xmm8" },
470
  16, // { X86_REG_XMM9, "xmm9" },
471
  16, // { X86_REG_XMM10, "xmm10" },
472
  16, // { X86_REG_XMM11, "xmm11" },
473
  16, // { X86_REG_XMM12, "xmm12" },
474
  16, // { X86_REG_XMM13, "xmm13" },
475
  16, // { X86_REG_XMM14, "xmm14" },
476
  16, // { X86_REG_XMM15, "xmm15" },
477
  16, // { X86_REG_XMM16, "xmm16" },
478
  16, // { X86_REG_XMM17, "xmm17" },
479
  16, // { X86_REG_XMM18, "xmm18" },
480
  16, // { X86_REG_XMM19, "xmm19" },
481
  16, // { X86_REG_XMM20, "xmm20" },
482
  16, // { X86_REG_XMM21, "xmm21" },
483
  16, // { X86_REG_XMM22, "xmm22" },
484
  16, // { X86_REG_XMM23, "xmm23" },
485
  16, // { X86_REG_XMM24, "xmm24" },
486
  16, // { X86_REG_XMM25, "xmm25" },
487
  16, // { X86_REG_XMM26, "xmm26" },
488
  16, // { X86_REG_XMM27, "xmm27" },
489
  16, // { X86_REG_XMM28, "xmm28" },
490
  16, // { X86_REG_XMM29, "xmm29" },
491
  16, // { X86_REG_XMM30, "xmm30" },
492
  16, // { X86_REG_XMM31, "xmm31" },
493
  32, // { X86_REG_YMM0, "ymm0" },
494
  32, // { X86_REG_YMM1, "ymm1" },
495
  32, // { X86_REG_YMM2, "ymm2" },
496
  32, // { X86_REG_YMM3, "ymm3" },
497
  32, // { X86_REG_YMM4, "ymm4" },
498
  32, // { X86_REG_YMM5, "ymm5" },
499
  32, // { X86_REG_YMM6, "ymm6" },
500
  32, // { X86_REG_YMM7, "ymm7" },
501
  32, // { X86_REG_YMM8, "ymm8" },
502
  32, // { X86_REG_YMM9, "ymm9" },
503
  32, // { X86_REG_YMM10, "ymm10" },
504
  32, // { X86_REG_YMM11, "ymm11" },
505
  32, // { X86_REG_YMM12, "ymm12" },
506
  32, // { X86_REG_YMM13, "ymm13" },
507
  32, // { X86_REG_YMM14, "ymm14" },
508
  32, // { X86_REG_YMM15, "ymm15" },
509
  32, // { X86_REG_YMM16, "ymm16" },
510
  32, // { X86_REG_YMM17, "ymm17" },
511
  32, // { X86_REG_YMM18, "ymm18" },
512
  32, // { X86_REG_YMM19, "ymm19" },
513
  32, // { X86_REG_YMM20, "ymm20" },
514
  32, // { X86_REG_YMM21, "ymm21" },
515
  32, // { X86_REG_YMM22, "ymm22" },
516
  32, // { X86_REG_YMM23, "ymm23" },
517
  32, // { X86_REG_YMM24, "ymm24" },
518
  32, // { X86_REG_YMM25, "ymm25" },
519
  32, // { X86_REG_YMM26, "ymm26" },
520
  32, // { X86_REG_YMM27, "ymm27" },
521
  32, // { X86_REG_YMM28, "ymm28" },
522
  32, // { X86_REG_YMM29, "ymm29" },
523
  32, // { X86_REG_YMM30, "ymm30" },
524
  32, // { X86_REG_YMM31, "ymm31" },
525
  64, // { X86_REG_ZMM0, "zmm0" },
526
  64, // { X86_REG_ZMM1, "zmm1" },
527
  64, // { X86_REG_ZMM2, "zmm2" },
528
  64, // { X86_REG_ZMM3, "zmm3" },
529
  64, // { X86_REG_ZMM4, "zmm4" },
530
  64, // { X86_REG_ZMM5, "zmm5" },
531
  64, // { X86_REG_ZMM6, "zmm6" },
532
  64, // { X86_REG_ZMM7, "zmm7" },
533
  64, // { X86_REG_ZMM8, "zmm8" },
534
  64, // { X86_REG_ZMM9, "zmm9" },
535
  64, // { X86_REG_ZMM10, "zmm10" },
536
  64, // { X86_REG_ZMM11, "zmm11" },
537
  64, // { X86_REG_ZMM12, "zmm12" },
538
  64, // { X86_REG_ZMM13, "zmm13" },
539
  64, // { X86_REG_ZMM14, "zmm14" },
540
  64, // { X86_REG_ZMM15, "zmm15" },
541
  64, // { X86_REG_ZMM16, "zmm16" },
542
  64, // { X86_REG_ZMM17, "zmm17" },
543
  64, // { X86_REG_ZMM18, "zmm18" },
544
  64, // { X86_REG_ZMM19, "zmm19" },
545
  64, // { X86_REG_ZMM20, "zmm20" },
546
  64, // { X86_REG_ZMM21, "zmm21" },
547
  64, // { X86_REG_ZMM22, "zmm22" },
548
  64, // { X86_REG_ZMM23, "zmm23" },
549
  64, // { X86_REG_ZMM24, "zmm24" },
550
  64, // { X86_REG_ZMM25, "zmm25" },
551
  64, // { X86_REG_ZMM26, "zmm26" },
552
  64, // { X86_REG_ZMM27, "zmm27" },
553
  64, // { X86_REG_ZMM28, "zmm28" },
554
  64, // { X86_REG_ZMM29, "zmm29" },
555
  64, // { X86_REG_ZMM30, "zmm30" },
556
  64, // { X86_REG_ZMM31, "zmm31" },
557
  1,  // { X86_REG_R8B, "r8b" },
558
  1,  // { X86_REG_R9B, "r9b" },
559
  1,  // { X86_REG_R10B, "r10b" },
560
  1,  // { X86_REG_R11B, "r11b" },
561
  1,  // { X86_REG_R12B, "r12b" },
562
  1,  // { X86_REG_R13B, "r13b" },
563
  1,  // { X86_REG_R14B, "r14b" },
564
  1,  // { X86_REG_R15B, "r15b" },
565
  4,  // { X86_REG_R8D, "r8d" },
566
  4,  // { X86_REG_R9D, "r9d" },
567
  4,  // { X86_REG_R10D, "r10d" },
568
  4,  // { X86_REG_R11D, "r11d" },
569
  4,  // { X86_REG_R12D, "r12d" },
570
  4,  // { X86_REG_R13D, "r13d" },
571
  4,  // { X86_REG_R14D, "r14d" },
572
  4,  // { X86_REG_R15D, "r15d" },
573
  2,  // { X86_REG_R8W, "r8w" },
574
  2,  // { X86_REG_R9W, "r9w" },
575
  2,  // { X86_REG_R10W, "r10w" },
576
  2,  // { X86_REG_R11W, "r11w" },
577
  2,  // { X86_REG_R12W, "r12w" },
578
  2,  // { X86_REG_R13W, "r13w" },
579
  2,  // { X86_REG_R14W, "r14w" },
580
  2,  // { X86_REG_R15W, "r15w" },
581
  16, // { X86_REG_BND0, "bnd0" },
582
  16, // { X86_REG_BND1, "bnd0" },
583
  16, // { X86_REG_BND2, "bnd0" },
584
  16, // { X86_REG_BND3, "bnd0" },
585
};
586
587
// register size in 64bit mode
588
const uint8_t regsize_map_64 [] = {
589
  0,  //  { X86_REG_INVALID, NULL },
590
  1,  // { X86_REG_AH, "ah" },
591
  1,  // { X86_REG_AL, "al" },
592
  2,  // { X86_REG_AX, "ax" },
593
  1,  // { X86_REG_BH, "bh" },
594
  1,  // { X86_REG_BL, "bl" },
595
  2,  // { X86_REG_BP, "bp" },
596
  1,  // { X86_REG_BPL, "bpl" },
597
  2,  // { X86_REG_BX, "bx" },
598
  1,  // { X86_REG_CH, "ch" },
599
  1,  // { X86_REG_CL, "cl" },
600
  2,  // { X86_REG_CS, "cs" },
601
  2,  // { X86_REG_CX, "cx" },
602
  1,  // { X86_REG_DH, "dh" },
603
  2,  // { X86_REG_DI, "di" },
604
  1,  // { X86_REG_DIL, "dil" },
605
  1,  // { X86_REG_DL, "dl" },
606
  2,  // { X86_REG_DS, "ds" },
607
  2,  // { X86_REG_DX, "dx" },
608
  4,  // { X86_REG_EAX, "eax" },
609
  4,  // { X86_REG_EBP, "ebp" },
610
  4,  // { X86_REG_EBX, "ebx" },
611
  4,  // { X86_REG_ECX, "ecx" },
612
  4,  // { X86_REG_EDI, "edi" },
613
  4,  // { X86_REG_EDX, "edx" },
614
  8,  // { X86_REG_EFLAGS, "flags" },
615
  4,  // { X86_REG_EIP, "eip" },
616
  4,  // { X86_REG_EIZ, "eiz" },
617
  2,  // { X86_REG_ES, "es" },
618
  4,  // { X86_REG_ESI, "esi" },
619
  4,  // { X86_REG_ESP, "esp" },
620
  10, // { X86_REG_FPSW, "fpsw" },
621
  2,  // { X86_REG_FS, "fs" },
622
  2,  // { X86_REG_GS, "gs" },
623
  2,  // { X86_REG_IP, "ip" },
624
  8,  // { X86_REG_RAX, "rax" },
625
  8,  // { X86_REG_RBP, "rbp" },
626
  8,  // { X86_REG_RBX, "rbx" },
627
  8,  // { X86_REG_RCX, "rcx" },
628
  8,  // { X86_REG_RDI, "rdi" },
629
  8,  // { X86_REG_RDX, "rdx" },
630
  8,  // { X86_REG_RIP, "rip" },
631
  8,  // { X86_REG_RIZ, "riz" },
632
  8,  // { X86_REG_RSI, "rsi" },
633
  8,  // { X86_REG_RSP, "rsp" },
634
  2,  // { X86_REG_SI, "si" },
635
  1,  // { X86_REG_SIL, "sil" },
636
  2,  // { X86_REG_SP, "sp" },
637
  1,  // { X86_REG_SPL, "spl" },
638
  2,  // { X86_REG_SS, "ss" },
639
  8,  // { X86_REG_CR0, "cr0" },
640
  8,  // { X86_REG_CR1, "cr1" },
641
  8,  // { X86_REG_CR2, "cr2" },
642
  8,  // { X86_REG_CR3, "cr3" },
643
  8,  // { X86_REG_CR4, "cr4" },
644
  8,  // { X86_REG_CR5, "cr5" },
645
  8,  // { X86_REG_CR6, "cr6" },
646
  8,  // { X86_REG_CR7, "cr7" },
647
  8,  // { X86_REG_CR8, "cr8" },
648
  8,  // { X86_REG_CR9, "cr9" },
649
  8,  // { X86_REG_CR10, "cr10" },
650
  8,  // { X86_REG_CR11, "cr11" },
651
  8,  // { X86_REG_CR12, "cr12" },
652
  8,  // { X86_REG_CR13, "cr13" },
653
  8,  // { X86_REG_CR14, "cr14" },
654
  8,  // { X86_REG_CR15, "cr15" },
655
  8,  // { X86_REG_DR0, "dr0" },
656
  8,  // { X86_REG_DR1, "dr1" },
657
  8,  // { X86_REG_DR2, "dr2" },
658
  8,  // { X86_REG_DR3, "dr3" },
659
  8,  // { X86_REG_DR4, "dr4" },
660
  8,  // { X86_REG_DR5, "dr5" },
661
  8,  // { X86_REG_DR6, "dr6" },
662
  8,  // { X86_REG_DR7, "dr7" },
663
  8,  // { X86_REG_DR8, "dr8" },
664
  8,  // { X86_REG_DR9, "dr9" },
665
  8,  // { X86_REG_DR10, "dr10" },
666
  8,  // { X86_REG_DR11, "dr11" },
667
  8,  // { X86_REG_DR12, "dr12" },
668
  8,  // { X86_REG_DR13, "dr13" },
669
  8,  // { X86_REG_DR14, "dr14" },
670
  8,  // { X86_REG_DR15, "dr15" },
671
  10, // { X86_REG_FP0, "fp0" },
672
  10, // { X86_REG_FP1, "fp1" },
673
  10, // { X86_REG_FP2, "fp2" },
674
  10, // { X86_REG_FP3, "fp3" },
675
  10, // { X86_REG_FP4, "fp4" },
676
  10, // { X86_REG_FP5, "fp5" },
677
  10, // { X86_REG_FP6, "fp6" },
678
  10, // { X86_REG_FP7, "fp7" },
679
  2,  // { X86_REG_K0, "k0" },
680
  2,  // { X86_REG_K1, "k1" },
681
  2,  // { X86_REG_K2, "k2" },
682
  2,  // { X86_REG_K3, "k3" },
683
  2,  // { X86_REG_K4, "k4" },
684
  2,  // { X86_REG_K5, "k5" },
685
  2,  // { X86_REG_K6, "k6" },
686
  2,  // { X86_REG_K7, "k7" },
687
  8,  // { X86_REG_MM0, "mm0" },
688
  8,  // { X86_REG_MM1, "mm1" },
689
  8,  // { X86_REG_MM2, "mm2" },
690
  8,  // { X86_REG_MM3, "mm3" },
691
  8,  // { X86_REG_MM4, "mm4" },
692
  8,  // { X86_REG_MM5, "mm5" },
693
  8,  // { X86_REG_MM6, "mm6" },
694
  8,  // { X86_REG_MM7, "mm7" },
695
  8,  // { X86_REG_R8, "r8" },
696
  8,  // { X86_REG_R9, "r9" },
697
  8,  // { X86_REG_R10, "r10" },
698
  8,  // { X86_REG_R11, "r11" },
699
  8,  // { X86_REG_R12, "r12" },
700
  8,  // { X86_REG_R13, "r13" },
701
  8,  // { X86_REG_R14, "r14" },
702
  8,  // { X86_REG_R15, "r15" },
703
  10, // { X86_REG_ST0, "st0" },
704
  10, // { X86_REG_ST1, "st1" },
705
  10, // { X86_REG_ST2, "st2" },
706
  10, // { X86_REG_ST3, "st3" },
707
  10, // { X86_REG_ST4, "st4" },
708
  10, // { X86_REG_ST5, "st5" },
709
  10, // { X86_REG_ST6, "st6" },
710
  10, // { X86_REG_ST7, "st7" },
711
  16, // { X86_REG_XMM0, "xmm0" },
712
  16, // { X86_REG_XMM1, "xmm1" },
713
  16, // { X86_REG_XMM2, "xmm2" },
714
  16, // { X86_REG_XMM3, "xmm3" },
715
  16, // { X86_REG_XMM4, "xmm4" },
716
  16, // { X86_REG_XMM5, "xmm5" },
717
  16, // { X86_REG_XMM6, "xmm6" },
718
  16, // { X86_REG_XMM7, "xmm7" },
719
  16, // { X86_REG_XMM8, "xmm8" },
720
  16, // { X86_REG_XMM9, "xmm9" },
721
  16, // { X86_REG_XMM10, "xmm10" },
722
  16, // { X86_REG_XMM11, "xmm11" },
723
  16, // { X86_REG_XMM12, "xmm12" },
724
  16, // { X86_REG_XMM13, "xmm13" },
725
  16, // { X86_REG_XMM14, "xmm14" },
726
  16, // { X86_REG_XMM15, "xmm15" },
727
  16, // { X86_REG_XMM16, "xmm16" },
728
  16, // { X86_REG_XMM17, "xmm17" },
729
  16, // { X86_REG_XMM18, "xmm18" },
730
  16, // { X86_REG_XMM19, "xmm19" },
731
  16, // { X86_REG_XMM20, "xmm20" },
732
  16, // { X86_REG_XMM21, "xmm21" },
733
  16, // { X86_REG_XMM22, "xmm22" },
734
  16, // { X86_REG_XMM23, "xmm23" },
735
  16, // { X86_REG_XMM24, "xmm24" },
736
  16, // { X86_REG_XMM25, "xmm25" },
737
  16, // { X86_REG_XMM26, "xmm26" },
738
  16, // { X86_REG_XMM27, "xmm27" },
739
  16, // { X86_REG_XMM28, "xmm28" },
740
  16, // { X86_REG_XMM29, "xmm29" },
741
  16, // { X86_REG_XMM30, "xmm30" },
742
  16, // { X86_REG_XMM31, "xmm31" },
743
  32, // { X86_REG_YMM0, "ymm0" },
744
  32, // { X86_REG_YMM1, "ymm1" },
745
  32, // { X86_REG_YMM2, "ymm2" },
746
  32, // { X86_REG_YMM3, "ymm3" },
747
  32, // { X86_REG_YMM4, "ymm4" },
748
  32, // { X86_REG_YMM5, "ymm5" },
749
  32, // { X86_REG_YMM6, "ymm6" },
750
  32, // { X86_REG_YMM7, "ymm7" },
751
  32, // { X86_REG_YMM8, "ymm8" },
752
  32, // { X86_REG_YMM9, "ymm9" },
753
  32, // { X86_REG_YMM10, "ymm10" },
754
  32, // { X86_REG_YMM11, "ymm11" },
755
  32, // { X86_REG_YMM12, "ymm12" },
756
  32, // { X86_REG_YMM13, "ymm13" },
757
  32, // { X86_REG_YMM14, "ymm14" },
758
  32, // { X86_REG_YMM15, "ymm15" },
759
  32, // { X86_REG_YMM16, "ymm16" },
760
  32, // { X86_REG_YMM17, "ymm17" },
761
  32, // { X86_REG_YMM18, "ymm18" },
762
  32, // { X86_REG_YMM19, "ymm19" },
763
  32, // { X86_REG_YMM20, "ymm20" },
764
  32, // { X86_REG_YMM21, "ymm21" },
765
  32, // { X86_REG_YMM22, "ymm22" },
766
  32, // { X86_REG_YMM23, "ymm23" },
767
  32, // { X86_REG_YMM24, "ymm24" },
768
  32, // { X86_REG_YMM25, "ymm25" },
769
  32, // { X86_REG_YMM26, "ymm26" },
770
  32, // { X86_REG_YMM27, "ymm27" },
771
  32, // { X86_REG_YMM28, "ymm28" },
772
  32, // { X86_REG_YMM29, "ymm29" },
773
  32, // { X86_REG_YMM30, "ymm30" },
774
  32, // { X86_REG_YMM31, "ymm31" },
775
  64, // { X86_REG_ZMM0, "zmm0" },
776
  64, // { X86_REG_ZMM1, "zmm1" },
777
  64, // { X86_REG_ZMM2, "zmm2" },
778
  64, // { X86_REG_ZMM3, "zmm3" },
779
  64, // { X86_REG_ZMM4, "zmm4" },
780
  64, // { X86_REG_ZMM5, "zmm5" },
781
  64, // { X86_REG_ZMM6, "zmm6" },
782
  64, // { X86_REG_ZMM7, "zmm7" },
783
  64, // { X86_REG_ZMM8, "zmm8" },
784
  64, // { X86_REG_ZMM9, "zmm9" },
785
  64, // { X86_REG_ZMM10, "zmm10" },
786
  64, // { X86_REG_ZMM11, "zmm11" },
787
  64, // { X86_REG_ZMM12, "zmm12" },
788
  64, // { X86_REG_ZMM13, "zmm13" },
789
  64, // { X86_REG_ZMM14, "zmm14" },
790
  64, // { X86_REG_ZMM15, "zmm15" },
791
  64, // { X86_REG_ZMM16, "zmm16" },
792
  64, // { X86_REG_ZMM17, "zmm17" },
793
  64, // { X86_REG_ZMM18, "zmm18" },
794
  64, // { X86_REG_ZMM19, "zmm19" },
795
  64, // { X86_REG_ZMM20, "zmm20" },
796
  64, // { X86_REG_ZMM21, "zmm21" },
797
  64, // { X86_REG_ZMM22, "zmm22" },
798
  64, // { X86_REG_ZMM23, "zmm23" },
799
  64, // { X86_REG_ZMM24, "zmm24" },
800
  64, // { X86_REG_ZMM25, "zmm25" },
801
  64, // { X86_REG_ZMM26, "zmm26" },
802
  64, // { X86_REG_ZMM27, "zmm27" },
803
  64, // { X86_REG_ZMM28, "zmm28" },
804
  64, // { X86_REG_ZMM29, "zmm29" },
805
  64, // { X86_REG_ZMM30, "zmm30" },
806
  64, // { X86_REG_ZMM31, "zmm31" },
807
  1,  // { X86_REG_R8B, "r8b" },
808
  1,  // { X86_REG_R9B, "r9b" },
809
  1,  // { X86_REG_R10B, "r10b" },
810
  1,  // { X86_REG_R11B, "r11b" },
811
  1,  // { X86_REG_R12B, "r12b" },
812
  1,  // { X86_REG_R13B, "r13b" },
813
  1,  // { X86_REG_R14B, "r14b" },
814
  1,  // { X86_REG_R15B, "r15b" },
815
  4,  // { X86_REG_R8D, "r8d" },
816
  4,  // { X86_REG_R9D, "r9d" },
817
  4,  // { X86_REG_R10D, "r10d" },
818
  4,  // { X86_REG_R11D, "r11d" },
819
  4,  // { X86_REG_R12D, "r12d" },
820
  4,  // { X86_REG_R13D, "r13d" },
821
  4,  // { X86_REG_R14D, "r14d" },
822
  4,  // { X86_REG_R15D, "r15d" },
823
  2,  // { X86_REG_R8W, "r8w" },
824
  2,  // { X86_REG_R9W, "r9w" },
825
  2,  // { X86_REG_R10W, "r10w" },
826
  2,  // { X86_REG_R11W, "r11w" },
827
  2,  // { X86_REG_R12W, "r12w" },
828
  2,  // { X86_REG_R13W, "r13w" },
829
  2,  // { X86_REG_R14W, "r14w" },
830
  2,  // { X86_REG_R15W, "r15w" },
831
  16, // { X86_REG_BND0, "bnd0" },
832
  16, // { X86_REG_BND1, "bnd0" },
833
  16, // { X86_REG_BND2, "bnd0" },
834
  16, // { X86_REG_BND3, "bnd0" },
835
};
836
837
const char *X86_reg_name(csh handle, unsigned int reg)
838
2.20M
{
839
2.20M
#ifndef CAPSTONE_DIET
840
2.20M
  cs_struct *ud = (cs_struct *)handle;
841
842
2.20M
  if (reg >= ARR_SIZE(reg_name_maps))
843
0
    return NULL;
844
845
2.20M
  if (reg == X86_REG_EFLAGS) {
846
993k
    if (ud->mode & CS_MODE_32)
847
324k
      return "eflags";
848
669k
    if (ud->mode & CS_MODE_64)
849
344k
      return "rflags";
850
669k
  }
851
852
1.53M
  return reg_name_maps[reg].name;
853
#else
854
  return NULL;
855
#endif
856
2.20M
}
857
858
#ifndef CAPSTONE_DIET
859
static const char * const insn_name_maps[] = {
860
  NULL, // X86_INS_INVALID
861
#ifndef CAPSTONE_X86_REDUCE
862
#include "X86MappingInsnName.inc"
863
#else
864
#include "X86MappingInsnName_reduce.inc"
865
#endif
866
};
867
#endif
868
869
// NOTE: insn_name_maps[] is sorted in order
870
const char *X86_insn_name(csh handle, unsigned int id)
871
1.53M
{
872
1.53M
#ifndef CAPSTONE_DIET
873
1.53M
  if (id >= ARR_SIZE(insn_name_maps))
874
0
    return NULL;
875
876
1.53M
  return insn_name_maps[id];
877
#else
878
  return NULL;
879
#endif
880
1.53M
}
881
882
#ifndef CAPSTONE_DIET
883
static const name_map group_name_maps[] = {
884
  // generic groups
885
  { X86_GRP_INVALID, NULL },
886
  { X86_GRP_JUMP, "jump" },
887
  { X86_GRP_CALL, "call" },
888
  { X86_GRP_RET, "ret" },
889
  { X86_GRP_INT, "int" },
890
  { X86_GRP_IRET, "iret" },
891
  { X86_GRP_PRIVILEGE, "privilege" },
892
  { X86_GRP_BRANCH_RELATIVE, "branch_relative" },
893
894
  // architecture-specific groups
895
  { X86_GRP_VM, "vm" },
896
  { X86_GRP_3DNOW, "3dnow" },
897
  { X86_GRP_AES, "aes" },
898
  { X86_GRP_ADX, "adx" },
899
  { X86_GRP_AVX, "avx" },
900
  { X86_GRP_AVX2, "avx2" },
901
  { X86_GRP_AVX512, "avx512" },
902
  { X86_GRP_BMI, "bmi" },
903
  { X86_GRP_BMI2, "bmi2" },
904
  { X86_GRP_CMOV, "cmov" },
905
  { X86_GRP_F16C, "fc16" },
906
  { X86_GRP_FMA, "fma" },
907
  { X86_GRP_FMA4, "fma4" },
908
  { X86_GRP_FSGSBASE, "fsgsbase" },
909
  { X86_GRP_HLE, "hle" },
910
  { X86_GRP_MMX, "mmx" },
911
  { X86_GRP_MODE32, "mode32" },
912
  { X86_GRP_MODE64, "mode64" },
913
  { X86_GRP_RTM, "rtm" },
914
  { X86_GRP_SHA, "sha" },
915
  { X86_GRP_SSE1, "sse1" },
916
  { X86_GRP_SSE2, "sse2" },
917
  { X86_GRP_SSE3, "sse3" },
918
  { X86_GRP_SSE41, "sse41" },
919
  { X86_GRP_SSE42, "sse42" },
920
  { X86_GRP_SSE4A, "sse4a" },
921
  { X86_GRP_SSSE3, "ssse3" },
922
  { X86_GRP_PCLMUL, "pclmul" },
923
  { X86_GRP_XOP, "xop" },
924
  { X86_GRP_CDI, "cdi" },
925
  { X86_GRP_ERI, "eri" },
926
  { X86_GRP_TBM, "tbm" },
927
  { X86_GRP_16BITMODE, "16bitmode" },
928
  { X86_GRP_NOT64BITMODE, "not64bitmode" },
929
  { X86_GRP_SGX,  "sgx" },
930
  { X86_GRP_DQI,  "dqi" },
931
  { X86_GRP_BWI,  "bwi" },
932
  { X86_GRP_PFI,  "pfi" },
933
  { X86_GRP_VLX,  "vlx" },
934
  { X86_GRP_SMAP, "smap" },
935
  { X86_GRP_NOVLX, "novlx" },
936
  { X86_GRP_FPU, "fpu" },
937
};
938
#endif
939
940
const char *X86_group_name(csh handle, unsigned int id)
941
713k
{
942
713k
#ifndef CAPSTONE_DIET
943
713k
  return id2name(group_name_maps, ARR_SIZE(group_name_maps), id);
944
#else
945
  return NULL;
946
#endif
947
713k
}
948
949
#define GET_INSTRINFO_ENUM
950
#ifdef CAPSTONE_X86_REDUCE
951
#include "X86GenInstrInfo_reduce.inc"
952
953
const insn_map_x86 insns[] = {  // reduce x86 instructions
954
#include "X86MappingInsn_reduce.inc"
955
};
956
#else
957
#include "X86GenInstrInfo.inc"
958
959
const insn_map_x86 insns[] = {  // full x86 instructions
960
#include "X86MappingInsn.inc"
961
};
962
#endif
963
964
#ifndef CAPSTONE_DIET
965
// in arr, replace r1 = r2
966
static void arr_replace(uint16_t *arr, uint8_t max, x86_reg r1, x86_reg r2)
967
313k
{
968
313k
  uint8_t i;
969
970
477k
  for(i = 0; i < max; i++) {
971
432k
    if (arr[i] == r1) {
972
268k
      arr[i] = r2;
973
268k
      break;
974
268k
    }
975
432k
  }
976
313k
}
977
#endif
978
979
// O(1) lookup with per-handle table. Used by functions that have a cs_struct *.
980
static inline unsigned int find_insn_h(cs_struct *h, unsigned int id)
981
5.35M
{
982
5.35M
  if (h && h->x86_insn_lut && id <= h->x86_insn_lut_max)
983
5.35M
    return (unsigned int)(int16_t)h->x86_insn_lut[id];
984
985
0
  return find_insn(id);
986
5.35M
}
987
988
// Fallback find_insn for callers without a cs_struct * (e.g., the decoder).
989
// Uses binary search since no per-handle table is available.
990
unsigned int find_insn(unsigned int id)
991
31.4k
{
992
31.4k
  unsigned int left, right, m;
993
31.4k
  unsigned int max = ARR_SIZE(insns);
994
995
31.4k
  right = max - 1;
996
997
31.4k
  if (id < insns[0].id || id > insns[right].id)
998
160
    return -1;
999
1000
31.2k
  left = 0;
1001
1002
402k
  while(left <= right) {
1003
402k
    m = (left + right) / 2;
1004
402k
    if (id == insns[m].id) {
1005
31.2k
      return m;
1006
31.2k
    }
1007
1008
371k
    if (id < insns[m].id)
1009
165k
      right = m - 1;
1010
205k
    else
1011
205k
      left = m + 1;
1012
371k
  }
1013
1014
0
  return -1;
1015
31.2k
}
1016
1017
// given internal insn id, return public instruction info
1018
void X86_get_insn_id(cs_struct *h, cs_insn *insn, unsigned int id)
1019
776k
{
1020
776k
  unsigned int i = find_insn_h(h, id);
1021
776k
  if (i != -1) {
1022
776k
    insn->id = insns[i].mapid;
1023
1024
776k
    if (h->detail) {
1025
776k
#ifndef CAPSTONE_DIET
1026
776k
      memcpy(insn->detail->regs_read, insns[i].regs_use, sizeof(insns[i].regs_use));
1027
776k
      insn->detail->regs_read_count = (uint8_t)count_positive(insns[i].regs_use);
1028
1029
      // special cases when regs_write[] depends on arch
1030
776k
      switch(id) {
1031
775k
        default:
1032
775k
          memcpy(insn->detail->regs_write, insns[i].regs_mod, sizeof(insns[i].regs_mod));
1033
775k
          insn->detail->regs_write_count = (uint8_t)count_positive(insns[i].regs_mod);
1034
775k
          break;
1035
657
        case X86_RDTSC:
1036
657
          if (h->mode == CS_MODE_64) {
1037
71
            memcpy(insn->detail->regs_write, insns[i].regs_mod, sizeof(insns[i].regs_mod));
1038
71
            insn->detail->regs_write_count = (uint8_t)count_positive(insns[i].regs_mod);
1039
586
          } else {
1040
586
            insn->detail->regs_write[0] = X86_REG_EAX;
1041
586
            insn->detail->regs_write[1] = X86_REG_EDX;
1042
586
            insn->detail->regs_write_count = 2;
1043
586
          }
1044
657
          break;
1045
497
        case X86_RDTSCP:
1046
497
          if (h->mode == CS_MODE_64) {
1047
250
            memcpy(insn->detail->regs_write, insns[i].regs_mod, sizeof(insns[i].regs_mod));
1048
250
            insn->detail->regs_write_count = (uint8_t)count_positive(insns[i].regs_mod);
1049
250
          } else {
1050
247
            insn->detail->regs_write[0] = X86_REG_EAX;
1051
247
            insn->detail->regs_write[1] = X86_REG_ECX;
1052
247
            insn->detail->regs_write[2] = X86_REG_EDX;
1053
247
            insn->detail->regs_write_count = 3;
1054
247
          }
1055
497
          break;
1056
776k
      }
1057
1058
776k
      switch(insn->id) {
1059
710k
        default:
1060
710k
          break;
1061
1062
710k
        case X86_INS_LOOP:
1063
4.13k
        case X86_INS_LOOPE:
1064
6.11k
        case X86_INS_LOOPNE:
1065
6.11k
          switch(h->mode) {
1066
0
            default: break;
1067
2.00k
            case CS_MODE_16:
1068
2.00k
                 insn->detail->regs_read[0] = X86_REG_CX;
1069
2.00k
                 insn->detail->regs_read_count = 1;
1070
2.00k
                 insn->detail->regs_write[0] = X86_REG_CX;
1071
2.00k
                 insn->detail->regs_write_count = 1;
1072
2.00k
                 break;
1073
1.27k
            case CS_MODE_32:
1074
1.27k
                 insn->detail->regs_read[0] = X86_REG_ECX;
1075
1.27k
                 insn->detail->regs_read_count = 1;
1076
1.27k
                 insn->detail->regs_write[0] = X86_REG_ECX;
1077
1.27k
                 insn->detail->regs_write_count = 1;
1078
1.27k
                 break;
1079
2.83k
            case CS_MODE_64:
1080
2.83k
                 insn->detail->regs_read[0] = X86_REG_RCX;
1081
2.83k
                 insn->detail->regs_read_count = 1;
1082
2.83k
                 insn->detail->regs_write[0] = X86_REG_RCX;
1083
2.83k
                 insn->detail->regs_write_count = 1;
1084
2.83k
                 break;
1085
6.11k
          }
1086
1087
          // LOOPE & LOOPNE also read EFLAGS
1088
6.11k
          if (insn->id != X86_INS_LOOP) {
1089
4.05k
            insn->detail->regs_read[1] = X86_REG_EFLAGS;
1090
4.05k
            insn->detail->regs_read_count = 2;
1091
4.05k
          }
1092
1093
6.11k
          break;
1094
1095
3.09k
        case X86_INS_LODSB:
1096
8.28k
        case X86_INS_LODSD:
1097
9.88k
        case X86_INS_LODSQ:
1098
11.8k
        case X86_INS_LODSW:
1099
11.8k
          switch(h->mode) {
1100
2.25k
            default:
1101
2.25k
              break;
1102
2.97k
            case CS_MODE_16:
1103
2.97k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_ESI, X86_REG_SI);
1104
2.97k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_ESI, X86_REG_SI);
1105
2.97k
              break;
1106
6.66k
            case CS_MODE_64:
1107
6.66k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_ESI, X86_REG_RSI);
1108
6.66k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_ESI, X86_REG_RSI);
1109
6.66k
              break;
1110
11.8k
          }
1111
11.8k
          break;
1112
1113
11.8k
        case X86_INS_SCASB:
1114
4.59k
        case X86_INS_SCASW:
1115
4.98k
        case X86_INS_SCASQ:
1116
10.7k
        case X86_INS_STOSB:
1117
12.9k
        case X86_INS_STOSD:
1118
13.8k
        case X86_INS_STOSQ:
1119
15.4k
        case X86_INS_STOSW:
1120
15.4k
          switch(h->mode) {
1121
2.66k
            default:
1122
2.66k
              break;
1123
7.91k
            case CS_MODE_16:
1124
7.91k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_EDI, X86_REG_DI);
1125
7.91k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_EDI, X86_REG_DI);
1126
7.91k
              break;
1127
4.88k
            case CS_MODE_64:
1128
4.88k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_EDI, X86_REG_RDI);
1129
4.88k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_EDI, X86_REG_RDI);
1130
4.88k
              break;
1131
15.4k
          }
1132
15.4k
          break;
1133
1134
15.4k
        case X86_INS_CMPSB:
1135
7.54k
        case X86_INS_CMPSD:
1136
8.31k
        case X86_INS_CMPSQ:
1137
9.89k
        case X86_INS_CMPSW:
1138
12.3k
        case X86_INS_MOVSB:
1139
13.9k
        case X86_INS_MOVSW:
1140
17.8k
        case X86_INS_MOVSD:
1141
18.5k
        case X86_INS_MOVSQ:
1142
18.5k
          switch(h->mode) {
1143
5.48k
            default:
1144
5.48k
              break;
1145
5.48k
            case CS_MODE_16:
1146
4.91k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_EDI, X86_REG_DI);
1147
4.91k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_EDI, X86_REG_DI);
1148
4.91k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_ESI, X86_REG_SI);
1149
4.91k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_ESI, X86_REG_SI);
1150
4.91k
              break;
1151
8.11k
            case CS_MODE_64:
1152
8.11k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_EDI, X86_REG_RDI);
1153
8.11k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_EDI, X86_REG_RDI);
1154
8.11k
              arr_replace(insn->detail->regs_read, insn->detail->regs_read_count, X86_REG_ESI, X86_REG_RSI);
1155
8.11k
              arr_replace(insn->detail->regs_write, insn->detail->regs_write_count, X86_REG_ESI, X86_REG_RSI);
1156
8.11k
              break;
1157
18.5k
          }
1158
18.5k
          break;
1159
1160
18.5k
        case X86_INS_RET:
1161
13.5k
          switch(h->mode) {
1162
3.98k
            case CS_MODE_16:
1163
3.98k
              insn->detail->regs_write[0] = X86_REG_SP;
1164
3.98k
              insn->detail->regs_read[0] = X86_REG_SP;
1165
3.98k
              break;
1166
4.24k
            case CS_MODE_32:
1167
4.24k
              insn->detail->regs_write[0] = X86_REG_ESP;
1168
4.24k
              insn->detail->regs_read[0] = X86_REG_ESP;
1169
4.24k
              break;
1170
5.30k
            default:  // 64-bit
1171
5.30k
              insn->detail->regs_write[0] = X86_REG_RSP;
1172
5.30k
              insn->detail->regs_read[0] = X86_REG_RSP;
1173
5.30k
              break;
1174
13.5k
          }
1175
13.5k
          insn->detail->regs_write_count = 1;
1176
13.5k
          insn->detail->regs_read_count = 1;
1177
13.5k
          break;
1178
776k
      }
1179
1180
776k
      memcpy(insn->detail->groups, insns[i].groups, sizeof(insns[i].groups));
1181
776k
      insn->detail->groups_count = (uint8_t)count_positive8(insns[i].groups);
1182
1183
776k
      if (insns[i].branch || insns[i].indirect_branch) {
1184
        // this insn also belongs to JUMP group. add JUMP group
1185
51.7k
        insn->detail->groups[insn->detail->groups_count] = X86_GRP_JUMP;
1186
51.7k
        insn->detail->groups_count++;
1187
51.7k
      }
1188
1189
776k
      switch (insns[i].id) {
1190
1.45k
        case X86_OUT8ir:
1191
2.43k
        case X86_OUT16ir:
1192
3.00k
        case X86_OUT32ir:
1193
3.00k
          if (insn->detail->x86.operands[0].imm == -78) {
1194
            // Writing to port 0xb2 causes an SMI on most platforms
1195
            // See: http://cs.gmu.edu/~tr-admin/papers/GMU-CS-TR-2011-8.pdf
1196
0
            insn->detail->groups[insn->detail->groups_count] = X86_GRP_INT;
1197
0
            insn->detail->groups_count++;
1198
0
          }
1199
3.00k
          break;
1200
1201
773k
        default:
1202
773k
          break;
1203
776k
      }
1204
776k
#endif
1205
776k
    }
1206
776k
  }
1207
776k
}
1208
1209
// map special instructions with accumulate registers.
1210
// this is needed because LLVM embeds these register names into AsmStrs[],
1211
// but not separately in operands
1212
struct insn_reg {
1213
  uint16_t insn;
1214
  x86_reg reg;
1215
  enum cs_ac_type access;
1216
};
1217
1218
struct insn_reg2 {
1219
  uint16_t insn;
1220
  x86_reg reg1, reg2;
1221
  enum cs_ac_type access1, access2;
1222
};
1223
1224
static inline uint16_t pack_insn_reg(x86_reg reg, enum cs_ac_type access)
1225
2.74M
{
1226
2.74M
  return (uint16_t)(((unsigned int)access << 12) |
1227
2.74M
    ((unsigned int)reg & 0x0fff));
1228
2.74M
}
1229
1230
static inline x86_reg unpack_insn_reg(uint16_t val, enum cs_ac_type *access)
1231
135k
{
1232
135k
  if (access)
1233
135k
    *access = (enum cs_ac_type)(val >> 12);
1234
135k
  return (x86_reg)(val & 0x0fff);
1235
135k
}
1236
1237
static const struct insn_reg insn_regs_att[] = {
1238
  { X86_INSB, X86_REG_DX, CS_AC_READ },
1239
  { X86_INSL, X86_REG_DX, CS_AC_READ },
1240
  { X86_INSW, X86_REG_DX, CS_AC_READ },
1241
  { X86_MOV16o16a, X86_REG_AX, CS_AC_READ },
1242
  { X86_MOV16o32a, X86_REG_AX, CS_AC_READ },
1243
  { X86_MOV16o64a, X86_REG_AX, CS_AC_READ },
1244
  { X86_MOV32o16a, X86_REG_EAX, CS_AC_READ },
1245
  { X86_MOV32o32a, X86_REG_EAX, CS_AC_READ },
1246
  { X86_MOV32o64a, X86_REG_EAX, CS_AC_READ },
1247
  { X86_MOV64o32a, X86_REG_RAX, CS_AC_READ },
1248
  { X86_MOV64o64a, X86_REG_RAX, CS_AC_READ },
1249
  { X86_MOV8o16a, X86_REG_AL, CS_AC_READ },
1250
  { X86_MOV8o32a, X86_REG_AL, CS_AC_READ },
1251
  { X86_MOV8o64a, X86_REG_AL, CS_AC_READ },
1252
  { X86_OUT16ir, X86_REG_AX, CS_AC_READ },
1253
  { X86_OUT32ir, X86_REG_EAX, CS_AC_READ },
1254
  { X86_OUT8ir, X86_REG_AL, CS_AC_READ },
1255
  { X86_POPDS16, X86_REG_DS, CS_AC_WRITE },
1256
  { X86_POPDS32, X86_REG_DS, CS_AC_WRITE },
1257
  { X86_POPES16, X86_REG_ES, CS_AC_WRITE },
1258
  { X86_POPES32, X86_REG_ES, CS_AC_WRITE },
1259
  { X86_POPFS16, X86_REG_FS, CS_AC_WRITE },
1260
  { X86_POPFS32, X86_REG_FS, CS_AC_WRITE },
1261
  { X86_POPFS64, X86_REG_FS, CS_AC_WRITE },
1262
  { X86_POPGS16, X86_REG_GS, CS_AC_WRITE },
1263
  { X86_POPGS32, X86_REG_GS, CS_AC_WRITE },
1264
  { X86_POPGS64, X86_REG_GS, CS_AC_WRITE },
1265
  { X86_POPSS16, X86_REG_SS, CS_AC_WRITE },
1266
  { X86_POPSS32, X86_REG_SS, CS_AC_WRITE },
1267
  { X86_PUSHCS16, X86_REG_CS, CS_AC_READ },
1268
  { X86_PUSHCS32, X86_REG_CS, CS_AC_READ },
1269
  { X86_PUSHDS16, X86_REG_DS, CS_AC_READ },
1270
  { X86_PUSHDS32, X86_REG_DS, CS_AC_READ },
1271
  { X86_PUSHES16, X86_REG_ES, CS_AC_READ },
1272
  { X86_PUSHES32, X86_REG_ES, CS_AC_READ },
1273
  { X86_PUSHFS16, X86_REG_FS, CS_AC_READ },
1274
  { X86_PUSHFS32, X86_REG_FS, CS_AC_READ },
1275
  { X86_PUSHFS64, X86_REG_FS, CS_AC_READ },
1276
  { X86_PUSHGS16, X86_REG_GS, CS_AC_READ },
1277
  { X86_PUSHGS32, X86_REG_GS, CS_AC_READ },
1278
  { X86_PUSHGS64, X86_REG_GS, CS_AC_READ },
1279
  { X86_PUSHSS16, X86_REG_SS, CS_AC_READ },
1280
  { X86_PUSHSS32, X86_REG_SS, CS_AC_READ },
1281
  { X86_RCL16rCL, X86_REG_CL, CS_AC_READ },
1282
  { X86_RCL32rCL, X86_REG_CL, CS_AC_READ },
1283
  { X86_RCL64rCL, X86_REG_CL, CS_AC_READ },
1284
  { X86_RCL8rCL, X86_REG_CL, CS_AC_READ },
1285
  { X86_RCR16rCL, X86_REG_CL, CS_AC_READ },
1286
  { X86_RCR32rCL, X86_REG_CL, CS_AC_READ },
1287
  { X86_RCR64rCL, X86_REG_CL, CS_AC_READ },
1288
  { X86_RCR8rCL, X86_REG_CL, CS_AC_READ },
1289
  { X86_ROL16rCL, X86_REG_CL, CS_AC_READ },
1290
  { X86_ROL32rCL, X86_REG_CL, CS_AC_READ },
1291
  { X86_ROL64rCL, X86_REG_CL, CS_AC_READ },
1292
  { X86_ROL8rCL, X86_REG_CL, CS_AC_READ },
1293
  { X86_ROR16rCL, X86_REG_CL, CS_AC_READ },
1294
  { X86_ROR32rCL, X86_REG_CL, CS_AC_READ },
1295
  { X86_ROR64rCL, X86_REG_CL, CS_AC_READ },
1296
  { X86_ROR8rCL, X86_REG_CL, CS_AC_READ },
1297
  { X86_SAL16rCL, X86_REG_CL, CS_AC_READ },
1298
  { X86_SAL32rCL, X86_REG_CL, CS_AC_READ },
1299
  { X86_SAL64rCL, X86_REG_CL, CS_AC_READ },
1300
  { X86_SAL8rCL, X86_REG_CL, CS_AC_READ },
1301
  { X86_SAR16rCL, X86_REG_CL, CS_AC_READ },
1302
  { X86_SAR32rCL, X86_REG_CL, CS_AC_READ },
1303
  { X86_SAR64rCL, X86_REG_CL, CS_AC_READ },
1304
  { X86_SAR8rCL, X86_REG_CL, CS_AC_READ },
1305
  { X86_SHL16rCL, X86_REG_CL, CS_AC_READ },
1306
  { X86_SHL32rCL, X86_REG_CL, CS_AC_READ },
1307
  { X86_SHL64rCL, X86_REG_CL, CS_AC_READ },
1308
  { X86_SHL8rCL, X86_REG_CL, CS_AC_READ },
1309
  { X86_SHLD16mrCL, X86_REG_CL, CS_AC_READ },
1310
  { X86_SHLD16rrCL, X86_REG_CL, CS_AC_READ },
1311
  { X86_SHLD32mrCL, X86_REG_CL, CS_AC_READ },
1312
  { X86_SHLD32rrCL, X86_REG_CL, CS_AC_READ },
1313
  { X86_SHLD64mrCL, X86_REG_CL, CS_AC_READ },
1314
  { X86_SHLD64rrCL, X86_REG_CL, CS_AC_READ },
1315
  { X86_SHR16rCL, X86_REG_CL, CS_AC_READ },
1316
  { X86_SHR32rCL, X86_REG_CL, CS_AC_READ },
1317
  { X86_SHR64rCL, X86_REG_CL, CS_AC_READ },
1318
  { X86_SHR8rCL, X86_REG_CL, CS_AC_READ },
1319
  { X86_SHRD16mrCL, X86_REG_CL, CS_AC_READ },
1320
  { X86_SHRD16rrCL, X86_REG_CL, CS_AC_READ },
1321
  { X86_SHRD32mrCL, X86_REG_CL, CS_AC_READ },
1322
  { X86_SHRD32rrCL, X86_REG_CL, CS_AC_READ },
1323
  { X86_SHRD64mrCL, X86_REG_CL, CS_AC_READ },
1324
  { X86_SHRD64rrCL, X86_REG_CL, CS_AC_READ },
1325
  { X86_XCHG16ar, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1326
  { X86_XCHG32ar, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1327
  { X86_XCHG64ar, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1328
};
1329
1330
static const struct insn_reg insn_regs_att_extra[] = {
1331
  // dummy entry, to avoid empty array
1332
  { 0, 0 },
1333
#ifndef CAPSTONE_X86_REDUCE
1334
  { X86_ADD_FrST0, X86_REG_ST0, CS_AC_READ },
1335
  { X86_DIVR_FrST0, X86_REG_ST0, CS_AC_READ },
1336
  { X86_DIV_FrST0, X86_REG_ST0, CS_AC_READ },
1337
  { X86_FNSTSW16r, X86_REG_AX, CS_AC_READ },
1338
  { X86_MUL_FrST0, X86_REG_ST0, CS_AC_READ },
1339
  { X86_SKINIT, X86_REG_EAX, CS_AC_READ },
1340
  { X86_SUBR_FrST0, X86_REG_ST0, CS_AC_READ },
1341
  { X86_SUB_FrST0, X86_REG_ST0, CS_AC_READ },
1342
  { X86_VMLOAD32, X86_REG_EAX, CS_AC_READ },
1343
  { X86_VMLOAD64, X86_REG_RAX, CS_AC_READ },
1344
  { X86_VMRUN32, X86_REG_EAX, CS_AC_READ },
1345
  { X86_VMRUN64, X86_REG_RAX, CS_AC_READ },
1346
  { X86_VMSAVE32, X86_REG_EAX, CS_AC_READ },
1347
  { X86_VMSAVE64, X86_REG_RAX, CS_AC_READ },
1348
#endif
1349
};
1350
1351
static const struct insn_reg insn_regs_intel[] = {
1352
  { X86_ADC16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1353
  { X86_ADC32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1354
  { X86_ADC64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1355
  { X86_ADC8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1356
  { X86_ADD16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1357
  { X86_ADD32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1358
  { X86_ADD64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1359
  { X86_ADD8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1360
  { X86_AND16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1361
  { X86_AND32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1362
  { X86_AND64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1363
  { X86_AND8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1364
  { X86_CMP16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1365
  { X86_CMP32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1366
  { X86_CMP64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1367
  { X86_CMP8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1368
  { X86_IN16ri, X86_REG_AX, CS_AC_WRITE },
1369
  { X86_IN32ri, X86_REG_EAX, CS_AC_WRITE },
1370
  { X86_IN8ri, X86_REG_AL, CS_AC_WRITE },
1371
  { X86_LODSB, X86_REG_AL, CS_AC_WRITE },
1372
  { X86_LODSL, X86_REG_EAX, CS_AC_WRITE },
1373
  { X86_LODSQ, X86_REG_RAX, CS_AC_WRITE },
1374
  { X86_LODSW, X86_REG_AX, CS_AC_WRITE },
1375
  { X86_MOV16ao16, X86_REG_AX, CS_AC_WRITE },    // 16-bit A1 1020                  // mov     ax, word ptr [0x2010]
1376
  { X86_MOV16ao32, X86_REG_AX, CS_AC_WRITE },    // 32-bit A1 10203040              // mov     ax, word ptr [0x40302010]
1377
  { X86_MOV16ao64, X86_REG_AX, CS_AC_WRITE },    // 64-bit 66 A1 1020304050607080   // movabs  ax, word ptr [0x8070605040302010]
1378
  { X86_MOV32ao16, X86_REG_EAX, CS_AC_WRITE },   // 32-bit 67 A1 1020               // mov     eax, dword ptr [0x2010]
1379
  { X86_MOV32ao32, X86_REG_EAX, CS_AC_WRITE },   // 32-bit A1 10203040              // mov     eax, dword ptr [0x40302010]
1380
  { X86_MOV32ao64, X86_REG_EAX, CS_AC_WRITE },   // 64-bit A1 1020304050607080      // movabs  eax, dword ptr [0x8070605040302010]
1381
  { X86_MOV64ao32, X86_REG_RAX, CS_AC_WRITE },   // 64-bit 48 8B04 10203040         // mov     rax, qword ptr [0x40302010]
1382
  { X86_MOV64ao64, X86_REG_RAX, CS_AC_WRITE },   // 64-bit 48 A1 1020304050607080   // movabs  rax, qword ptr [0x8070605040302010]
1383
  { X86_MOV8ao16, X86_REG_AL, CS_AC_WRITE },     // 16-bit A0 1020                  // mov     al, byte ptr [0x2010]
1384
  { X86_MOV8ao32, X86_REG_AL, CS_AC_WRITE },     // 32-bit A0 10203040              // mov     al, byte ptr [0x40302010]
1385
  { X86_MOV8ao64, X86_REG_AL, CS_AC_WRITE },     // 64-bit 66 A0 1020304050607080   // movabs  al, byte ptr [0x8070605040302010]
1386
  { X86_OR16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1387
  { X86_OR32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1388
  { X86_OR64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1389
  { X86_OR8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1390
  { X86_OUTSB, X86_REG_DX, CS_AC_WRITE },
1391
  { X86_OUTSL, X86_REG_DX, CS_AC_WRITE },
1392
  { X86_OUTSW, X86_REG_DX, CS_AC_WRITE },
1393
  { X86_POPDS16, X86_REG_DS, CS_AC_WRITE },
1394
  { X86_POPDS32, X86_REG_DS, CS_AC_WRITE },
1395
  { X86_POPES16, X86_REG_ES, CS_AC_WRITE },
1396
  { X86_POPES32, X86_REG_ES, CS_AC_WRITE },
1397
  { X86_POPFS16, X86_REG_FS, CS_AC_WRITE },
1398
  { X86_POPFS32, X86_REG_FS, CS_AC_WRITE },
1399
  { X86_POPFS64, X86_REG_FS, CS_AC_WRITE },
1400
  { X86_POPGS16, X86_REG_GS, CS_AC_WRITE },
1401
  { X86_POPGS32, X86_REG_GS, CS_AC_WRITE },
1402
  { X86_POPGS64, X86_REG_GS, CS_AC_WRITE },
1403
  { X86_POPSS16, X86_REG_SS, CS_AC_WRITE },
1404
  { X86_POPSS32, X86_REG_SS, CS_AC_WRITE },
1405
  { X86_PUSHCS16, X86_REG_CS, CS_AC_READ },
1406
  { X86_PUSHCS32, X86_REG_CS, CS_AC_READ },
1407
  { X86_PUSHDS16, X86_REG_DS, CS_AC_READ },
1408
  { X86_PUSHDS32, X86_REG_DS, CS_AC_READ },
1409
  { X86_PUSHES16, X86_REG_ES, CS_AC_READ },
1410
  { X86_PUSHES32, X86_REG_ES, CS_AC_READ },
1411
  { X86_PUSHFS16, X86_REG_FS, CS_AC_READ },
1412
  { X86_PUSHFS32, X86_REG_FS, CS_AC_READ },
1413
  { X86_PUSHFS64, X86_REG_FS, CS_AC_READ },
1414
  { X86_PUSHGS16, X86_REG_GS, CS_AC_READ },
1415
  { X86_PUSHGS32, X86_REG_GS, CS_AC_READ },
1416
  { X86_PUSHGS64, X86_REG_GS, CS_AC_READ },
1417
  { X86_PUSHSS16, X86_REG_SS, CS_AC_READ },
1418
  { X86_PUSHSS32, X86_REG_SS, CS_AC_READ },
1419
  { X86_SBB16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1420
  { X86_SBB32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1421
  { X86_SBB64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1422
  { X86_SBB8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1423
  { X86_SCASB, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1424
  { X86_SCASL, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1425
  { X86_SCASQ, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1426
  { X86_SCASW, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1427
  { X86_SUB16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1428
  { X86_SUB32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1429
  { X86_SUB64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1430
  { X86_SUB8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1431
  { X86_TEST16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1432
  { X86_TEST32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1433
  { X86_TEST64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1434
  { X86_TEST8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1435
  { X86_XOR16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1436
  { X86_XOR32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1437
  { X86_XOR64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1438
  { X86_XOR8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1439
};
1440
1441
static const struct insn_reg insn_regs_intel_extra[] = {
1442
  // dummy entry, to avoid empty array
1443
  { 0, 0, 0 },
1444
#ifndef CAPSTONE_X86_REDUCE
1445
  { X86_CMOVBE_F, X86_REG_ST0, CS_AC_WRITE },
1446
  { X86_CMOVB_F, X86_REG_ST0, CS_AC_WRITE },
1447
  { X86_CMOVE_F, X86_REG_ST0, CS_AC_WRITE },
1448
  { X86_CMOVNBE_F, X86_REG_ST0, CS_AC_WRITE },
1449
  { X86_CMOVNB_F, X86_REG_ST0, CS_AC_WRITE },
1450
  { X86_CMOVNE_F, X86_REG_ST0, CS_AC_WRITE },
1451
  { X86_CMOVNP_F, X86_REG_ST0, CS_AC_WRITE },
1452
  { X86_CMOVP_F, X86_REG_ST0, CS_AC_WRITE },
1453
  // { X86_COMP_FST0r, X86_REG_ST0, CS_AC_WRITE },
1454
  // { X86_COM_FST0r, X86_REG_ST0, CS_AC_WRITE },
1455
  { X86_FNSTSW16r, X86_REG_AX, CS_AC_WRITE },
1456
  { X86_SKINIT, X86_REG_EAX, CS_AC_WRITE },
1457
  { X86_VMLOAD32, X86_REG_EAX, CS_AC_WRITE },
1458
  { X86_VMLOAD64, X86_REG_RAX, CS_AC_WRITE },
1459
  { X86_VMRUN32, X86_REG_EAX, CS_AC_WRITE },
1460
  { X86_VMRUN64, X86_REG_RAX, CS_AC_WRITE },
1461
  { X86_VMSAVE32, X86_REG_EAX, CS_AC_READ },
1462
  { X86_VMSAVE64, X86_REG_RAX, CS_AC_READ },
1463
  { X86_XCH_F, X86_REG_ST0, CS_AC_WRITE },
1464
#endif
1465
};
1466
1467
static const struct insn_reg2 insn_regs_intel2[] = {
1468
  { X86_IN16rr, X86_REG_AX, X86_REG_DX, CS_AC_WRITE, CS_AC_READ },
1469
  { X86_IN32rr, X86_REG_EAX, X86_REG_DX, CS_AC_WRITE, CS_AC_READ },
1470
  { X86_IN8rr, X86_REG_AL, X86_REG_DX, CS_AC_WRITE, CS_AC_READ },
1471
  { X86_INVLPGA32, X86_REG_EAX, X86_REG_ECX, CS_AC_READ, CS_AC_READ },
1472
  { X86_INVLPGA64, X86_REG_RAX, X86_REG_ECX, CS_AC_READ, CS_AC_READ },
1473
  { X86_OUT16rr, X86_REG_DX, X86_REG_AX, CS_AC_READ, CS_AC_READ },
1474
  { X86_OUT32rr, X86_REG_DX, X86_REG_EAX, CS_AC_READ, CS_AC_READ },
1475
  { X86_OUT8rr, X86_REG_DX, X86_REG_AL, CS_AC_READ, CS_AC_READ },
1476
};
1477
1478
static int binary_search1(const struct insn_reg *insns, unsigned int max, unsigned int id)
1479
0
{
1480
0
  unsigned int first, last, mid;
1481
1482
0
  first = 0;
1483
0
  last = max -1;
1484
1485
0
  if (insns[0].insn > id || insns[last].insn < id) {
1486
    // not found
1487
0
    return -1;
1488
0
  }
1489
1490
0
  while (first <= last) {
1491
0
    mid = (first + last) / 2;
1492
0
    if (insns[mid].insn < id) {
1493
0
      first = mid + 1;
1494
0
    } else if (insns[mid].insn == id) {
1495
0
      return mid;
1496
0
    } else {
1497
0
      if (mid == 0)
1498
0
        break;
1499
0
      last = mid - 1;
1500
0
    }
1501
0
  }
1502
1503
  // not found
1504
0
  return -1;
1505
0
}
1506
1507
static int binary_search2(const struct insn_reg2 *insns, unsigned int max, unsigned int id)
1508
1.39M
{
1509
1.39M
  unsigned int first, last, mid;
1510
1511
1.39M
  first = 0;
1512
1.39M
  last = max -1;
1513
1514
1.39M
  if (insns[0].insn > id || insns[last].insn < id) {
1515
    // not found
1516
1.03M
    return -1;
1517
1.03M
  }
1518
1519
1.43M
  while (first <= last) {
1520
1.09M
    mid = (first + last) / 2;
1521
1.09M
    if (insns[mid].insn < id) {
1522
709k
      first = mid + 1;
1523
709k
    } else if (insns[mid].insn == id) {
1524
28.7k
      return mid;
1525
359k
    } else {
1526
359k
      if (mid == 0)
1527
0
        break;
1528
359k
      last = mid - 1;
1529
359k
    }
1530
1.09M
  }
1531
1532
  // not found
1533
337k
  return -1;
1534
366k
}
1535
1536
// Build per-handle O(1) lookup tables for instruction mapping.
1537
// Called from X86_global_init() during cs_open(). Each handle gets its own
1538
// copy of the lookup tables, making this thread-safe.
1539
void X86_build_lookup_tables(cs_struct *h)
1540
6.98k
{
1541
6.98k
  unsigned int i;
1542
6.98k
  unsigned int max = ARR_SIZE(insns);
1543
6.98k
  unsigned int id_max;
1544
1545
6.98k
  if (h->x86_insn_lut)
1546
0
    return;
1547
1548
6.98k
  id_max = insns[max - 1].id;
1549
6.98k
  h->x86_insn_lut_max = id_max;
1550
1551
6.98k
  h->x86_insn_lut = (uint16_t *)cs_mem_malloc((id_max + 1) * sizeof(uint16_t));
1552
6.98k
  CS_ASSERT_RET(h->x86_insn_lut);
1553
1554
6.98k
  memset(h->x86_insn_lut, 0xff, (id_max + 1) * sizeof(uint16_t));
1555
1556
105M
  for (i = 0; i < max; i++) {
1557
105M
    h->x86_insn_lut[insns[i].id] = (uint16_t)i;
1558
105M
  }
1559
1560
  // Build insn_reg lookup table (low 16 bits = Intel, high 16 bits = ATT).
1561
6.98k
  h->x86_insn_reg_lut = (uint32_t *)cs_mem_calloc(id_max + 1, sizeof(uint32_t));
1562
6.98k
  CS_ASSERT_RET(h->x86_insn_reg_lut);
1563
1564
614k
  for (i = 0; i < ARR_SIZE(insn_regs_intel); i++) {
1565
607k
    unsigned int insn_id = insn_regs_intel[i].insn;
1566
607k
    if (insn_id <= id_max)
1567
607k
      h->x86_insn_reg_lut[insn_id] =
1568
607k
        (h->x86_insn_reg_lut[insn_id] & 0xffff0000) |
1569
607k
        pack_insn_reg(insn_regs_intel[i].reg, insn_regs_intel[i].access);
1570
607k
  }
1571
132k
  for (i = 0; i < ARR_SIZE(insn_regs_intel_extra); i++) {
1572
125k
    unsigned int insn_id = insn_regs_intel_extra[i].insn;
1573
125k
    if (insn_id && insn_id <= id_max &&
1574
118k
        !(h->x86_insn_reg_lut[insn_id] & 0xffff))
1575
118k
      h->x86_insn_reg_lut[insn_id] =
1576
118k
        (h->x86_insn_reg_lut[insn_id] & 0xffff0000) |
1577
118k
        pack_insn_reg(insn_regs_intel_extra[i].reg, insn_regs_intel_extra[i].access);
1578
125k
  }
1579
1580
635k
  for (i = 0; i < ARR_SIZE(insn_regs_att); i++) {
1581
628k
    unsigned int insn_id = insn_regs_att[i].insn;
1582
628k
    if (insn_id <= id_max)
1583
628k
      h->x86_insn_reg_lut[insn_id] =
1584
628k
        (h->x86_insn_reg_lut[insn_id] & 0x0000ffff) |
1585
628k
        ((uint32_t)pack_insn_reg(insn_regs_att[i].reg, insn_regs_att[i].access) << 16);
1586
628k
  }
1587
111k
  for (i = 0; i < ARR_SIZE(insn_regs_att_extra); i++) {
1588
104k
    unsigned int insn_id = insn_regs_att_extra[i].insn;
1589
104k
    if (insn_id && insn_id <= id_max &&
1590
97.8k
        !(h->x86_insn_reg_lut[insn_id] >> 16))
1591
97.8k
      h->x86_insn_reg_lut[insn_id] =
1592
97.8k
        (h->x86_insn_reg_lut[insn_id] & 0x0000ffff) |
1593
97.8k
        ((uint32_t)pack_insn_reg(insn_regs_att_extra[i].reg, insn_regs_att_extra[i].access) << 16);
1594
104k
  }
1595
6.98k
}
1596
1597
// return register of given instruction id
1598
// return 0 if not found
1599
// this is to handle instructions embedding accumulate registers into AsmStrs[]
1600
// Falls back to binary search when per-handle LUT is not available.
1601
x86_reg X86_insn_reg_intel(unsigned int id, enum cs_ac_type *access)
1602
0
{
1603
  // The per-handle LUT is used via X86_insn_reg_intel_h() from the printer.
1604
  // This fallback uses binary search for safety.
1605
0
  int i;
1606
1607
0
  i = binary_search1(insn_regs_intel, ARR_SIZE(insn_regs_intel), id);
1608
0
  if (i != -1) {
1609
0
    if (access) {
1610
0
      *access = insn_regs_intel[i].access;
1611
0
    }
1612
0
    return insn_regs_intel[i].reg;
1613
0
  }
1614
1615
0
  i = binary_search1(insn_regs_intel_extra, ARR_SIZE(insn_regs_intel_extra), id);
1616
0
  if (i != -1) {
1617
0
    if (access) {
1618
0
      *access = insn_regs_intel_extra[i].access;
1619
0
    }
1620
0
    return insn_regs_intel_extra[i].reg;
1621
0
  }
1622
1623
0
  return 0;
1624
0
}
1625
1626
// Fast per-handle variant used from the printer (which has access to cs_struct via MCInst).
1627
x86_reg X86_insn_reg_intel_h(cs_struct *h, unsigned int id, enum cs_ac_type *access)
1628
782k
{
1629
782k
  if (h && h->x86_insn_reg_lut && id <= h->x86_insn_lut_max) {
1630
782k
    uint16_t val = (uint16_t)(h->x86_insn_reg_lut[id] & 0xffff);
1631
782k
    if (val)
1632
88.9k
      return unpack_insn_reg(val, access);
1633
693k
    return 0;
1634
782k
  }
1635
0
  return X86_insn_reg_intel(id, access);
1636
782k
}
1637
1638
bool X86_insn_reg_intel2(unsigned int id, x86_reg *reg1, enum cs_ac_type *access1, x86_reg *reg2, enum cs_ac_type *access2)
1639
693k
{
1640
693k
  int i = binary_search2(insn_regs_intel2, ARR_SIZE(insn_regs_intel2), id);
1641
693k
  if (i != -1) {
1642
13.4k
    *reg1 = insn_regs_intel2[i].reg1;
1643
13.4k
    *reg2 = insn_regs_intel2[i].reg2;
1644
13.4k
    if (access1)
1645
13.4k
      *access1 = insn_regs_intel2[i].access1;
1646
13.4k
    if (access2)
1647
13.4k
      *access2 = insn_regs_intel2[i].access2;
1648
13.4k
    return true;
1649
13.4k
  }
1650
1651
  // not found
1652
679k
  return false;
1653
693k
}
1654
1655
x86_reg X86_insn_reg_att(unsigned int id, enum cs_ac_type *access)
1656
0
{
1657
0
  int i;
1658
1659
0
  i = binary_search1(insn_regs_att, ARR_SIZE(insn_regs_att), id);
1660
0
  if (i != -1) {
1661
0
    if (access)
1662
0
      *access = insn_regs_att[i].access;
1663
0
    return insn_regs_att[i].reg;
1664
0
  }
1665
1666
0
  i = binary_search1(insn_regs_att_extra, ARR_SIZE(insn_regs_att_extra), id);
1667
0
  if (i != -1) {
1668
0
    if (access)
1669
0
      *access = insn_regs_att_extra[i].access;
1670
0
    return insn_regs_att_extra[i].reg;
1671
0
  }
1672
1673
0
  return 0;
1674
0
}
1675
1676
x86_reg X86_insn_reg_att_h(cs_struct *h, unsigned int id, enum cs_ac_type *access)
1677
751k
{
1678
751k
  if (h && h->x86_insn_reg_lut && id <= h->x86_insn_lut_max) {
1679
751k
    uint16_t val = (uint16_t)(h->x86_insn_reg_lut[id] >> 16);
1680
751k
    if (val)
1681
46.0k
      return unpack_insn_reg(val, access);
1682
705k
    return 0;
1683
751k
  }
1684
0
  return X86_insn_reg_att(id, access);
1685
751k
}
1686
1687
// ATT just reuses Intel data, but with the order of registers reversed
1688
bool X86_insn_reg_att2(unsigned int id, x86_reg *reg1, enum cs_ac_type *access1, x86_reg *reg2, enum cs_ac_type *access2)
1689
705k
{
1690
705k
  int i = binary_search2(insn_regs_intel2, ARR_SIZE(insn_regs_intel2), id);
1691
705k
  if (i != -1) {
1692
15.2k
    *reg1 = insn_regs_intel2[i].reg2;
1693
15.2k
    *reg2 = insn_regs_intel2[i].reg1;
1694
15.2k
    if (access1)
1695
15.2k
      *access1 = insn_regs_intel2[i].access2;
1696
15.2k
    if (access2)
1697
15.2k
      *access2 = insn_regs_intel2[i].access1;
1698
15.2k
    return true;
1699
15.2k
  }
1700
1701
  // not found
1702
690k
  return false;
1703
705k
}
1704
1705
// given MCInst's id, find out if this insn is valid for REPNE prefix
1706
static bool valid_repne(cs_struct *h, unsigned int opcode)
1707
24.2k
{
1708
24.2k
  unsigned int id;
1709
24.2k
  unsigned int i = find_insn_h(h, opcode);
1710
24.2k
  if (i != -1) {
1711
24.2k
    id = insns[i].mapid;
1712
24.2k
    switch(id) {
1713
16.7k
      default:
1714
16.7k
        return false;
1715
1716
238
      case X86_INS_CMPSB:
1717
381
      case X86_INS_CMPSS:
1718
455
      case X86_INS_CMPSW:
1719
548
      case X86_INS_CMPSQ:
1720
1721
924
      case X86_INS_SCASB:
1722
1.12k
      case X86_INS_SCASW:
1723
1.16k
      case X86_INS_SCASQ:
1724
1725
1.37k
      case X86_INS_MOVSB:
1726
1.40k
      case X86_INS_MOVSS:
1727
1.47k
      case X86_INS_MOVSW:
1728
1.74k
      case X86_INS_MOVSQ:
1729
1730
1.87k
      case X86_INS_LODSB:
1731
2.48k
      case X86_INS_LODSW:
1732
2.56k
      case X86_INS_LODSD:
1733
2.91k
      case X86_INS_LODSQ:
1734
1735
3.09k
      case X86_INS_STOSB:
1736
3.23k
      case X86_INS_STOSW:
1737
3.54k
      case X86_INS_STOSD:
1738
3.92k
      case X86_INS_STOSQ:
1739
1740
4.09k
      case X86_INS_INSB:
1741
4.32k
      case X86_INS_INSW:
1742
4.58k
      case X86_INS_INSD:
1743
1744
5.12k
      case X86_INS_OUTSB:
1745
5.42k
      case X86_INS_OUTSW:
1746
5.64k
      case X86_INS_OUTSD:
1747
1748
5.64k
        return true;
1749
1750
808
      case X86_INS_MOVSD:
1751
808
        if (opcode == X86_MOVSW) // REP MOVSB
1752
0
          return true;
1753
808
        return false;
1754
1755
711
      case X86_INS_CMPSD:
1756
711
        if (opcode == X86_CMPSL) // REP CMPSD
1757
110
          return true;
1758
601
        return false;
1759
1760
317
      case X86_INS_SCASD:
1761
317
        if (opcode == X86_SCASL) // REP SCASD
1762
317
          return true;
1763
0
        return false;
1764
24.2k
    }
1765
24.2k
  }
1766
1767
  // not found
1768
0
  return false;
1769
24.2k
}
1770
1771
// given MCInst's id, find out if this insn is valid for BND prefix
1772
// BND prefix is valid for CALL/JMP/RET
1773
#ifndef CAPSTONE_DIET
1774
static bool valid_bnd(cs_struct *h, unsigned int opcode)
1775
31.8k
{
1776
31.8k
  unsigned int id;
1777
31.8k
  unsigned int i = find_insn_h(h, opcode);
1778
31.8k
  if (i != -1) {
1779
31.8k
    id = insns[i].mapid;
1780
31.8k
    switch(id) {
1781
21.7k
      default:
1782
21.7k
        return false;
1783
1784
308
      case X86_INS_JAE:
1785
647
      case X86_INS_JA:
1786
1.06k
      case X86_INS_JBE:
1787
2.70k
      case X86_INS_JB:
1788
3.32k
      case X86_INS_JCXZ:
1789
3.63k
      case X86_INS_JECXZ:
1790
4.05k
      case X86_INS_JE:
1791
4.52k
      case X86_INS_JGE:
1792
4.83k
      case X86_INS_JG:
1793
5.11k
      case X86_INS_JLE:
1794
5.35k
      case X86_INS_JL:
1795
5.61k
      case X86_INS_JMP:
1796
6.52k
      case X86_INS_JNE:
1797
6.89k
      case X86_INS_JNO:
1798
7.27k
      case X86_INS_JNP:
1799
7.46k
      case X86_INS_JNS:
1800
7.66k
      case X86_INS_JO:
1801
8.08k
      case X86_INS_JP:
1802
8.30k
      case X86_INS_JRCXZ:
1803
8.83k
      case X86_INS_JS:
1804
1805
9.05k
      case X86_INS_CALL:
1806
9.60k
      case X86_INS_RET:
1807
9.83k
      case X86_INS_RETF:
1808
10.1k
      case X86_INS_RETFQ:
1809
10.1k
        return true;
1810
31.8k
    }
1811
31.8k
  }
1812
1813
  // not found
1814
0
  return false;
1815
31.8k
}
1816
#endif
1817
1818
// return true if the opcode is XCHG [mem]
1819
static bool xchg_mem(unsigned int opcode)
1820
46.8k
{
1821
46.8k
  switch(opcode) {
1822
45.6k
    default:
1823
45.6k
      return false;
1824
534
    case X86_XCHG8rm:
1825
704
    case X86_XCHG16rm:
1826
1.06k
    case X86_XCHG32rm:
1827
1.23k
    case X86_XCHG64rm:
1828
1.23k
         return true;
1829
46.8k
  }
1830
46.8k
}
1831
1832
// given MCInst's id, find out if this insn is valid for REP prefix
1833
static bool valid_rep(cs_struct *h, unsigned int opcode)
1834
38.4k
{
1835
38.4k
  unsigned int id;
1836
38.4k
  unsigned int i = find_insn_h(h, opcode);
1837
38.4k
  if (i != -1) {
1838
38.4k
    id = insns[i].mapid;
1839
38.4k
    switch(id) {
1840
28.7k
      default:
1841
28.7k
        return false;
1842
1843
312
      case X86_INS_MOVSB:
1844
577
      case X86_INS_MOVSW:
1845
1.14k
      case X86_INS_MOVSQ:
1846
1847
1.68k
      case X86_INS_LODSB:
1848
1.83k
      case X86_INS_LODSW:
1849
2.63k
      case X86_INS_LODSQ:
1850
1851
3.25k
      case X86_INS_STOSB:
1852
3.58k
      case X86_INS_STOSW:
1853
4.05k
      case X86_INS_STOSQ:
1854
1855
4.36k
      case X86_INS_INSB:
1856
4.70k
      case X86_INS_INSW:
1857
5.18k
      case X86_INS_INSD:
1858
1859
5.82k
      case X86_INS_OUTSB:
1860
6.06k
      case X86_INS_OUTSW:
1861
6.53k
      case X86_INS_OUTSD:
1862
6.53k
        return true;
1863
1864
      // following are some confused instructions, which have the same
1865
      // mnemonics in 128bit media instructions. Intel is horribly crazy!
1866
1.21k
      case X86_INS_MOVSD:
1867
1.21k
        if (opcode == X86_MOVSL) // REP MOVSD
1868
997
          return true;
1869
213
        return false;
1870
1871
936
      case X86_INS_LODSD:
1872
936
        if (opcode == X86_LODSL) // REP LODSD
1873
936
          return true;
1874
0
        return false;
1875
1876
953
      case X86_INS_STOSD:
1877
953
        if (opcode == X86_STOSL) // REP STOSD
1878
953
          return true;
1879
0
        return false;
1880
38.4k
    }
1881
38.4k
  }
1882
1883
  // not found
1884
0
  return false;
1885
38.4k
}
1886
1887
// given MCInst's id, find if this is a "repz ret" instruction
1888
// gcc generates "repz ret" (f3 c3) instructions in some cases as an
1889
// optimization for AMD platforms, see:
1890
// https://gcc.gnu.org/legacy-ml/gcc-patches/2003-05/msg02117.html
1891
static bool valid_ret_repz(cs_struct *h, unsigned int opcode)
1892
24.5k
{
1893
24.5k
  unsigned int id;
1894
24.5k
  unsigned int i = find_insn_h(h, opcode);
1895
1896
24.5k
  if (i != -1) {
1897
24.5k
    id = insns[i].mapid;
1898
24.5k
    return id == X86_INS_RET;
1899
24.5k
  }
1900
1901
  // not found
1902
0
  return false;
1903
24.5k
}
1904
1905
// given MCInst's id, find out if this insn is valid for REPE prefix
1906
static bool valid_repe(cs_struct *h, unsigned int opcode)
1907
29.0k
{
1908
29.0k
  unsigned int id;
1909
29.0k
  unsigned int i = find_insn_h(h, opcode);
1910
29.0k
  if (i != -1) {
1911
29.0k
    id = insns[i].mapid;
1912
29.0k
    switch(id) {
1913
24.3k
      default:
1914
24.3k
        return false;
1915
1916
335
      case X86_INS_CMPSB:
1917
862
      case X86_INS_CMPSW:
1918
1.14k
      case X86_INS_CMPSQ:
1919
1920
1.65k
      case X86_INS_SCASB:
1921
2.07k
      case X86_INS_SCASW:
1922
2.91k
      case X86_INS_SCASQ:
1923
2.91k
        return true;
1924
1925
      // following are some confused instructions, which have the same
1926
      // mnemonics in 128bit media instructions. Intel is horribly crazy!
1927
1.19k
      case X86_INS_CMPSD:
1928
1.19k
        if (opcode == X86_CMPSL) // REP CMPSD
1929
994
          return true;
1930
201
        return false;
1931
1932
522
      case X86_INS_SCASD:
1933
522
        if (opcode == X86_SCASL) // REP SCASD
1934
522
          return true;
1935
0
        return false;
1936
29.0k
    }
1937
29.0k
  }
1938
1939
  // not found
1940
0
  return false;
1941
29.0k
}
1942
1943
// Given MCInst's id, find out if this insn is valid for NOTRACK prefix.
1944
// NOTRACK prefix is valid for CALL/JMP.
1945
static bool valid_notrack(cs_struct *h, unsigned int opcode)
1946
5.15k
{
1947
5.15k
  unsigned int id;
1948
5.15k
  unsigned int i = find_insn_h(h, opcode);
1949
5.15k
  if (i != -1) {
1950
5.15k
    id = insns[i].mapid;
1951
5.15k
    switch(id) {
1952
4.77k
      default:
1953
4.77k
        return false;
1954
146
      case X86_INS_CALL:
1955
380
      case X86_INS_JMP:
1956
380
        return true;
1957
5.15k
    }
1958
5.15k
  }
1959
1960
  // not found
1961
0
  return false;
1962
5.15k
}
1963
1964
#ifndef CAPSTONE_DIET
1965
// add *CX register to regs_read[] & regs_write[]
1966
static void add_cx(MCInst *MI)
1967
27.3k
{
1968
27.3k
  if (MI->csh->detail) {
1969
27.3k
    x86_reg cx;
1970
1971
27.3k
    if (MI->csh->mode & CS_MODE_16)
1972
7.48k
      cx = X86_REG_CX;
1973
19.8k
    else if (MI->csh->mode & CS_MODE_32)
1974
9.07k
      cx = X86_REG_ECX;
1975
10.7k
    else  // 64-bit
1976
10.7k
      cx = X86_REG_RCX;
1977
1978
27.3k
    MI->flat_insn->detail->regs_read[MI->flat_insn->detail->regs_read_count] = cx;
1979
27.3k
    MI->flat_insn->detail->regs_read_count++;
1980
1981
27.3k
    MI->flat_insn->detail->regs_write[MI->flat_insn->detail->regs_write_count] = cx;
1982
27.3k
    MI->flat_insn->detail->regs_write_count++;
1983
27.3k
  }
1984
27.3k
}
1985
#endif
1986
1987
// return true if we patch the mnemonic
1988
bool X86_lockrep(MCInst *MI, SStream *O)
1989
776k
{
1990
776k
  unsigned int opcode;
1991
776k
  bool res = false;
1992
1993
776k
  switch(MI->x86_prefix[0]) {
1994
697k
    default:
1995
697k
      break;
1996
697k
    case 0xf0:
1997
31.9k
#ifndef CAPSTONE_DIET
1998
31.9k
      if (MI->xAcquireRelease == 0xf2)
1999
837
        SStream_concat(O, "xacquire|lock|");
2000
31.1k
      else if (MI->xAcquireRelease == 0xf3)
2001
1.20k
        SStream_concat(O, "xrelease|lock|");
2002
29.9k
      else
2003
29.9k
        SStream_concat(O, "lock|");
2004
31.9k
#endif
2005
31.9k
      break;
2006
24.4k
    case 0xf2:  // repne
2007
24.4k
      opcode = MCInst_getOpcode(MI);
2008
2009
24.4k
#ifndef CAPSTONE_DIET // only care about memonic in standard (non-diet) mode
2010
24.4k
      if (xchg_mem(opcode) && MI->xAcquireRelease) {
2011
248
        SStream_concat(O, "xacquire|");
2012
24.2k
      } else if (valid_repne(MI->csh, opcode)) {
2013
6.07k
        SStream_concat(O, "repne|");
2014
6.07k
        add_cx(MI);
2015
18.1k
      } else if (valid_bnd(MI->csh, opcode)) {
2016
5.82k
        SStream_concat(O, "bnd|");
2017
12.3k
      } else {
2018
        // invalid prefix
2019
12.3k
        MI->x86_prefix[0] = 0;
2020
2021
        // handle special cases
2022
12.3k
#ifndef CAPSTONE_X86_REDUCE
2023
#if 0
2024
        if (opcode == X86_MULPDrr) {
2025
          MCInst_setOpcode(MI, X86_MULSDrr);
2026
          SStream_concat0(O, "mulsd\t");
2027
          res = true;
2028
        }
2029
#endif
2030
12.3k
#endif
2031
12.3k
      }
2032
#else // diet mode -> only patch opcode in special cases
2033
      if (!valid_repne(MI->csh, opcode)) {
2034
        MI->x86_prefix[0] = 0;
2035
      }
2036
#ifndef CAPSTONE_X86_REDUCE
2037
#if 0
2038
      // handle special cases
2039
      if (opcode == X86_MULPDrr) {
2040
        MCInst_setOpcode(MI, X86_MULSDrr);
2041
      }
2042
#endif
2043
#endif
2044
#endif
2045
24.4k
      break;
2046
2047
22.4k
    case 0xf3:
2048
22.4k
      opcode = MCInst_getOpcode(MI);
2049
2050
22.4k
#ifndef CAPSTONE_DIET // only care about memonic in standard (non-diet) mode
2051
22.4k
      if (xchg_mem(opcode) && MI->xAcquireRelease) {
2052
454
        SStream_concat(O, "xrelease|");
2053
21.9k
      } else if (valid_rep(MI->csh, opcode)) {
2054
5.03k
        SStream_concat(O, "rep|");
2055
5.03k
        add_cx(MI);
2056
16.9k
      } else if (valid_repe(MI->csh, opcode)) {
2057
1.87k
        SStream_concat(O, "repe|");
2058
1.87k
        add_cx(MI);
2059
15.0k
      } else if (valid_ret_repz(MI->csh, opcode)) {
2060
339
        SStream_concat(O, "repz|");
2061
14.7k
      } else {
2062
        // invalid prefix
2063
14.7k
        MI->x86_prefix[0] = 0;
2064
2065
        // handle special cases
2066
14.7k
#ifndef CAPSTONE_X86_REDUCE
2067
#if 0
2068
        // FIXME: remove this special case?
2069
        if (opcode == X86_MULPDrr) {
2070
          MCInst_setOpcode(MI, X86_MULSSrr);
2071
          SStream_concat0(O, "mulss\t");
2072
          res = true;
2073
        }
2074
#endif
2075
14.7k
#endif
2076
14.7k
      }
2077
#else // diet mode -> only patch opcode in special cases
2078
      if (!valid_rep(MI->csh, opcode) && !valid_repe(MI->csh, opcode)) {
2079
        MI->x86_prefix[0] = 0;
2080
      }
2081
#ifndef CAPSTONE_X86_REDUCE
2082
#if 0
2083
      // handle special cases
2084
      // FIXME: remove this special case?
2085
      if (opcode == X86_MULPDrr) {
2086
        MCInst_setOpcode(MI, X86_MULSSrr);
2087
      }
2088
#endif
2089
#endif
2090
#endif
2091
22.4k
      break;
2092
776k
  }
2093
2094
776k
  switch(MI->x86_prefix[1]) {
2095
773k
    default:
2096
773k
      break;
2097
773k
    case 0x3e:
2098
3.13k
      opcode = MCInst_getOpcode(MI);
2099
3.13k
      if (valid_notrack(MI->csh, opcode)) {
2100
186
        SStream_concat(O, "notrack|");
2101
186
      }
2102
3.13k
      break;
2103
776k
  }
2104
2105
  // copy normalized prefix[] back to x86.prefix[]
2106
776k
  if (MI->csh->detail)
2107
776k
    memcpy(MI->flat_insn->detail->x86.prefix, MI->x86_prefix, ARR_SIZE(MI->x86_prefix));
2108
2109
776k
  return res;
2110
776k
}
2111
2112
void op_addReg(MCInst *MI, int reg)
2113
111k
{
2114
111k
  if (MI->csh->detail) {
2115
111k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_REG;
2116
111k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].reg = reg;
2117
111k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->csh->regsize_map[reg];
2118
111k
    MI->flat_insn->detail->x86.op_count++;
2119
111k
  }
2120
2121
111k
  if (MI->op1_size == 0)
2122
71.8k
    MI->op1_size = MI->csh->regsize_map[reg];
2123
111k
}
2124
2125
void op_addImm(MCInst *MI, int v)
2126
6.52k
{
2127
6.52k
  if (MI->csh->detail) {
2128
6.52k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].type = X86_OP_IMM;
2129
6.52k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].imm = v;
2130
    // if op_count > 0, then this operand's size is taken from the destination op
2131
6.52k
    if (MI->csh->syntax != CS_OPT_SYNTAX_ATT) {
2132
6.52k
      if (MI->flat_insn->detail->x86.op_count > 0)
2133
6.52k
        MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->flat_insn->detail->x86.operands[0].size;
2134
0
      else
2135
0
        MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count].size = MI->imm_size;
2136
6.52k
    } else
2137
0
      MI->has_imm = true;
2138
6.52k
    MI->flat_insn->detail->x86.op_count++;
2139
6.52k
  }
2140
2141
6.52k
  if (MI->op1_size == 0)
2142
0
    MI->op1_size = MI->imm_size;
2143
6.52k
}
2144
2145
void op_addXopCC(MCInst *MI, int v)
2146
4.01k
{
2147
4.01k
  if (MI->csh->detail) {
2148
4.01k
    MI->flat_insn->detail->x86.xop_cc = v;
2149
4.01k
  }
2150
4.01k
}
2151
2152
void op_addSseCC(MCInst *MI, int v)
2153
0
{
2154
0
  if (MI->csh->detail) {
2155
0
    MI->flat_insn->detail->x86.sse_cc = v;
2156
0
  }
2157
0
}
2158
2159
void op_addAvxCC(MCInst *MI, int v)
2160
19.0k
{
2161
19.0k
  if (MI->csh->detail) {
2162
19.0k
    MI->flat_insn->detail->x86.avx_cc = v;
2163
19.0k
  }
2164
19.0k
}
2165
2166
void op_addAvxRoundingMode(MCInst *MI, int v)
2167
5.21k
{
2168
5.21k
  if (MI->csh->detail) {
2169
5.21k
    MI->flat_insn->detail->x86.avx_rm = v;
2170
5.21k
  }
2171
5.21k
}
2172
2173
// below functions supply details to X86GenAsmWriter*.inc
2174
void op_addAvxZeroOpmask(MCInst *MI)
2175
9.54k
{
2176
9.54k
  if (MI->csh->detail) {
2177
    // link with the previous operand
2178
9.54k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count - 1].avx_zero_opmask = true;
2179
9.54k
  }
2180
9.54k
}
2181
2182
void op_addAvxSae(MCInst *MI)
2183
12.2k
{
2184
12.2k
  if (MI->csh->detail) {
2185
12.2k
    MI->flat_insn->detail->x86.avx_sae = true;
2186
12.2k
  }
2187
12.2k
}
2188
2189
void op_addAvxBroadcast(MCInst *MI, x86_avx_bcast v)
2190
15.2k
{
2191
15.2k
  if (MI->csh->detail) {
2192
    // link with the previous operand
2193
15.2k
    MI->flat_insn->detail->x86.operands[MI->flat_insn->detail->x86.op_count - 1].avx_bcast = v;
2194
15.2k
  }
2195
15.2k
}
2196
2197
#ifndef CAPSTONE_DIET
2198
// map instruction to its characteristics
2199
typedef struct insn_op {
2200
  uint64_t flags; // how this instruction update EFLAGS(arithmetic instrcutions) of FPU FLAGS(for FPU instructions)
2201
  uint8_t access[6];
2202
} insn_op;
2203
2204
static const insn_op insn_ops[] = {
2205
#ifdef CAPSTONE_X86_REDUCE
2206
#include "X86MappingInsnOp_reduce.inc"
2207
#else
2208
#include "X86MappingInsnOp.inc"
2209
#endif
2210
};
2211
2212
// given internal insn id, return operand access info
2213
const uint8_t *X86_get_op_access(cs_struct *h, unsigned int id, uint64_t *eflags)
2214
3.64M
{
2215
3.64M
  unsigned int i = find_insn_h(h, id);
2216
3.64M
  if (i != -1) {
2217
3.64M
    *eflags = insn_ops[i].flags;
2218
3.64M
    return insn_ops[i].access;
2219
3.64M
  }
2220
2221
0
  return NULL;
2222
3.64M
}
2223
2224
void X86_reg_access(const cs_insn *insn,
2225
    cs_regs regs_read, uint8_t *regs_read_count,
2226
    cs_regs regs_write, uint8_t *regs_write_count)
2227
0
{
2228
0
  uint8_t i;
2229
0
  uint8_t read_count, write_count;
2230
0
  cs_x86 *x86 = &(insn->detail->x86);
2231
2232
0
  read_count = insn->detail->regs_read_count;
2233
0
  write_count = insn->detail->regs_write_count;
2234
2235
  // implicit registers
2236
0
  memcpy(regs_read, insn->detail->regs_read, read_count * sizeof(insn->detail->regs_read[0]));
2237
0
  memcpy(regs_write, insn->detail->regs_write, write_count * sizeof(insn->detail->regs_write[0]));
2238
2239
  // explicit registers
2240
0
  for (i = 0; i < x86->op_count; i++) {
2241
0
    cs_x86_op *op = &(x86->operands[i]);
2242
0
    switch((int)op->type) {
2243
0
      case X86_OP_REG:
2244
0
        if ((op->access & CS_AC_READ) && !arr_exist(regs_read, read_count, op->reg)) {
2245
0
          regs_read[read_count] = op->reg;
2246
0
          read_count++;
2247
0
        }
2248
0
        if ((op->access & CS_AC_WRITE) && !arr_exist(regs_write, write_count, op->reg)) {
2249
0
          regs_write[write_count] = op->reg;
2250
0
          write_count++;
2251
0
        }
2252
0
        break;
2253
0
      case X86_OP_MEM:
2254
        // registers appeared in memory references always being read
2255
0
        if ((op->mem.segment != X86_REG_INVALID)) {
2256
0
          regs_read[read_count] = op->mem.segment;
2257
0
          read_count++;
2258
0
        }
2259
0
        if ((op->mem.base != X86_REG_INVALID) && !arr_exist(regs_read, read_count, op->mem.base)) {
2260
0
          regs_read[read_count] = op->mem.base;
2261
0
          read_count++;
2262
0
        }
2263
0
        if ((op->mem.index != X86_REG_INVALID) && !arr_exist(regs_read, read_count, op->mem.index)) {
2264
0
          regs_read[read_count] = op->mem.index;
2265
0
          read_count++;
2266
0
        }
2267
0
      default:
2268
0
        break;
2269
0
    }
2270
0
  }
2271
2272
0
  *regs_read_count = read_count;
2273
0
  *regs_write_count = write_count;
2274
0
}
2275
#endif
2276
2277
// map immediate size to instruction id
2278
// this array is sorted for binary searching
2279
static const struct size_id {
2280
  uint8_t   enc_size;
2281
  uint8_t   size;
2282
  uint16_t  id;
2283
} x86_imm_size[] = {
2284
#include "X86ImmSize.inc"
2285
};
2286
2287
// given the instruction name, return the size of its immediate operand (or 0)
2288
uint8_t X86_immediate_size(unsigned int id, uint8_t *enc_size)
2289
256k
{
2290
  // binary searching since the IDs are sorted in order
2291
256k
  unsigned int left, right, m;
2292
2293
256k
  right = ARR_SIZE(x86_imm_size) - 1;
2294
2295
256k
  if (id < x86_imm_size[0].id || id > x86_imm_size[right].id)
2296
    // not found
2297
0
    return 0;
2298
2299
256k
  left = 0;
2300
2301
1.98M
  while (left <= right) {
2302
1.89M
    m = (left + right) / 2;
2303
1.89M
    if (id == x86_imm_size[m].id) {
2304
165k
      if (enc_size != NULL)
2305
164k
        *enc_size = x86_imm_size[m].enc_size;
2306
2307
165k
      return x86_imm_size[m].size;
2308
165k
    }
2309
2310
1.73M
    if (id > x86_imm_size[m].id)
2311
830k
      left = m + 1;
2312
900k
    else {
2313
900k
      if (m == 0)
2314
0
        break;
2315
900k
      right = m - 1;
2316
900k
    }
2317
1.73M
  }
2318
2319
  // not found
2320
90.6k
  return 0;
2321
256k
}
2322
2323
#define GET_REGINFO_ENUM
2324
#include "X86GenRegisterInfo.inc"
2325
2326
// map internal register id to public register id
2327
static const struct register_map {
2328
  unsigned short    id;
2329
  unsigned short    pub_id;
2330
} reg_map [] = {
2331
  // first dummy map
2332
  { 0, 0 },
2333
#include "X86MappingReg.inc"
2334
};
2335
2336
// return 0 on invalid input, or public register ID otherwise
2337
// NOTE: reg_map is sorted in order of internal register
2338
unsigned short X86_register_map(unsigned short id)
2339
3.97M
{
2340
3.97M
  if (id < ARR_SIZE(reg_map))
2341
3.97M
    return reg_map[id].pub_id;
2342
2343
0
  return 0;
2344
3.97M
}
2345
2346
/// The post-printer function. Used to fixup flaws in the disassembly information
2347
/// of certain instructions.
2348
1.53M
void X86_postprinter(csh handle, cs_insn *insn, char *mnem, MCInst *mci) {
2349
1.53M
  if (!insn || !insn->detail) {
2350
0
    return;
2351
0
  }
2352
1.53M
  switch (insn->id) {
2353
1.52M
  default:
2354
1.52M
    break;
2355
1.52M
  case X86_INS_RCL:
2356
    // Addmissing 1 immediate
2357
14.1k
    if (insn->detail->x86.op_count > 1) {
2358
13.0k
      return;
2359
13.0k
    }
2360
1.13k
    insn->detail->x86.operands[1].imm = 1;
2361
1.13k
    insn->detail->x86.operands[1].type = X86_OP_IMM;
2362
1.13k
    insn->detail->x86.operands[1].access = CS_AC_READ;
2363
1.13k
    insn->detail->x86.op_count++;
2364
1.13k
    break;
2365
1.53M
  }
2366
1.53M
}
2367
2368
2369
#endif