Coverage Report

Created: 2026-05-30 06:22

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
757k
{
65
757k
  return sib_base_map[r];
66
757k
}
67
68
x86_reg x86_map_sib_index(int r)
69
757k
{
70
757k
  return sib_index_map[r];
71
757k
}
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
1.24M
{
711
1.24M
#ifndef CAPSTONE_DIET
712
1.24M
  cs_struct *ud = (cs_struct *)handle;
713
714
1.24M
  if (reg >= ARR_SIZE(reg_name_maps))
715
0
    return NULL;
716
717
1.24M
  if (reg == X86_REG_EFLAGS) {
718
501k
    if (ud->mode & CS_MODE_32)
719
182k
      return "eflags";
720
319k
    if (ud->mode & CS_MODE_64)
721
160k
      return "rflags";
722
319k
  }
723
724
900k
  return reg_name_maps[reg].name;
725
#else
726
  return NULL;
727
#endif
728
1.24M
}
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
757k
{
744
757k
#ifndef CAPSTONE_DIET
745
757k
  if (id >= ARR_SIZE(insn_name_maps))
746
0
    return NULL;
747
748
757k
  return insn_name_maps[id];
749
#else
750
  return NULL;
751
#endif
752
757k
}
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
355k
{
814
355k
#ifndef CAPSTONE_DIET
815
355k
  return id2name(group_name_maps, ARR_SIZE(group_name_maps), id);
816
#else
817
  return NULL;
818
#endif
819
355k
}
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
216k
{
842
216k
  uint8_t i;
843
844
331k
  for (i = 0; i < max; i++) {
845
294k
    if (arr[i] == r1) {
846
179k
      arr[i] = r2;
847
179k
      break;
848
179k
    }
849
294k
  }
850
216k
}
851
#endif
852
853
// look for @id in @insns
854
// return -1 if not found
855
unsigned int find_insn(unsigned int id)
856
12.1k
{
857
  // binary searching since the IDs are sorted in order
858
12.1k
  unsigned int left, right, m;
859
12.1k
  unsigned int max = ARR_SIZE(insns);
860
861
12.1k
  right = max - 1;
862
863
12.1k
  if (id < insns[0].id || id > insns[right].id)
864
    // not found
865
81
    return -1;
866
867
12.1k
  left = 0;
868
869
160k
  while (left <= right) {
870
160k
    m = (left + right) / 2;
871
160k
    if (id == insns[m].id) {
872
12.1k
      return m;
873
12.1k
    }
874
875
148k
    if (id < insns[m].id)
876
59.7k
      right = m - 1;
877
88.8k
    else
878
88.8k
      left = m + 1;
879
148k
  }
880
881
  // not found
882
  // printf("NOT FOUNDDDDDDDDDDDDDDD id = %u\n", id);
883
0
  return -1;
884
12.1k
}
885
886
static inline unsigned int find_insn_h(cs_struct *h, unsigned int id)
887
2.59M
{
888
2.59M
  if (h && h->x86_insn_lut && id <= h->x86_insn_lut_max)
889
2.59M
    return (unsigned int)(int16_t)h->x86_insn_lut[id];
890
891
0
  return find_insn(id);
892
2.59M
}
893
894
// given internal insn id, return public instruction info
895
void X86_get_insn_id(cs_struct *h, cs_insn *insn, unsigned int id)
896
757k
{
897
757k
  unsigned int i = find_insn_h(h, id);
898
757k
  if (i != -1) {
899
757k
    insn->id = insns[i].mapid;
900
901
757k
    if (h->detail_opt) {
902
757k
#ifndef CAPSTONE_DIET
903
757k
      memcpy(insn->detail->regs_read, insns[i].regs_use,
904
757k
             sizeof(insns[i].regs_use));
905
757k
      insn->detail->regs_read_count =
906
757k
        (uint8_t)count_positive(insns[i].regs_use);
907
908
      // special cases when regs_write[] depends on arch
909
757k
      switch (id) {
910
756k
      default:
911
756k
        memcpy(insn->detail->regs_write,
912
756k
               insns[i].regs_mod,
913
756k
               sizeof(insns[i].regs_mod));
914
756k
        insn->detail->regs_write_count =
915
756k
          (uint8_t)count_positive(
916
756k
            insns[i].regs_mod);
917
756k
        break;
918
557
      case X86_RDTSC:
919
557
        if (h->mode == CS_MODE_64) {
920
34
          memcpy(insn->detail->regs_write,
921
34
                 insns[i].regs_mod,
922
34
                 sizeof(insns[i].regs_mod));
923
34
          insn->detail->regs_write_count =
924
34
            (uint8_t)count_positive(
925
34
              insns[i].regs_mod);
926
523
        } else {
927
523
          insn->detail->regs_write[0] =
928
523
            X86_REG_EAX;
929
523
          insn->detail->regs_write[1] =
930
523
            X86_REG_EDX;
931
523
          insn->detail->regs_write_count = 2;
932
523
        }
933
557
        break;
934
662
      case X86_RDTSCP:
935
662
        if (h->mode == CS_MODE_64) {
936
139
          memcpy(insn->detail->regs_write,
937
139
                 insns[i].regs_mod,
938
139
                 sizeof(insns[i].regs_mod));
939
139
          insn->detail->regs_write_count =
940
139
            (uint8_t)count_positive(
941
139
              insns[i].regs_mod);
942
523
        } else {
943
523
          insn->detail->regs_write[0] =
944
523
            X86_REG_EAX;
945
523
          insn->detail->regs_write[1] =
946
523
            X86_REG_ECX;
947
523
          insn->detail->regs_write[2] =
948
523
            X86_REG_EDX;
949
523
          insn->detail->regs_write_count = 3;
950
523
        }
951
662
        break;
952
757k
      }
953
757k
      switch (insn->id) {
954
752k
      default:
955
752k
        break;
956
957
752k
      case X86_INS_LOOP:
958
3.19k
      case X86_INS_LOOPE:
959
5.00k
      case X86_INS_LOOPNE:
960
5.00k
        switch (h->mode) {
961
1.90k
        default:
962
1.90k
          break;
963
1.90k
        case CS_MODE_16:
964
1.82k
          arr_replace(
965
1.82k
            insn->detail->regs_read,
966
1.82k
            insn->detail->regs_read_count,
967
1.82k
            X86_REG_EIP, X86_REG_IP);
968
1.82k
          arr_replace(
969
1.82k
            insn->detail->regs_write,
970
1.82k
            insn->detail->regs_write_count,
971
1.82k
            X86_REG_EIP, X86_REG_IP);
972
1.82k
          arr_replace(
973
1.82k
            insn->detail->regs_read,
974
1.82k
            insn->detail->regs_read_count,
975
1.82k
            X86_REG_ECX, X86_REG_CX);
976
1.82k
          arr_replace(
977
1.82k
            insn->detail->regs_write,
978
1.82k
            insn->detail->regs_write_count,
979
1.82k
            X86_REG_ECX, X86_REG_CX);
980
1.82k
          break;
981
1.28k
        case CS_MODE_64:
982
1.28k
          arr_replace(
983
1.28k
            insn->detail->regs_read,
984
1.28k
            insn->detail->regs_read_count,
985
1.28k
            X86_REG_EIP, X86_REG_RIP);
986
1.28k
          arr_replace(
987
1.28k
            insn->detail->regs_write,
988
1.28k
            insn->detail->regs_write_count,
989
1.28k
            X86_REG_EIP, X86_REG_RIP);
990
1.28k
          arr_replace(
991
1.28k
            insn->detail->regs_read,
992
1.28k
            insn->detail->regs_read_count,
993
1.28k
            X86_REG_ECX, X86_REG_RCX);
994
1.28k
          arr_replace(
995
1.28k
            insn->detail->regs_write,
996
1.28k
            insn->detail->regs_write_count,
997
1.28k
            X86_REG_ECX, X86_REG_RCX);
998
1.28k
          break;
999
5.00k
        }
1000
757k
      }
1001
1002
757k
      switch (insn->id) {
1003
690k
      default:
1004
690k
        break;
1005
690k
      case X86_INS_LODSB:
1006
7.21k
      case X86_INS_LODSD:
1007
7.66k
      case X86_INS_LODSQ:
1008
9.26k
      case X86_INS_LODSW:
1009
9.26k
        switch (h->mode) {
1010
3.34k
        default:
1011
3.34k
          break;
1012
3.34k
        case CS_MODE_16:
1013
2.10k
          arr_replace(
1014
2.10k
            insn->detail->regs_read,
1015
2.10k
            insn->detail->regs_read_count,
1016
2.10k
            X86_REG_ESI, X86_REG_SI);
1017
2.10k
          arr_replace(
1018
2.10k
            insn->detail->regs_write,
1019
2.10k
            insn->detail->regs_write_count,
1020
2.10k
            X86_REG_ESI, X86_REG_SI);
1021
2.10k
          break;
1022
3.82k
        case CS_MODE_64:
1023
3.82k
          arr_replace(
1024
3.82k
            insn->detail->regs_read,
1025
3.82k
            insn->detail->regs_read_count,
1026
3.82k
            X86_REG_ESI, X86_REG_RSI);
1027
3.82k
          arr_replace(
1028
3.82k
            insn->detail->regs_write,
1029
3.82k
            insn->detail->regs_write_count,
1030
3.82k
            X86_REG_ESI, X86_REG_RSI);
1031
3.82k
          break;
1032
9.26k
        }
1033
9.26k
        break;
1034
1035
9.26k
      case X86_INS_SCASB:
1036
5.00k
      case X86_INS_SCASD:
1037
5.99k
      case X86_INS_SCASW:
1038
7.28k
      case X86_INS_SCASQ:
1039
8.95k
      case X86_INS_STOSB:
1040
11.1k
      case X86_INS_STOSD:
1041
11.7k
      case X86_INS_STOSQ:
1042
13.3k
      case X86_INS_STOSW:
1043
13.3k
        switch (h->mode) {
1044
4.27k
        default:
1045
4.27k
          break;
1046
4.27k
        case CS_MODE_16:
1047
3.75k
          arr_replace(
1048
3.75k
            insn->detail->regs_read,
1049
3.75k
            insn->detail->regs_read_count,
1050
3.75k
            X86_REG_EDI, X86_REG_DI);
1051
3.75k
          arr_replace(
1052
3.75k
            insn->detail->regs_write,
1053
3.75k
            insn->detail->regs_write_count,
1054
3.75k
            X86_REG_EDI, X86_REG_DI);
1055
3.75k
          break;
1056
5.28k
        case CS_MODE_64:
1057
5.28k
          arr_replace(
1058
5.28k
            insn->detail->regs_read,
1059
5.28k
            insn->detail->regs_read_count,
1060
5.28k
            X86_REG_EDI, X86_REG_RDI);
1061
5.28k
          arr_replace(
1062
5.28k
            insn->detail->regs_write,
1063
5.28k
            insn->detail->regs_write_count,
1064
5.28k
            X86_REG_EDI, X86_REG_RDI);
1065
5.28k
          break;
1066
13.3k
        }
1067
13.3k
        break;
1068
1069
13.3k
      case X86_INS_CMPSB:
1070
7.66k
      case X86_INS_CMPSD:
1071
8.04k
      case X86_INS_CMPSQ:
1072
10.3k
      case X86_INS_CMPSW:
1073
12.7k
      case X86_INS_MOVSB:
1074
15.0k
      case X86_INS_MOVSW:
1075
18.7k
      case X86_INS_MOVSD:
1076
20.2k
      case X86_INS_MOVSQ:
1077
20.2k
        switch (h->mode) {
1078
4.79k
        default:
1079
4.79k
          break;
1080
7.56k
        case CS_MODE_16:
1081
7.56k
          arr_replace(
1082
7.56k
            insn->detail->regs_read,
1083
7.56k
            insn->detail->regs_read_count,
1084
7.56k
            X86_REG_EDI, X86_REG_DI);
1085
7.56k
          arr_replace(
1086
7.56k
            insn->detail->regs_write,
1087
7.56k
            insn->detail->regs_write_count,
1088
7.56k
            X86_REG_EDI, X86_REG_DI);
1089
7.56k
          arr_replace(
1090
7.56k
            insn->detail->regs_read,
1091
7.56k
            insn->detail->regs_read_count,
1092
7.56k
            X86_REG_ESI, X86_REG_SI);
1093
7.56k
          arr_replace(
1094
7.56k
            insn->detail->regs_write,
1095
7.56k
            insn->detail->regs_write_count,
1096
7.56k
            X86_REG_ESI, X86_REG_SI);
1097
7.56k
          break;
1098
7.87k
        case CS_MODE_64:
1099
7.87k
          arr_replace(
1100
7.87k
            insn->detail->regs_read,
1101
7.87k
            insn->detail->regs_read_count,
1102
7.87k
            X86_REG_EDI, X86_REG_RDI);
1103
7.87k
          arr_replace(
1104
7.87k
            insn->detail->regs_write,
1105
7.87k
            insn->detail->regs_write_count,
1106
7.87k
            X86_REG_EDI, X86_REG_RDI);
1107
7.87k
          arr_replace(
1108
7.87k
            insn->detail->regs_read,
1109
7.87k
            insn->detail->regs_read_count,
1110
7.87k
            X86_REG_ESI, X86_REG_RSI);
1111
7.87k
          arr_replace(
1112
7.87k
            insn->detail->regs_write,
1113
7.87k
            insn->detail->regs_write_count,
1114
7.87k
            X86_REG_ESI, X86_REG_RSI);
1115
7.87k
          break;
1116
20.2k
        }
1117
20.2k
        break;
1118
1119
20.2k
      case X86_INS_ENTER:
1120
3.62k
      case X86_INS_LEAVE:
1121
3.62k
        switch (h->mode) {
1122
1.49k
        default:
1123
1.49k
          break;
1124
1.49k
        case CS_MODE_16:
1125
1.04k
          arr_replace(
1126
1.04k
            insn->detail->regs_read,
1127
1.04k
            insn->detail->regs_read_count,
1128
1.04k
            X86_REG_EBP, X86_REG_BP);
1129
1.04k
          arr_replace(
1130
1.04k
            insn->detail->regs_read,
1131
1.04k
            insn->detail->regs_read_count,
1132
1.04k
            X86_REG_ESP, X86_REG_SP);
1133
1.04k
          arr_replace(
1134
1.04k
            insn->detail->regs_write,
1135
1.04k
            insn->detail->regs_write_count,
1136
1.04k
            X86_REG_EBP, X86_REG_BP);
1137
1.04k
          arr_replace(
1138
1.04k
            insn->detail->regs_write,
1139
1.04k
            insn->detail->regs_write_count,
1140
1.04k
            X86_REG_ESP, X86_REG_SP);
1141
1.04k
          break;
1142
1.08k
        case CS_MODE_64:
1143
1.08k
          arr_replace(
1144
1.08k
            insn->detail->regs_read,
1145
1.08k
            insn->detail->regs_read_count,
1146
1.08k
            X86_REG_EBP, X86_REG_RBP);
1147
1.08k
          arr_replace(
1148
1.08k
            insn->detail->regs_read,
1149
1.08k
            insn->detail->regs_read_count,
1150
1.08k
            X86_REG_ESP, X86_REG_RSP);
1151
1.08k
          arr_replace(
1152
1.08k
            insn->detail->regs_write,
1153
1.08k
            insn->detail->regs_write_count,
1154
1.08k
            X86_REG_EBP, X86_REG_RBP);
1155
1.08k
          arr_replace(
1156
1.08k
            insn->detail->regs_write,
1157
1.08k
            insn->detail->regs_write_count,
1158
1.08k
            X86_REG_ESP, X86_REG_RSP);
1159
3.62k
        }
1160
3.62k
        break;
1161
1162
4.16k
      case X86_INS_INSB:
1163
6.51k
      case X86_INS_INSW:
1164
10.2k
      case X86_INS_INSD:
1165
10.2k
        switch (h->mode) {
1166
3.95k
        default:
1167
3.95k
          break;
1168
3.95k
        case CS_MODE_16:
1169
3.24k
          arr_replace(
1170
3.24k
            insn->detail->regs_read,
1171
3.24k
            insn->detail->regs_read_count,
1172
3.24k
            X86_REG_EDI, X86_REG_DI);
1173
3.24k
          arr_replace(
1174
3.24k
            insn->detail->regs_write,
1175
3.24k
            insn->detail->regs_write_count,
1176
3.24k
            X86_REG_EDI, X86_REG_DI);
1177
3.24k
          break;
1178
3.00k
        case CS_MODE_64:
1179
3.00k
          arr_replace(
1180
3.00k
            insn->detail->regs_read,
1181
3.00k
            insn->detail->regs_read_count,
1182
3.00k
            X86_REG_EDI, X86_REG_RDI);
1183
3.00k
          arr_replace(
1184
3.00k
            insn->detail->regs_write,
1185
3.00k
            insn->detail->regs_write_count,
1186
3.00k
            X86_REG_EDI, X86_REG_RDI);
1187
3.00k
          break;
1188
10.2k
        }
1189
10.2k
        break;
1190
1191
10.2k
      case X86_INS_OUTSB:
1192
6.15k
      case X86_INS_OUTSW:
1193
10.2k
      case X86_INS_OUTSD:
1194
10.2k
        switch (h->mode) {
1195
4.91k
        default:
1196
4.91k
          break;
1197
4.91k
        case CS_MODE_64:
1198
1.73k
          arr_replace(
1199
1.73k
            insn->detail->regs_read,
1200
1.73k
            insn->detail->regs_read_count,
1201
1.73k
            X86_REG_ESI, X86_REG_RSI);
1202
1.73k
          arr_replace(
1203
1.73k
            insn->detail->regs_write,
1204
1.73k
            insn->detail->regs_write_count,
1205
1.73k
            X86_REG_ESI, X86_REG_RSI);
1206
1.73k
          break;
1207
3.56k
        case CS_MODE_16:
1208
3.56k
          arr_replace(
1209
3.56k
            insn->detail->regs_read,
1210
3.56k
            insn->detail->regs_read_count,
1211
3.56k
            X86_REG_ESI, X86_REG_SI);
1212
3.56k
          arr_replace(
1213
3.56k
            insn->detail->regs_write,
1214
3.56k
            insn->detail->regs_write_count,
1215
3.56k
            X86_REG_ESI, X86_REG_SI);
1216
3.56k
          break;
1217
10.2k
        }
1218
10.2k
        break;
1219
757k
      }
1220
1221
757k
      switch (insn->id) {
1222
706k
      default:
1223
706k
        break;
1224
706k
      case X86_INS_LODSB:
1225
7.21k
      case X86_INS_LODSD:
1226
8.81k
      case X86_INS_LODSW:
1227
11.8k
      case X86_INS_CMPSB:
1228
16.4k
      case X86_INS_CMPSD:
1229
18.7k
      case X86_INS_CMPSW:
1230
21.2k
      case X86_INS_MOVSB:
1231
23.4k
      case X86_INS_MOVSW:
1232
27.2k
      case X86_INS_MOVSD:
1233
32.0k
      case X86_INS_OUTSB:
1234
33.3k
      case X86_INS_OUTSW:
1235
37.4k
      case X86_INS_OUTSD:
1236
37.4k
        switch (h->mode) {
1237
11.1k
        default:
1238
11.1k
          break;
1239
13.2k
        case CS_MODE_16:
1240
26.2k
        case CS_MODE_32: {
1241
26.2k
          int pos = insn->detail->regs_read_count;
1242
26.2k
          insn->detail->regs_read[pos] =
1243
26.2k
            X86_REG_DS;
1244
26.2k
          insn->detail->regs_read_count += 1;
1245
26.2k
        } break;
1246
37.4k
        }
1247
37.4k
        break;
1248
1249
37.4k
      case X86_INS_JMP:
1250
12.0k
      case X86_INS_LJMP:
1251
12.0k
        switch (h->mode) {
1252
3.44k
        default:
1253
3.44k
          break;
1254
4.49k
        case CS_MODE_16:
1255
4.49k
          arr_replace(
1256
4.49k
            insn->detail->regs_read,
1257
4.49k
            insn->detail->regs_read_count,
1258
4.49k
            X86_REG_EIP, X86_REG_IP);
1259
4.49k
          arr_replace(
1260
4.49k
            insn->detail->regs_write,
1261
4.49k
            insn->detail->regs_write_count,
1262
4.49k
            X86_REG_EIP, X86_REG_IP);
1263
4.49k
          break;
1264
4.14k
        case CS_MODE_64:
1265
4.14k
          arr_replace(
1266
4.14k
            insn->detail->regs_read,
1267
4.14k
            insn->detail->regs_read_count,
1268
4.14k
            X86_REG_EIP, X86_REG_RIP);
1269
4.14k
          arr_replace(
1270
4.14k
            insn->detail->regs_write,
1271
4.14k
            insn->detail->regs_write_count,
1272
4.14k
            X86_REG_EIP, X86_REG_RIP);
1273
4.14k
          break;
1274
12.0k
        }
1275
12.0k
        break;
1276
1277
12.0k
      case X86_INS_SYSENTER: {
1278
975
        switch (h->mode) {
1279
246
        default:
1280
246
          break;
1281
631
        case CS_MODE_16:
1282
631
          arr_replace(
1283
631
            insn->detail->regs_write,
1284
631
            insn->detail->regs_write_count,
1285
631
            X86_REG_EIP, X86_REG_IP);
1286
631
          arr_replace(
1287
631
            insn->detail->regs_write,
1288
631
            insn->detail->regs_write_count,
1289
631
            X86_REG_ESP, X86_REG_SP);
1290
631
          break;
1291
98
        case CS_MODE_64:
1292
98
          arr_replace(
1293
98
            insn->detail->regs_write,
1294
98
            insn->detail->regs_write_count,
1295
98
            X86_REG_EIP, X86_REG_RIP);
1296
98
          arr_replace(
1297
98
            insn->detail->regs_write,
1298
98
            insn->detail->regs_write_count,
1299
98
            X86_REG_ESP, X86_REG_RSP);
1300
98
          break;
1301
975
        }
1302
975
        break;
1303
975
      } break;
1304
975
      case X86_INS_SYSEXIT: {
1305
694
        switch (h->mode) {
1306
96
        default:
1307
96
          break;
1308
551
        case CS_MODE_16:
1309
551
          arr_replace(
1310
551
            insn->detail->regs_read,
1311
551
            insn->detail->regs_read_count,
1312
551
            X86_REG_ECX, X86_REG_CX);
1313
551
          arr_replace(
1314
551
            insn->detail->regs_read,
1315
551
            insn->detail->regs_read_count,
1316
551
            X86_REG_EDX, X86_REG_DX);
1317
551
          arr_replace(
1318
551
            insn->detail->regs_write,
1319
551
            insn->detail->regs_write_count,
1320
551
            X86_REG_EIP, X86_REG_IP);
1321
551
          arr_replace(
1322
551
            insn->detail->regs_write,
1323
551
            insn->detail->regs_write_count,
1324
551
            X86_REG_ESP, X86_REG_SP);
1325
551
          break;
1326
47
        case CS_MODE_64:
1327
47
          arr_replace(
1328
47
            insn->detail->regs_read,
1329
47
            insn->detail->regs_read_count,
1330
47
            X86_REG_ECX, X86_REG_RCX);
1331
47
          arr_replace(
1332
47
            insn->detail->regs_read,
1333
47
            insn->detail->regs_read_count,
1334
47
            X86_REG_EDX, X86_REG_RDX);
1335
47
          arr_replace(
1336
47
            insn->detail->regs_write,
1337
47
            insn->detail->regs_write_count,
1338
47
            X86_REG_EIP, X86_REG_RIP);
1339
47
          arr_replace(
1340
47
            insn->detail->regs_write,
1341
47
            insn->detail->regs_write_count,
1342
47
            X86_REG_ESP, X86_REG_RSP);
1343
47
          break;
1344
694
        }
1345
694
        break;
1346
694
      } break;
1347
757k
      }
1348
1349
757k
      memcpy(insn->detail->groups, insns[i].groups,
1350
757k
             sizeof(insns[i].groups));
1351
757k
      insn->detail->groups_count =
1352
757k
        (uint8_t)count_positive8(insns[i].groups);
1353
1354
757k
      if (insns[i].branch || insns[i].indirect_branch) {
1355
        // this insn also belongs to JUMP group. add JUMP group
1356
47.5k
        insn->detail
1357
47.5k
          ->groups[insn->detail->groups_count] =
1358
47.5k
          X86_GRP_JUMP;
1359
47.5k
        insn->detail->groups_count++;
1360
1361
47.5k
        switch (h->mode) {
1362
17.6k
        default:
1363
17.6k
          break;
1364
17.6k
        case CS_MODE_16:
1365
15.5k
          arr_replace(
1366
15.5k
            insn->detail->regs_read,
1367
15.5k
            insn->detail->regs_read_count,
1368
15.5k
            X86_REG_EIP, X86_REG_IP);
1369
15.5k
          arr_replace(
1370
15.5k
            insn->detail->regs_write,
1371
15.5k
            insn->detail->regs_write_count,
1372
15.5k
            X86_REG_EIP, X86_REG_IP);
1373
15.5k
          break;
1374
14.3k
        case CS_MODE_64:
1375
14.3k
          arr_replace(
1376
14.3k
            insn->detail->regs_read,
1377
14.3k
            insn->detail->regs_read_count,
1378
14.3k
            X86_REG_EIP, X86_REG_RIP);
1379
14.3k
          arr_replace(
1380
14.3k
            insn->detail->regs_write,
1381
14.3k
            insn->detail->regs_write_count,
1382
14.3k
            X86_REG_EIP, X86_REG_RIP);
1383
14.3k
          break;
1384
47.5k
        }
1385
47.5k
      }
1386
1387
757k
      switch (insns[i].id) {
1388
1.10k
      case X86_OUT8ir:
1389
1.51k
      case X86_OUT16ir:
1390
3.18k
      case X86_OUT32ir:
1391
3.18k
        if (insn->detail->x86.operands[0].imm == -78) {
1392
          // Writing to port 0xb2 causes an SMI on most platforms
1393
          // See: http://cs.gmu.edu/~tr-admin/papers/GMU-CS-TR-2011-8.pdf
1394
0
          insn->detail->groups
1395
0
            [insn->detail->groups_count] =
1396
0
            X86_GRP_INT;
1397
0
          insn->detail->groups_count++;
1398
0
        }
1399
3.18k
        break;
1400
1401
754k
      default:
1402
754k
        break;
1403
757k
      }
1404
757k
#endif
1405
757k
    }
1406
757k
  }
1407
757k
}
1408
1409
// map special instructions with accumulate registers.
1410
// this is needed because LLVM embeds these register names into AsmStrs[],
1411
// but not separately in operands
1412
struct insn_reg {
1413
  uint16_t insn;
1414
  x86_reg reg;
1415
  enum cs_ac_type access;
1416
};
1417
1418
struct insn_reg2 {
1419
  uint16_t insn;
1420
  x86_reg reg1, reg2;
1421
  enum cs_ac_type access1, access2;
1422
};
1423
1424
static inline uint16_t pack_insn_reg(x86_reg reg, enum cs_ac_type access)
1425
1.29M
{
1426
1.29M
  return (uint16_t)(((unsigned int)access << 12) |
1427
1.29M
        ((unsigned int)reg & 0x0fff));
1428
1.29M
}
1429
1430
static inline x86_reg unpack_insn_reg(uint16_t value, enum cs_ac_type *access)
1431
73.0k
{
1432
73.0k
  if (access)
1433
73.0k
    *access = (enum cs_ac_type)(value >> 12);
1434
73.0k
  return (x86_reg)(value & 0x0fff);
1435
73.0k
}
1436
1437
static const struct insn_reg insn_regs_att[] = {
1438
  { X86_INSB, X86_REG_DX, CS_AC_READ },
1439
  { X86_INSL, X86_REG_DX, CS_AC_READ },
1440
  { X86_INSW, X86_REG_DX, CS_AC_READ },
1441
  { X86_MOV16o16a, X86_REG_AX, CS_AC_READ },
1442
  { X86_MOV16o32a, X86_REG_AX, CS_AC_READ },
1443
  { X86_MOV16o64a, X86_REG_AX, CS_AC_READ },
1444
  { X86_MOV32o16a, X86_REG_EAX, CS_AC_READ },
1445
  { X86_MOV32o32a, X86_REG_EAX, CS_AC_READ },
1446
  { X86_MOV32o64a, X86_REG_EAX, CS_AC_READ },
1447
  { X86_MOV64o32a, X86_REG_RAX, CS_AC_READ },
1448
  { X86_MOV64o64a, X86_REG_RAX, CS_AC_READ },
1449
  { X86_MOV8o16a, X86_REG_AL, CS_AC_READ },
1450
  { X86_MOV8o32a, X86_REG_AL, CS_AC_READ },
1451
  { X86_MOV8o64a, X86_REG_AL, CS_AC_READ },
1452
  { X86_OUT16ir, X86_REG_AX, CS_AC_READ },
1453
  { X86_OUT32ir, X86_REG_EAX, CS_AC_READ },
1454
  { X86_OUT8ir, X86_REG_AL, CS_AC_READ },
1455
  { X86_POPDS16, X86_REG_DS, CS_AC_WRITE },
1456
  { X86_POPDS32, X86_REG_DS, CS_AC_WRITE },
1457
  { X86_POPES16, X86_REG_ES, CS_AC_WRITE },
1458
  { X86_POPES32, X86_REG_ES, CS_AC_WRITE },
1459
  { X86_POPFS16, X86_REG_FS, CS_AC_WRITE },
1460
  { X86_POPFS32, X86_REG_FS, CS_AC_WRITE },
1461
  { X86_POPFS64, X86_REG_FS, CS_AC_WRITE },
1462
  { X86_POPGS16, X86_REG_GS, CS_AC_WRITE },
1463
  { X86_POPGS32, X86_REG_GS, CS_AC_WRITE },
1464
  { X86_POPGS64, X86_REG_GS, CS_AC_WRITE },
1465
  { X86_POPSS16, X86_REG_SS, CS_AC_WRITE },
1466
  { X86_POPSS32, X86_REG_SS, CS_AC_WRITE },
1467
  { X86_PUSHCS16, X86_REG_CS, CS_AC_READ },
1468
  { X86_PUSHCS32, X86_REG_CS, CS_AC_READ },
1469
  { X86_PUSHDS16, X86_REG_DS, CS_AC_READ },
1470
  { X86_PUSHDS32, X86_REG_DS, CS_AC_READ },
1471
  { X86_PUSHES16, X86_REG_ES, CS_AC_READ },
1472
  { X86_PUSHES32, X86_REG_ES, CS_AC_READ },
1473
  { X86_PUSHFS16, X86_REG_FS, CS_AC_READ },
1474
  { X86_PUSHFS32, X86_REG_FS, CS_AC_READ },
1475
  { X86_PUSHFS64, X86_REG_FS, CS_AC_READ },
1476
  { X86_PUSHGS16, X86_REG_GS, CS_AC_READ },
1477
  { X86_PUSHGS32, X86_REG_GS, CS_AC_READ },
1478
  { X86_PUSHGS64, X86_REG_GS, CS_AC_READ },
1479
  { X86_PUSHSS16, X86_REG_SS, CS_AC_READ },
1480
  { X86_PUSHSS32, X86_REG_SS, CS_AC_READ },
1481
  { X86_RCL16rCL, X86_REG_CL, CS_AC_READ },
1482
  { X86_RCL32rCL, X86_REG_CL, CS_AC_READ },
1483
  { X86_RCL64rCL, X86_REG_CL, CS_AC_READ },
1484
  { X86_RCL8rCL, X86_REG_CL, CS_AC_READ },
1485
  { X86_RCR16rCL, X86_REG_CL, CS_AC_READ },
1486
  { X86_RCR32rCL, X86_REG_CL, CS_AC_READ },
1487
  { X86_RCR64rCL, X86_REG_CL, CS_AC_READ },
1488
  { X86_RCR8rCL, X86_REG_CL, CS_AC_READ },
1489
  { X86_ROL16rCL, X86_REG_CL, CS_AC_READ },
1490
  { X86_ROL32rCL, X86_REG_CL, CS_AC_READ },
1491
  { X86_ROL64rCL, X86_REG_CL, CS_AC_READ },
1492
  { X86_ROL8rCL, X86_REG_CL, CS_AC_READ },
1493
  { X86_ROR16rCL, X86_REG_CL, CS_AC_READ },
1494
  { X86_ROR32rCL, X86_REG_CL, CS_AC_READ },
1495
  { X86_ROR64rCL, X86_REG_CL, CS_AC_READ },
1496
  { X86_ROR8rCL, X86_REG_CL, CS_AC_READ },
1497
  { X86_SAL16rCL, X86_REG_CL, CS_AC_READ },
1498
  { X86_SAL32rCL, X86_REG_CL, CS_AC_READ },
1499
  { X86_SAL64rCL, X86_REG_CL, CS_AC_READ },
1500
  { X86_SAL8rCL, X86_REG_CL, CS_AC_READ },
1501
  { X86_SAR16rCL, X86_REG_CL, CS_AC_READ },
1502
  { X86_SAR32rCL, X86_REG_CL, CS_AC_READ },
1503
  { X86_SAR64rCL, X86_REG_CL, CS_AC_READ },
1504
  { X86_SAR8rCL, X86_REG_CL, CS_AC_READ },
1505
  { X86_SHL16rCL, X86_REG_CL, CS_AC_READ },
1506
  { X86_SHL32rCL, X86_REG_CL, CS_AC_READ },
1507
  { X86_SHL64rCL, X86_REG_CL, CS_AC_READ },
1508
  { X86_SHL8rCL, X86_REG_CL, CS_AC_READ },
1509
  { X86_SHLD16mrCL, X86_REG_CL, CS_AC_READ },
1510
  { X86_SHLD16rrCL, X86_REG_CL, CS_AC_READ },
1511
  { X86_SHLD32mrCL, X86_REG_CL, CS_AC_READ },
1512
  { X86_SHLD32rrCL, X86_REG_CL, CS_AC_READ },
1513
  { X86_SHLD64mrCL, X86_REG_CL, CS_AC_READ },
1514
  { X86_SHLD64rrCL, X86_REG_CL, CS_AC_READ },
1515
  { X86_SHR16rCL, X86_REG_CL, CS_AC_READ },
1516
  { X86_SHR32rCL, X86_REG_CL, CS_AC_READ },
1517
  { X86_SHR64rCL, X86_REG_CL, CS_AC_READ },
1518
  { X86_SHR8rCL, X86_REG_CL, CS_AC_READ },
1519
  { X86_SHRD16mrCL, X86_REG_CL, CS_AC_READ },
1520
  { X86_SHRD16rrCL, X86_REG_CL, CS_AC_READ },
1521
  { X86_SHRD32mrCL, X86_REG_CL, CS_AC_READ },
1522
  { X86_SHRD32rrCL, X86_REG_CL, CS_AC_READ },
1523
  { X86_SHRD64mrCL, X86_REG_CL, CS_AC_READ },
1524
  { X86_SHRD64rrCL, X86_REG_CL, CS_AC_READ },
1525
  { X86_XCHG16ar, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1526
  { X86_XCHG32ar, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1527
  { X86_XCHG64ar, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1528
};
1529
1530
static const struct insn_reg insn_regs_att_extra[] = {
1531
  // dummy entry, to avoid empty array
1532
  { 0, 0 },
1533
#ifndef CAPSTONE_X86_REDUCE
1534
  { X86_ADD_FrST0, X86_REG_ST0, CS_AC_READ },
1535
  { X86_DIVR_FrST0, X86_REG_ST0, CS_AC_READ },
1536
  { X86_DIV_FrST0, X86_REG_ST0, CS_AC_READ },
1537
  { X86_FNSTSW16r, X86_REG_AX, CS_AC_READ },
1538
  { X86_MUL_FrST0, X86_REG_ST0, CS_AC_READ },
1539
  { X86_SKINIT, X86_REG_EAX, CS_AC_READ },
1540
  { X86_SUBR_FrST0, X86_REG_ST0, CS_AC_READ },
1541
  { X86_SUB_FrST0, X86_REG_ST0, CS_AC_READ },
1542
  { X86_VMLOAD32, X86_REG_EAX, CS_AC_READ },
1543
  { X86_VMLOAD64, X86_REG_RAX, CS_AC_READ },
1544
  { X86_VMRUN32, X86_REG_EAX, CS_AC_READ },
1545
  { X86_VMRUN64, X86_REG_RAX, CS_AC_READ },
1546
  { X86_VMSAVE32, X86_REG_EAX, CS_AC_READ },
1547
  { X86_VMSAVE64, X86_REG_RAX, CS_AC_READ },
1548
#endif
1549
};
1550
1551
static const struct insn_reg insn_regs_intel[] = {
1552
  { X86_ADC16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1553
  { X86_ADC32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1554
  { X86_ADC64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1555
  { X86_ADC8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1556
  { X86_ADD16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1557
  { X86_ADD32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1558
  { X86_ADD64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1559
  { X86_ADD8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1560
  { X86_AND16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1561
  { X86_AND32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1562
  { X86_AND64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1563
  { X86_AND8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1564
  { X86_CMP16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1565
  { X86_CMP32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1566
  { X86_CMP64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1567
  { X86_CMP8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1568
  { X86_IN16ri, X86_REG_AX, CS_AC_WRITE },
1569
  { X86_IN32ri, X86_REG_EAX, CS_AC_WRITE },
1570
  { X86_IN8ri, X86_REG_AL, CS_AC_WRITE },
1571
  { X86_LODSB, X86_REG_AL, CS_AC_WRITE },
1572
  { X86_LODSL, X86_REG_EAX, CS_AC_WRITE },
1573
  { X86_LODSQ, X86_REG_RAX, CS_AC_WRITE },
1574
  { X86_LODSW, X86_REG_AX, CS_AC_WRITE },
1575
  { X86_MOV16ao16, X86_REG_AX,
1576
    CS_AC_WRITE }, // 16-bit A1 1020                  // mov     ax, word ptr [0x2010]
1577
  { X86_MOV16ao32, X86_REG_AX,
1578
    CS_AC_WRITE }, // 32-bit A1 10203040              // mov     ax, word ptr [0x40302010]
1579
  { X86_MOV16ao64, X86_REG_AX,
1580
    CS_AC_WRITE }, // 64-bit 66 A1 1020304050607080   // movabs  ax, word ptr [0x8070605040302010]
1581
  { X86_MOV32ao16, X86_REG_EAX,
1582
    CS_AC_WRITE }, // 32-bit 67 A1 1020               // mov     eax, dword ptr [0x2010]
1583
  { X86_MOV32ao32, X86_REG_EAX,
1584
    CS_AC_WRITE }, // 32-bit A1 10203040              // mov     eax, dword ptr [0x40302010]
1585
  { X86_MOV32ao64, X86_REG_EAX,
1586
    CS_AC_WRITE }, // 64-bit A1 1020304050607080      // movabs  eax, dword ptr [0x8070605040302010]
1587
  { X86_MOV64ao32, X86_REG_RAX,
1588
    CS_AC_WRITE }, // 64-bit 48 8B04 10203040         // mov     rax, qword ptr [0x40302010]
1589
  { X86_MOV64ao64, X86_REG_RAX,
1590
    CS_AC_WRITE }, // 64-bit 48 A1 1020304050607080   // movabs  rax, qword ptr [0x8070605040302010]
1591
  { X86_MOV8ao16, X86_REG_AL,
1592
    CS_AC_WRITE }, // 16-bit A0 1020                  // mov     al, byte ptr [0x2010]
1593
  { X86_MOV8ao32, X86_REG_AL,
1594
    CS_AC_WRITE }, // 32-bit A0 10203040              // mov     al, byte ptr [0x40302010]
1595
  { X86_MOV8ao64, X86_REG_AL,
1596
    CS_AC_WRITE }, // 64-bit 66 A0 1020304050607080   // movabs  al, byte ptr [0x8070605040302010]
1597
  { X86_OR16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1598
  { X86_OR32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1599
  { X86_OR64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1600
  { X86_OR8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1601
  { X86_OUTSB, X86_REG_DX, CS_AC_WRITE },
1602
  { X86_OUTSL, X86_REG_DX, CS_AC_WRITE },
1603
  { X86_OUTSW, X86_REG_DX, CS_AC_WRITE },
1604
  { X86_POPDS16, X86_REG_DS, CS_AC_WRITE },
1605
  { X86_POPDS32, X86_REG_DS, CS_AC_WRITE },
1606
  { X86_POPES16, X86_REG_ES, CS_AC_WRITE },
1607
  { X86_POPES32, X86_REG_ES, CS_AC_WRITE },
1608
  { X86_POPFS16, X86_REG_FS, CS_AC_WRITE },
1609
  { X86_POPFS32, X86_REG_FS, CS_AC_WRITE },
1610
  { X86_POPFS64, X86_REG_FS, CS_AC_WRITE },
1611
  { X86_POPGS16, X86_REG_GS, CS_AC_WRITE },
1612
  { X86_POPGS32, X86_REG_GS, CS_AC_WRITE },
1613
  { X86_POPGS64, X86_REG_GS, CS_AC_WRITE },
1614
  { X86_POPSS16, X86_REG_SS, CS_AC_WRITE },
1615
  { X86_POPSS32, X86_REG_SS, CS_AC_WRITE },
1616
  { X86_PUSHCS16, X86_REG_CS, CS_AC_READ },
1617
  { X86_PUSHCS32, X86_REG_CS, CS_AC_READ },
1618
  { X86_PUSHDS16, X86_REG_DS, CS_AC_READ },
1619
  { X86_PUSHDS32, X86_REG_DS, CS_AC_READ },
1620
  { X86_PUSHES16, X86_REG_ES, CS_AC_READ },
1621
  { X86_PUSHES32, X86_REG_ES, CS_AC_READ },
1622
  { X86_PUSHFS16, X86_REG_FS, CS_AC_READ },
1623
  { X86_PUSHFS32, X86_REG_FS, CS_AC_READ },
1624
  { X86_PUSHFS64, X86_REG_FS, CS_AC_READ },
1625
  { X86_PUSHGS16, X86_REG_GS, CS_AC_READ },
1626
  { X86_PUSHGS32, X86_REG_GS, CS_AC_READ },
1627
  { X86_PUSHGS64, X86_REG_GS, CS_AC_READ },
1628
  { X86_PUSHSS16, X86_REG_SS, CS_AC_READ },
1629
  { X86_PUSHSS32, X86_REG_SS, CS_AC_READ },
1630
  { X86_SBB16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1631
  { X86_SBB32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1632
  { X86_SBB64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1633
  { X86_SBB8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1634
  { X86_SCASB, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1635
  { X86_SCASL, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1636
  { X86_SCASQ, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1637
  { X86_SCASW, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1638
  { X86_SUB16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1639
  { X86_SUB32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1640
  { X86_SUB64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1641
  { X86_SUB8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1642
  { X86_TEST16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1643
  { X86_TEST32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1644
  { X86_TEST64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1645
  { X86_TEST8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1646
  { X86_XOR16i16, X86_REG_AX, CS_AC_WRITE | CS_AC_READ },
1647
  { X86_XOR32i32, X86_REG_EAX, CS_AC_WRITE | CS_AC_READ },
1648
  { X86_XOR64i32, X86_REG_RAX, CS_AC_WRITE | CS_AC_READ },
1649
  { X86_XOR8i8, X86_REG_AL, CS_AC_WRITE | CS_AC_READ },
1650
};
1651
1652
static const struct insn_reg insn_regs_intel_extra[] = {
1653
  // dummy entry, to avoid empty array
1654
  { 0, 0, 0 },
1655
#ifndef CAPSTONE_X86_REDUCE
1656
  { X86_CMOVBE_F, X86_REG_ST0, CS_AC_WRITE },
1657
  { X86_CMOVB_F, X86_REG_ST0, CS_AC_WRITE },
1658
  { X86_CMOVE_F, X86_REG_ST0, CS_AC_WRITE },
1659
  { X86_CMOVNBE_F, X86_REG_ST0, CS_AC_WRITE },
1660
  { X86_CMOVNB_F, X86_REG_ST0, CS_AC_WRITE },
1661
  { X86_CMOVNE_F, X86_REG_ST0, CS_AC_WRITE },
1662
  { X86_CMOVNP_F, X86_REG_ST0, CS_AC_WRITE },
1663
  { X86_CMOVP_F, X86_REG_ST0, CS_AC_WRITE },
1664
  // { X86_COMP_FST0r, X86_REG_ST0, CS_AC_WRITE },
1665
  // { X86_COM_FST0r, X86_REG_ST0, CS_AC_WRITE },
1666
  { X86_FNSTSW16r, X86_REG_AX, CS_AC_WRITE },
1667
  { X86_SKINIT, X86_REG_EAX, CS_AC_WRITE },
1668
  { X86_VMLOAD32, X86_REG_EAX, CS_AC_WRITE },
1669
  { X86_VMLOAD64, X86_REG_RAX, CS_AC_WRITE },
1670
  { X86_VMRUN32, X86_REG_EAX, CS_AC_WRITE },
1671
  { X86_VMRUN64, X86_REG_RAX, CS_AC_WRITE },
1672
  { X86_VMSAVE32, X86_REG_EAX, CS_AC_READ },
1673
  { X86_VMSAVE64, X86_REG_RAX, CS_AC_READ },
1674
  { X86_XCH_F, X86_REG_ST0, CS_AC_WRITE },
1675
#endif
1676
};
1677
1678
static const struct insn_reg2 insn_regs_intel2[] = {
1679
  { X86_IN16rr, X86_REG_AX, X86_REG_DX, CS_AC_WRITE, CS_AC_READ },
1680
  { X86_IN32rr, X86_REG_EAX, X86_REG_DX, CS_AC_WRITE, CS_AC_READ },
1681
  { X86_IN8rr, X86_REG_AL, X86_REG_DX, CS_AC_WRITE, CS_AC_READ },
1682
  { X86_INVLPGA32, X86_REG_EAX, X86_REG_ECX, CS_AC_READ, CS_AC_READ },
1683
  { X86_INVLPGA64, X86_REG_RAX, X86_REG_ECX, CS_AC_READ, CS_AC_READ },
1684
  { X86_OUT16rr, X86_REG_DX, X86_REG_AX, CS_AC_READ, CS_AC_READ },
1685
  { X86_OUT32rr, X86_REG_DX, X86_REG_EAX, CS_AC_READ, CS_AC_READ },
1686
  { X86_OUT8rr, X86_REG_DX, X86_REG_AL, CS_AC_READ, CS_AC_READ },
1687
};
1688
1689
static int binary_search1(const struct insn_reg *insns, unsigned int max,
1690
        unsigned int id)
1691
0
{
1692
0
  unsigned int first, last, mid;
1693
1694
0
  first = 0;
1695
0
  last = max - 1;
1696
1697
0
  if (insns[0].insn > id || insns[last].insn < id) {
1698
    // not found
1699
0
    return -1;
1700
0
  }
1701
1702
0
  while (first <= last) {
1703
0
    mid = (first + last) / 2;
1704
0
    if (insns[mid].insn < id) {
1705
0
      first = mid + 1;
1706
0
    } else if (insns[mid].insn == id) {
1707
0
      return mid;
1708
0
    } else {
1709
0
      if (mid == 0)
1710
0
        break;
1711
0
      last = mid - 1;
1712
0
    }
1713
0
  }
1714
1715
  // not found
1716
0
  return -1;
1717
0
}
1718
1719
static int binary_search2(const struct insn_reg2 *insns, unsigned int max,
1720
        unsigned int id)
1721
684k
{
1722
684k
  unsigned int first, last, mid;
1723
1724
684k
  first = 0;
1725
684k
  last = max - 1;
1726
1727
684k
  if (insns[0].insn > id || insns[last].insn < id) {
1728
    // not found
1729
501k
    return -1;
1730
501k
  }
1731
1732
715k
  while (first <= last) {
1733
547k
    mid = (first + last) / 2;
1734
547k
    if (insns[mid].insn < id) {
1735
351k
      first = mid + 1;
1736
351k
    } else if (insns[mid].insn == id) {
1737
15.9k
      return mid;
1738
180k
    } else {
1739
180k
      if (mid == 0)
1740
0
        break;
1741
180k
      last = mid - 1;
1742
180k
    }
1743
547k
  }
1744
1745
  // not found
1746
167k
  return -1;
1747
183k
}
1748
1749
void X86_build_lookup_tables(cs_struct *h)
1750
6.22k
{
1751
6.22k
  unsigned int i;
1752
6.22k
  unsigned int max = ARR_SIZE(insns);
1753
6.22k
  unsigned int id_max;
1754
1755
6.22k
  CS_ASSERT_RET(h && !h->x86_insn_lut);
1756
1757
6.22k
  id_max = insns[max - 1].id;
1758
6.22k
  h->x86_insn_lut_max = id_max;
1759
6.22k
  h->x86_insn_lut =
1760
6.22k
    (uint16_t *)cs_mem_malloc((id_max + 1) * sizeof(uint16_t));
1761
6.22k
  CS_ASSERT_RET(h->x86_insn_lut);
1762
1763
6.22k
  memset(h->x86_insn_lut, 0xff, (id_max + 1) * sizeof(uint16_t));
1764
94.2M
  for (i = 0; i < max; i++)
1765
94.2M
    h->x86_insn_lut[insns[i].id] = (uint16_t)i;
1766
1767
6.22k
  h->x86_insn_reg_lut =
1768
6.22k
    (uint32_t *)cs_mem_calloc(id_max + 1, sizeof(uint32_t));
1769
6.22k
  if (!h->x86_insn_reg_lut)
1770
0
    return;
1771
1772
547k
  for (i = 0; i < ARR_SIZE(insn_regs_intel); i++) {
1773
541k
    unsigned int insn_id = insn_regs_intel[i].insn;
1774
541k
    if (insn_id <= id_max)
1775
541k
      h->x86_insn_reg_lut[insn_id] =
1776
541k
        (h->x86_insn_reg_lut[insn_id] & 0xffff0000) |
1777
541k
        pack_insn_reg(insn_regs_intel[i].reg,
1778
541k
                insn_regs_intel[i].access);
1779
541k
  }
1780
1781
118k
  for (i = 0; i < ARR_SIZE(insn_regs_intel_extra); i++) {
1782
111k
    unsigned int insn_id = insn_regs_intel_extra[i].insn;
1783
111k
    if (insn_id && insn_id <= id_max &&
1784
105k
        !(h->x86_insn_reg_lut[insn_id] & 0xffff))
1785
105k
      h->x86_insn_reg_lut[insn_id] =
1786
105k
        (h->x86_insn_reg_lut[insn_id] & 0xffff0000) |
1787
105k
        pack_insn_reg(insn_regs_intel_extra[i].reg,
1788
105k
                insn_regs_intel_extra[i].access);
1789
111k
  }
1790
1791
566k
  for (i = 0; i < ARR_SIZE(insn_regs_att); i++) {
1792
559k
    unsigned int insn_id = insn_regs_att[i].insn;
1793
559k
    if (insn_id <= id_max)
1794
559k
      h->x86_insn_reg_lut[insn_id] =
1795
559k
        (h->x86_insn_reg_lut[insn_id] & 0x0000ffff) |
1796
559k
        ((uint32_t)pack_insn_reg(insn_regs_att[i].reg,
1797
559k
               insn_regs_att[i].access)
1798
559k
         << 16);
1799
559k
  }
1800
1801
99.5k
  for (i = 0; i < ARR_SIZE(insn_regs_att_extra); i++) {
1802
93.3k
    unsigned int insn_id = insn_regs_att_extra[i].insn;
1803
93.3k
    if (insn_id && insn_id <= id_max &&
1804
87.1k
        !(h->x86_insn_reg_lut[insn_id] >> 16))
1805
87.1k
      h->x86_insn_reg_lut[insn_id] =
1806
87.1k
        (h->x86_insn_reg_lut[insn_id] & 0x0000ffff) |
1807
87.1k
        ((uint32_t)pack_insn_reg(
1808
87.1k
           insn_regs_att_extra[i].reg,
1809
87.1k
           insn_regs_att_extra[i].access)
1810
87.1k
         << 16);
1811
93.3k
  }
1812
6.22k
}
1813
1814
// return register of given instruction id
1815
// return 0 if not found
1816
// this is to handle instructions embedding accumulate registers into AsmStrs[]
1817
x86_reg X86_insn_reg_intel(unsigned int id, enum cs_ac_type *access)
1818
0
{
1819
0
  int i;
1820
1821
0
  i = binary_search1(insn_regs_intel, ARR_SIZE(insn_regs_intel), id);
1822
0
  if (i != -1) {
1823
0
    if (access) {
1824
0
      *access = insn_regs_intel[i].access;
1825
0
    }
1826
0
    return insn_regs_intel[i].reg;
1827
0
  }
1828
1829
0
  i = binary_search1(insn_regs_intel_extra,
1830
0
         ARR_SIZE(insn_regs_intel_extra), id);
1831
0
  if (i != -1) {
1832
0
    if (access) {
1833
0
      *access = insn_regs_intel_extra[i].access;
1834
0
    }
1835
0
    return insn_regs_intel_extra[i].reg;
1836
0
  }
1837
1838
  // not found
1839
0
  return 0;
1840
0
}
1841
1842
x86_reg X86_insn_reg_intel_h(cs_struct *h, unsigned int id,
1843
           enum cs_ac_type *access)
1844
404k
{
1845
404k
  if (h && h->x86_insn_reg_lut && id <= h->x86_insn_lut_max) {
1846
404k
    uint16_t value = (uint16_t)(h->x86_insn_reg_lut[id] & 0xffff);
1847
404k
    if (value)
1848
50.5k
      return unpack_insn_reg(value, access);
1849
353k
    return 0;
1850
404k
  }
1851
1852
0
  return X86_insn_reg_intel(id, access);
1853
404k
}
1854
1855
bool X86_insn_reg_intel2(unsigned int id, x86_reg *reg1,
1856
       enum cs_ac_type *access1, x86_reg *reg2,
1857
       enum cs_ac_type *access2)
1858
353k
{
1859
353k
  int i = binary_search2(insn_regs_intel2, ARR_SIZE(insn_regs_intel2),
1860
353k
             id);
1861
353k
  if (i != -1) {
1862
8.38k
    *reg1 = insn_regs_intel2[i].reg1;
1863
8.38k
    *reg2 = insn_regs_intel2[i].reg2;
1864
8.38k
    if (access1)
1865
8.38k
      *access1 = insn_regs_intel2[i].access1;
1866
8.38k
    if (access2)
1867
8.38k
      *access2 = insn_regs_intel2[i].access2;
1868
8.38k
    return true;
1869
8.38k
  }
1870
1871
  // not found
1872
345k
  return false;
1873
353k
}
1874
1875
x86_reg X86_insn_reg_att(unsigned int id, enum cs_ac_type *access)
1876
0
{
1877
0
  int i;
1878
1879
0
  i = binary_search1(insn_regs_att, ARR_SIZE(insn_regs_att), id);
1880
0
  if (i != -1) {
1881
0
    if (access)
1882
0
      *access = insn_regs_att[i].access;
1883
0
    return insn_regs_att[i].reg;
1884
0
  }
1885
1886
0
  i = binary_search1(insn_regs_att_extra, ARR_SIZE(insn_regs_att_extra),
1887
0
         id);
1888
0
  if (i != -1) {
1889
0
    if (access)
1890
0
      *access = insn_regs_att_extra[i].access;
1891
0
    return insn_regs_att_extra[i].reg;
1892
0
  }
1893
1894
  // not found
1895
0
  return 0;
1896
0
}
1897
1898
x86_reg X86_insn_reg_att_h(cs_struct *h, unsigned int id,
1899
         enum cs_ac_type *access)
1900
353k
{
1901
353k
  if (h && h->x86_insn_reg_lut && id <= h->x86_insn_lut_max) {
1902
353k
    uint16_t value = (uint16_t)(h->x86_insn_reg_lut[id] >> 16);
1903
353k
    if (value)
1904
22.5k
      return unpack_insn_reg(value, access);
1905
331k
    return 0;
1906
353k
  }
1907
1908
0
  return X86_insn_reg_att(id, access);
1909
353k
}
1910
1911
// ATT just reuses Intel data, but with the order of registers reversed
1912
bool X86_insn_reg_att2(unsigned int id, x86_reg *reg1, enum cs_ac_type *access1,
1913
           x86_reg *reg2, enum cs_ac_type *access2)
1914
331k
{
1915
331k
  int i = binary_search2(insn_regs_intel2, ARR_SIZE(insn_regs_intel2),
1916
331k
             id);
1917
331k
  if (i != -1) {
1918
7.54k
    *reg1 = insn_regs_intel2[i].reg2;
1919
7.54k
    *reg2 = insn_regs_intel2[i].reg1;
1920
7.54k
    if (access1)
1921
7.54k
      *access1 = insn_regs_intel2[i].access2;
1922
7.54k
    if (access2)
1923
7.54k
      *access2 = insn_regs_intel2[i].access1;
1924
7.54k
    return true;
1925
7.54k
  }
1926
1927
  // not found
1928
323k
  return false;
1929
331k
}
1930
1931
// given MCInst's id, find out if this insn is valid for REPNE prefix
1932
static bool valid_repne(cs_struct *h, unsigned int opcode)
1933
21.0k
{
1934
21.0k
  unsigned int id;
1935
21.0k
  unsigned int i = find_insn_h(h, opcode);
1936
21.0k
  if (i != -1) {
1937
21.0k
    id = insns[i].mapid;
1938
21.0k
    switch (id) {
1939
12.7k
    default:
1940
12.7k
      return false;
1941
1942
232
    case X86_INS_CMPSB:
1943
232
    case X86_INS_CMPSS:
1944
587
    case X86_INS_CMPSW:
1945
762
    case X86_INS_CMPSQ:
1946
1947
904
    case X86_INS_SCASB:
1948
1.01k
    case X86_INS_SCASW:
1949
1.14k
    case X86_INS_SCASQ:
1950
1951
1.35k
    case X86_INS_MOVSB:
1952
1.35k
    case X86_INS_MOVSS:
1953
1.61k
    case X86_INS_MOVSW:
1954
1.97k
    case X86_INS_MOVSQ:
1955
1956
2.38k
    case X86_INS_LODSB:
1957
2.69k
    case X86_INS_LODSW:
1958
2.98k
    case X86_INS_LODSD:
1959
3.20k
    case X86_INS_LODSQ:
1960
1961
3.39k
    case X86_INS_STOSB:
1962
3.66k
    case X86_INS_STOSW:
1963
4.19k
    case X86_INS_STOSD:
1964
4.40k
    case X86_INS_STOSQ:
1965
1966
4.70k
    case X86_INS_INSB:
1967
4.93k
    case X86_INS_INSW:
1968
5.08k
    case X86_INS_INSD:
1969
1970
5.24k
    case X86_INS_OUTSB:
1971
5.45k
    case X86_INS_OUTSW:
1972
6.06k
    case X86_INS_OUTSD:
1973
1974
6.06k
      return true;
1975
1976
1.06k
    case X86_INS_MOVSD:
1977
1.06k
      if (opcode == X86_MOVSW) // REP MOVSB
1978
0
        return true;
1979
1.06k
      else if (opcode == X86_MOVSL) // REP MOVSD
1980
583
        return true;
1981
481
      return false;
1982
1983
843
    case X86_INS_CMPSD:
1984
843
      if (opcode == X86_CMPSL) // REP CMPSD
1985
401
        return true;
1986
442
      return false;
1987
1988
345
    case X86_INS_SCASD:
1989
345
      if (opcode == X86_SCASL) // REP SCASD
1990
345
        return true;
1991
0
      return false;
1992
21.0k
    }
1993
21.0k
  }
1994
1995
  // not found
1996
0
  return false;
1997
21.0k
}
1998
1999
// given MCInst's id, find out if this insn is valid for BND prefix
2000
// BND prefix is valid for CALL/JMP/RET
2001
#ifndef CAPSTONE_DIET
2002
static bool valid_bnd(cs_struct *h, unsigned int opcode)
2003
13.6k
{
2004
13.6k
  unsigned int id;
2005
13.6k
  unsigned int i = find_insn_h(h, opcode);
2006
13.6k
  if (i != -1) {
2007
13.6k
    id = insns[i].mapid;
2008
13.6k
    switch (id) {
2009
9.41k
    default:
2010
9.41k
      return false;
2011
2012
44
    case X86_INS_JAE:
2013
178
    case X86_INS_JA:
2014
420
    case X86_INS_JBE:
2015
924
    case X86_INS_JB:
2016
1.13k
    case X86_INS_JCXZ:
2017
1.34k
    case X86_INS_JECXZ:
2018
1.44k
    case X86_INS_JE:
2019
1.57k
    case X86_INS_JGE:
2020
1.65k
    case X86_INS_JG:
2021
1.72k
    case X86_INS_JLE:
2022
1.90k
    case X86_INS_JL:
2023
2.03k
    case X86_INS_JMP:
2024
2.70k
    case X86_INS_JNE:
2025
2.74k
    case X86_INS_JNO:
2026
2.94k
    case X86_INS_JNP:
2027
3.01k
    case X86_INS_JNS:
2028
3.09k
    case X86_INS_JO:
2029
3.37k
    case X86_INS_JP:
2030
3.39k
    case X86_INS_JRCXZ:
2031
3.56k
    case X86_INS_JS:
2032
2033
3.68k
    case X86_INS_CALL:
2034
4.03k
    case X86_INS_RET:
2035
4.18k
    case X86_INS_RETF:
2036
4.27k
    case X86_INS_RETFQ:
2037
4.27k
      return true;
2038
13.6k
    }
2039
13.6k
  }
2040
2041
  // not found
2042
0
  return false;
2043
13.6k
}
2044
#endif
2045
2046
// given MCInst's id, find out if this insn is valid for REP prefix
2047
static bool valid_rep(cs_struct *h, unsigned int opcode)
2048
16.4k
{
2049
16.4k
  unsigned int id;
2050
16.4k
  unsigned int i = find_insn_h(h, opcode);
2051
16.4k
  if (i != -1) {
2052
16.4k
    id = insns[i].mapid;
2053
16.4k
    switch (id) {
2054
12.0k
    default:
2055
12.0k
      return false;
2056
2057
119
    case X86_INS_MOVSB:
2058
285
    case X86_INS_MOVSW:
2059
563
    case X86_INS_MOVSQ:
2060
2061
761
    case X86_INS_LODSB:
2062
843
    case X86_INS_LODSW:
2063
969
    case X86_INS_LODSQ:
2064
2065
1.07k
    case X86_INS_STOSB:
2066
1.34k
    case X86_INS_STOSW:
2067
1.52k
    case X86_INS_STOSQ:
2068
2069
1.71k
    case X86_INS_INSB:
2070
1.79k
    case X86_INS_INSW:
2071
2.13k
    case X86_INS_INSD:
2072
2073
2.51k
    case X86_INS_OUTSB:
2074
2.55k
    case X86_INS_OUTSW:
2075
2.94k
    case X86_INS_OUTSD:
2076
2.94k
      return true;
2077
2078
    // following are some confused instructions, which have the same
2079
    // mnemonics in 128bit media instructions. Intel is horribly crazy!
2080
724
    case X86_INS_MOVSD:
2081
724
      if (opcode == X86_MOVSL) // REP MOVSD
2082
724
        return true;
2083
0
      return false;
2084
2085
393
    case X86_INS_LODSD:
2086
393
      if (opcode == X86_LODSL) // REP LODSD
2087
393
        return true;
2088
0
      return false;
2089
2090
328
    case X86_INS_STOSD:
2091
328
      if (opcode == X86_STOSL) // REP STOSD
2092
328
        return true;
2093
0
      return false;
2094
16.4k
    }
2095
16.4k
  }
2096
2097
  // not found
2098
0
  return false;
2099
16.4k
}
2100
2101
#ifndef CAPSTONE_DIET
2102
// given MCInst's id, find if this is a "repz ret" instruction
2103
// gcc generates "repz ret" (f3 c3) instructions in some cases as an
2104
// optimization for AMD platforms, see:
2105
// https://gcc.gnu.org/legacy-ml/gcc-patches/2003-05/msg02117.html
2106
static bool valid_ret_repz(cs_struct *h, unsigned int opcode)
2107
9.53k
{
2108
9.53k
  unsigned int id;
2109
9.53k
  unsigned int i = find_insn_h(h, opcode);
2110
2111
9.53k
  if (i != -1) {
2112
9.53k
    id = insns[i].mapid;
2113
9.53k
    return id == X86_INS_RET;
2114
9.53k
  }
2115
2116
  // not found
2117
0
  return false;
2118
9.53k
}
2119
#endif
2120
2121
// given MCInst's id, find out if this insn is valid for REPE prefix
2122
static bool valid_repe(cs_struct *h, unsigned int opcode)
2123
12.0k
{
2124
12.0k
  unsigned int id;
2125
12.0k
  unsigned int i = find_insn_h(h, opcode);
2126
12.0k
  if (i != -1) {
2127
12.0k
    id = insns[i].mapid;
2128
12.0k
    switch (id) {
2129
9.53k
    default:
2130
9.53k
      return false;
2131
2132
240
    case X86_INS_CMPSB:
2133
476
    case X86_INS_CMPSW:
2134
556
    case X86_INS_CMPSQ:
2135
2136
880
    case X86_INS_SCASB:
2137
1.10k
    case X86_INS_SCASW:
2138
1.64k
    case X86_INS_SCASQ:
2139
1.64k
      return true;
2140
2141
    // following are some confused instructions, which have the same
2142
    // mnemonics in 128bit media instructions. Intel is horribly crazy!
2143
658
    case X86_INS_CMPSD:
2144
658
      if (opcode == X86_CMPSL) // REP CMPSD
2145
658
        return true;
2146
0
      return false;
2147
2148
251
    case X86_INS_SCASD:
2149
251
      if (opcode == X86_SCASL) // REP SCASD
2150
251
        return true;
2151
0
      return false;
2152
12.0k
    }
2153
12.0k
  }
2154
2155
  // not found
2156
0
  return false;
2157
12.0k
}
2158
2159
// Given MCInst's id, find out if this insn is valid for NOTRACK prefix.
2160
// NOTRACK prefix is valid for CALL/JMP.
2161
static bool valid_notrack(cs_struct *h, unsigned int opcode)
2162
2.02k
{
2163
2.02k
  unsigned int id;
2164
2.02k
  unsigned int i = find_insn_h(h, opcode);
2165
2.02k
  if (i != -1) {
2166
2.02k
    id = insns[i].mapid;
2167
2.02k
    switch (id) {
2168
1.83k
    default:
2169
1.83k
      return false;
2170
103
    case X86_INS_CALL:
2171
194
    case X86_INS_JMP:
2172
194
      return true;
2173
2.02k
    }
2174
2.02k
  }
2175
2176
  // not found
2177
0
  return false;
2178
2.02k
}
2179
2180
#ifndef CAPSTONE_DIET
2181
// add *CX register to regs_read[] & regs_write[]
2182
static void add_cx(MCInst *MI)
2183
14.3k
{
2184
14.3k
  if (MI->csh->detail_opt) {
2185
14.3k
    x86_reg cx;
2186
2187
14.3k
    if (MI->csh->mode & CS_MODE_16)
2188
3.64k
      cx = X86_REG_CX;
2189
10.6k
    else if (MI->csh->mode & CS_MODE_32)
2190
5.83k
      cx = X86_REG_ECX;
2191
4.85k
    else // 64-bit
2192
4.85k
      cx = X86_REG_RCX;
2193
2194
14.3k
    MI->flat_insn->detail
2195
14.3k
      ->regs_read[MI->flat_insn->detail->regs_read_count] =
2196
14.3k
      cx;
2197
14.3k
    MI->flat_insn->detail->regs_read_count++;
2198
2199
14.3k
    MI->flat_insn->detail
2200
14.3k
      ->regs_write[MI->flat_insn->detail->regs_write_count] =
2201
14.3k
      cx;
2202
14.3k
    MI->flat_insn->detail->regs_write_count++;
2203
14.3k
  }
2204
14.3k
}
2205
#endif
2206
2207
// return true if we patch the mnemonic
2208
bool X86_lockrep(MCInst *MI, SStream *O)
2209
757k
{
2210
757k
  unsigned int opcode;
2211
757k
  bool res = false;
2212
2213
757k
#ifndef CAPSTONE_DIET
2214
757k
  switch (MI->xAcquireRelease) {
2215
575
  case 0xF2:
2216
575
    SStream_concat(O, "xacquire|");
2217
575
    break;
2218
1.82k
  case 0xF3:
2219
1.82k
    SStream_concat(O, "xrelease|");
2220
1.82k
    break;
2221
755k
  default:
2222
755k
    break;
2223
757k
  }
2224
757k
#endif
2225
2226
757k
  if (MI->xAcquireRelease) {
2227
2.39k
    if (MI->x86Lock) {
2228
      // Force LOCK prefix as group 0 prefix for XACQUIRE and XRELEASE if a LOCK is also present.
2229
      // This is an arbitrary choice, since there are effectively two group 0 prefixes present.
2230
      // The Intel SDM is not clear on how we should interpret group 0 in this case. It states:
2231
      // "it is only useful to include up to one prefix code from each of the four groups"
2232
      // ...and then defines instructions where both an F2/F3 and F0 are useful anyway.
2233
1.30k
      MI->x86_prefix[0] = 0xF0;
2234
1.30k
    }
2235
755k
  } else {
2236
755k
    switch (MI->x86_prefix[0]) {
2237
21.0k
    case 0xF2:
2238
21.0k
      opcode = MCInst_getOpcode(MI);
2239
21.0k
#ifndef CAPSTONE_DIET
2240
21.0k
      if (valid_repne(MI->csh, opcode)) {
2241
7.39k
        SStream_concat(O, "repne|");
2242
7.39k
        add_cx(MI);
2243
13.6k
      } else if (valid_bnd(MI->csh, opcode)) {
2244
4.27k
        SStream_concat(O, "bnd|");
2245
9.41k
      } else {
2246
        // invalid prefix
2247
9.41k
        MI->x86_prefix[0] = 0;
2248
9.41k
      }
2249
#else
2250
      if (!valid_repne(MI->csh, opcode)) {
2251
        MI->x86_prefix[0] = 0;
2252
      }
2253
#endif
2254
21.0k
      break;
2255
16.4k
    case 0xF3:
2256
16.4k
      opcode = MCInst_getOpcode(MI);
2257
16.4k
#ifndef CAPSTONE_DIET
2258
16.4k
      if (valid_rep(MI->csh, opcode)) {
2259
4.38k
        SStream_concat(O, "rep|");
2260
4.38k
        add_cx(MI);
2261
12.0k
      } else if (valid_repe(MI->csh, opcode)) {
2262
2.55k
        SStream_concat(O, "repe|");
2263
2.55k
        add_cx(MI);
2264
9.53k
      } else if (valid_ret_repz(MI->csh, opcode)) {
2265
102
        SStream_concat(O, "repz|");
2266
9.43k
      } else {
2267
        // invalid prefix
2268
9.43k
        MI->x86_prefix[0] = 0;
2269
9.43k
      }
2270
#else
2271
      if (!valid_rep(MI->csh, opcode) &&
2272
          !valid_repe(MI->csh, opcode)) {
2273
        MI->x86_prefix[0] = 0;
2274
      }
2275
#endif
2276
16.4k
      break;
2277
717k
    default:
2278
717k
      break;
2279
755k
    }
2280
755k
  }
2281
2282
  // LOCK and F2/F3 may both be present (for XACQUIRE/XRELEASE).
2283
  // There are also XRELEASEs that can be LOCKless.
2284
757k
  if (MI->x86Lock) {
2285
23.2k
    SStream_concat(O, "lock|");
2286
23.2k
  }
2287
2288
757k
  switch (MI->x86_prefix[1]) {
2289
755k
  default:
2290
755k
    break;
2291
755k
  case 0x3e:
2292
2.02k
    opcode = MCInst_getOpcode(MI);
2293
2.02k
    if (valid_notrack(MI->csh, opcode)) {
2294
194
      SStream_concat(O, "notrack|");
2295
194
    }
2296
2.02k
    break;
2297
757k
  }
2298
2299
  // copy normalized prefix[] back to x86.prefix[]
2300
757k
  if (MI->csh->detail_opt)
2301
757k
    memcpy(MI->flat_insn->detail->x86.prefix, MI->x86_prefix,
2302
757k
           ARR_SIZE(MI->x86_prefix));
2303
2304
757k
  return res;
2305
757k
}
2306
2307
void op_addReg(MCInst *MI, int reg)
2308
57.4k
{
2309
57.4k
  if (MI->csh->detail_opt) {
2310
57.4k
    MI->flat_insn->detail->x86
2311
57.4k
      .operands[MI->flat_insn->detail->x86.op_count]
2312
57.4k
      .type = X86_OP_REG;
2313
57.4k
    MI->flat_insn->detail->x86
2314
57.4k
      .operands[MI->flat_insn->detail->x86.op_count]
2315
57.4k
      .reg = reg;
2316
57.4k
    MI->flat_insn->detail->x86
2317
57.4k
      .operands[MI->flat_insn->detail->x86.op_count]
2318
57.4k
      .size = MI->csh->regsize_map[reg];
2319
57.4k
    MI->flat_insn->detail->x86.op_count++;
2320
57.4k
  }
2321
2322
57.4k
  if (MI->op1_size == 0)
2323
37.1k
    MI->op1_size = MI->csh->regsize_map[reg];
2324
57.4k
}
2325
2326
void op_addImm(MCInst *MI, int v)
2327
4.09k
{
2328
4.09k
  if (MI->csh->detail_opt) {
2329
4.09k
    MI->flat_insn->detail->x86
2330
4.09k
      .operands[MI->flat_insn->detail->x86.op_count]
2331
4.09k
      .type = X86_OP_IMM;
2332
4.09k
    MI->flat_insn->detail->x86
2333
4.09k
      .operands[MI->flat_insn->detail->x86.op_count]
2334
4.09k
      .imm = v;
2335
    // if op_count > 0, then this operand's size is taken from the destination op
2336
4.09k
    if (MI->csh->syntax != CS_OPT_SYNTAX_ATT) {
2337
4.09k
      if (MI->flat_insn->detail->x86.op_count > 0)
2338
4.09k
        MI->flat_insn->detail->x86
2339
4.09k
          .operands[MI->flat_insn->detail->x86
2340
4.09k
                .op_count]
2341
4.09k
          .size =
2342
4.09k
          MI->flat_insn->detail->x86.operands[0]
2343
4.09k
            .size;
2344
0
      else
2345
0
        MI->flat_insn->detail->x86
2346
0
          .operands[MI->flat_insn->detail->x86
2347
0
                .op_count]
2348
0
          .size = MI->imm_size;
2349
4.09k
    } else
2350
0
      MI->has_imm = true;
2351
4.09k
    MI->flat_insn->detail->x86.op_count++;
2352
4.09k
  }
2353
2354
4.09k
  if (MI->op1_size == 0)
2355
0
    MI->op1_size = MI->imm_size;
2356
4.09k
}
2357
2358
void op_addXopCC(MCInst *MI, int v)
2359
2.03k
{
2360
2.03k
  if (MI->csh->detail_opt) {
2361
2.03k
    MI->flat_insn->detail->x86.xop_cc = v;
2362
2.03k
  }
2363
2.03k
}
2364
2365
void op_addSseCC(MCInst *MI, int v)
2366
0
{
2367
0
  if (MI->csh->detail_opt) {
2368
0
    MI->flat_insn->detail->x86.sse_cc = v;
2369
0
  }
2370
0
}
2371
2372
void op_addAvxCC(MCInst *MI, int v)
2373
7.60k
{
2374
7.60k
  if (MI->csh->detail_opt) {
2375
7.60k
    MI->flat_insn->detail->x86.avx_cc = v;
2376
7.60k
  }
2377
7.60k
}
2378
2379
void op_addAvxRoundingMode(MCInst *MI, int v)
2380
1.72k
{
2381
1.72k
  if (MI->csh->detail_opt) {
2382
1.72k
    MI->flat_insn->detail->x86.avx_rm = v;
2383
1.72k
  }
2384
1.72k
}
2385
2386
// below functions supply details to X86GenAsmWriter*.inc
2387
void op_addAvxZeroOpmask(MCInst *MI)
2388
4.21k
{
2389
4.21k
  if (MI->csh->detail_opt) {
2390
    // link with the previous operand
2391
4.21k
    MI->flat_insn->detail->x86
2392
4.21k
      .operands[MI->flat_insn->detail->x86.op_count - 1]
2393
4.21k
      .avx_zero_opmask = true;
2394
4.21k
  }
2395
4.21k
}
2396
2397
void op_addAvxSae(MCInst *MI)
2398
4.21k
{
2399
4.21k
  if (MI->csh->detail_opt) {
2400
4.21k
    MI->flat_insn->detail->x86.avx_sae = true;
2401
4.21k
  }
2402
4.21k
}
2403
2404
void op_addAvxBroadcast(MCInst *MI, x86_avx_bcast v)
2405
5.98k
{
2406
5.98k
  if (MI->csh->detail_opt) {
2407
    // link with the previous operand
2408
5.98k
    MI->flat_insn->detail->x86
2409
5.98k
      .operands[MI->flat_insn->detail->x86.op_count - 1]
2410
5.98k
      .avx_bcast = v;
2411
5.98k
  }
2412
5.98k
}
2413
2414
#ifndef CAPSTONE_DIET
2415
// map instruction to its characteristics
2416
typedef struct insn_op {
2417
  uint64_t flags; // how this instruction update EFLAGS(arithmetic instructions) of FPU FLAGS(for FPU instructions)
2418
  uint8_t access[6];
2419
} insn_op;
2420
2421
static const insn_op insn_ops[] = {
2422
#ifdef CAPSTONE_X86_REDUCE
2423
#include "X86MappingInsnOp_reduce.inc"
2424
#else
2425
#include "X86MappingInsnOp.inc"
2426
#endif
2427
};
2428
2429
// given internal insn id, return operand access info
2430
const uint8_t *X86_get_op_access(cs_struct *h, unsigned int id,
2431
         uint64_t *eflags)
2432
1.76M
{
2433
1.76M
  unsigned int i = find_insn_h(h, id);
2434
1.76M
  if (i != -1) {
2435
1.76M
    *eflags = insn_ops[i].flags;
2436
1.76M
    return insn_ops[i].access;
2437
1.76M
  }
2438
2439
0
  return NULL;
2440
1.76M
}
2441
2442
void X86_reg_access(const cs_insn *insn, cs_regs regs_read,
2443
        uint8_t *regs_read_count, cs_regs regs_write,
2444
        uint8_t *regs_write_count)
2445
0
{
2446
0
  uint8_t i;
2447
0
  uint8_t read_count, write_count;
2448
0
  cs_x86 *x86 = &(insn->detail->x86);
2449
2450
0
  read_count = insn->detail->regs_read_count;
2451
0
  write_count = insn->detail->regs_write_count;
2452
2453
  // implicit registers
2454
0
  memcpy(regs_read, insn->detail->regs_read,
2455
0
         read_count * sizeof(insn->detail->regs_read[0]));
2456
0
  memcpy(regs_write, insn->detail->regs_write,
2457
0
         write_count * sizeof(insn->detail->regs_write[0]));
2458
2459
  // explicit registers
2460
0
  for (i = 0; i < x86->op_count; i++) {
2461
0
    cs_x86_op *op = &(x86->operands[i]);
2462
0
    switch ((int)op->type) {
2463
0
    case X86_OP_REG:
2464
0
      if ((op->access & CS_AC_READ) &&
2465
0
          !arr_exist(regs_read, read_count, op->reg)) {
2466
0
        regs_read[read_count] = op->reg;
2467
0
        read_count++;
2468
0
      }
2469
0
      if ((op->access & CS_AC_WRITE) &&
2470
0
          !arr_exist(regs_write, write_count, op->reg)) {
2471
0
        regs_write[write_count] = op->reg;
2472
0
        write_count++;
2473
0
      }
2474
0
      break;
2475
0
    case X86_OP_MEM:
2476
      // registers appeared in memory references always being read
2477
0
      if ((op->mem.segment != X86_REG_INVALID)) {
2478
0
        regs_read[read_count] = op->mem.segment;
2479
0
        read_count++;
2480
0
      }
2481
0
      if ((op->mem.base != X86_REG_INVALID) &&
2482
0
          !arr_exist(regs_read, read_count, op->mem.base)) {
2483
0
        regs_read[read_count] = op->mem.base;
2484
0
        read_count++;
2485
0
      }
2486
0
      if ((op->mem.index != X86_REG_INVALID) &&
2487
0
          !arr_exist(regs_read, read_count, op->mem.index)) {
2488
0
        regs_read[read_count] = op->mem.index;
2489
0
        read_count++;
2490
0
      }
2491
0
    default:
2492
0
      break;
2493
0
    }
2494
0
  }
2495
2496
0
  *regs_read_count = read_count;
2497
0
  *regs_write_count = write_count;
2498
0
}
2499
#endif
2500
2501
// map immediate size to instruction id
2502
// this array is sorted for binary searching
2503
static const struct size_id {
2504
  uint8_t enc_size;
2505
  uint8_t size;
2506
  uint16_t id;
2507
} x86_imm_size[] = {
2508
#include "X86ImmSize.inc"
2509
};
2510
2511
// given the instruction name, return the size of its immediate operand (or 0)
2512
uint8_t X86_immediate_size(unsigned int id, uint8_t *enc_size)
2513
131k
{
2514
  // binary searching since the IDs are sorted in order
2515
131k
  unsigned int left, right, m;
2516
2517
131k
  right = ARR_SIZE(x86_imm_size) - 1;
2518
2519
131k
  if (id < x86_imm_size[0].id || id > x86_imm_size[right].id)
2520
    // not found
2521
0
    return 0;
2522
2523
131k
  left = 0;
2524
2525
1.01M
  while (left <= right) {
2526
966k
    m = (left + right) / 2;
2527
966k
    if (id == x86_imm_size[m].id) {
2528
85.3k
      if (enc_size != NULL)
2529
84.6k
        *enc_size = x86_imm_size[m].enc_size;
2530
2531
85.3k
      return x86_imm_size[m].size;
2532
85.3k
    }
2533
2534
881k
    if (id > x86_imm_size[m].id)
2535
422k
      left = m + 1;
2536
458k
    else {
2537
458k
      if (m == 0)
2538
0
        break;
2539
458k
      right = m - 1;
2540
458k
    }
2541
881k
  }
2542
2543
  // not found
2544
46.0k
  return 0;
2545
131k
}
2546
2547
#define GET_REGINFO_ENUM
2548
#include "X86GenRegisterInfo.inc"
2549
2550
// map internal register id to public register id
2551
static const struct register_map {
2552
  unsigned short id;
2553
  unsigned short pub_id;
2554
} reg_map[] = {
2555
  // first dummy map
2556
  { 0, 0 },
2557
#include "X86MappingReg.inc"
2558
};
2559
2560
// return 0 on invalid input, or public register ID otherwise
2561
// NOTE: reg_map is sorted in order of internal register
2562
unsigned short X86_register_map(unsigned short id)
2563
1.88M
{
2564
1.88M
  if (id < ARR_SIZE(reg_map))
2565
1.88M
    return reg_map[id].pub_id;
2566
2567
0
  return 0;
2568
1.88M
}
2569
2570
/// The post-printer function. Used to fixup flaws in the disassembly information
2571
/// of certain instructions.
2572
void X86_postprinter(csh handle, cs_insn *insn, SStream *mnem, MCInst *mci)
2573
757k
{
2574
757k
  if (!insn || !insn->detail) {
2575
0
    return;
2576
0
  }
2577
757k
  switch (insn->id) {
2578
750k
  default:
2579
750k
    break;
2580
750k
  case X86_INS_RCL:
2581
    // Addmissing 1 immediate
2582
7.49k
    if (insn->detail->x86.op_count > 1) {
2583
6.76k
      return;
2584
6.76k
    }
2585
732
    insn->detail->x86.operands[1].imm = 1;
2586
732
    insn->detail->x86.operands[1].type = X86_OP_IMM;
2587
732
    insn->detail->x86.operands[1].access = CS_AC_READ;
2588
732
    insn->detail->x86.op_count++;
2589
732
    break;
2590
757k
  }
2591
757k
}
2592
2593
#endif