Coverage Report

Created: 2025-10-10 06:20

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