Coverage Report

Created: 2025-07-18 06:43

/src/capstonenext/arch/AArch64/AArch64Mapping.c
Line
Count
Source (jump to first uncovered line)
1
/* Capstone Disassembly Engine */
2
/* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2019 */
3
4
#ifdef CAPSTONE_HAS_AARCH64
5
6
#include <stdio.h> // debug
7
#include <string.h>
8
9
#include "capstone/aarch64.h"
10
11
#include "../../cs_simple_types.h"
12
#include "../../Mapping.h"
13
#include "../../MathExtras.h"
14
#include "../../utils.h"
15
16
#include "AArch64AddressingModes.h"
17
#include "AArch64BaseInfo.h"
18
#include "AArch64DisassemblerExtension.h"
19
#include "AArch64Linkage.h"
20
#include "AArch64Mapping.h"
21
22
3.16k
#define CHAR(c) #c[0]
23
24
7.23k
static float aarch64_exact_fp_to_fp(aarch64_exactfpimm exact) {
25
7.23k
  switch (exact) {
26
0
  default:
27
0
    CS_ASSERT(0 && "Not handled.");
28
0
    return 999.0;
29
530
  case AARCH64_EXACTFPIMM_HALF:
30
530
    return 0.5;
31
789
  case AARCH64_EXACTFPIMM_ONE:
32
789
    return 1.0;
33
1.21k
  case AARCH64_EXACTFPIMM_TWO:
34
1.21k
    return 2.0;
35
4.70k
  case AARCH64_EXACTFPIMM_ZERO:
36
4.70k
    return 0.0;
37
7.23k
  }
38
7.23k
}
39
40
#ifndef CAPSTONE_DIET
41
static const aarch64_reg aarch64_flag_regs[] = {
42
  AARCH64_REG_NZCV,
43
};
44
45
static const aarch64_sysreg aarch64_flag_sys_regs[] = {
46
  AARCH64_SYSREG_NZCV, AARCH64_SYSREG_PMOVSCLR_EL0,
47
  AARCH64_SYSREG_PMOVSSET_EL0, AARCH64_SYSREG_SPMOVSCLR_EL0,
48
  AARCH64_SYSREG_SPMOVSSET_EL0
49
};
50
#endif // CAPSTONE_DIET
51
52
static AArch64Layout_VectorLayout sme_reg_to_vas(aarch64_reg reg)
53
0
{
54
0
  switch (reg) {
55
0
  default:
56
0
    return AARCH64LAYOUT_INVALID;
57
0
  case AARCH64_REG_ZAB0:
58
0
    return AARCH64LAYOUT_VL_B;
59
0
  case AARCH64_REG_ZAH0:
60
0
  case AARCH64_REG_ZAH1:
61
0
    return AARCH64LAYOUT_VL_H;
62
0
  case AARCH64_REG_ZAS0:
63
0
  case AARCH64_REG_ZAS1:
64
0
  case AARCH64_REG_ZAS2:
65
0
  case AARCH64_REG_ZAS3:
66
0
    return AARCH64LAYOUT_VL_S;
67
0
  case AARCH64_REG_ZAD0:
68
0
  case AARCH64_REG_ZAD1:
69
0
  case AARCH64_REG_ZAD2:
70
0
  case AARCH64_REG_ZAD3:
71
0
  case AARCH64_REG_ZAD4:
72
0
  case AARCH64_REG_ZAD5:
73
0
  case AARCH64_REG_ZAD6:
74
0
  case AARCH64_REG_ZAD7:
75
0
    return AARCH64LAYOUT_VL_D;
76
0
  case AARCH64_REG_ZAQ0:
77
0
  case AARCH64_REG_ZAQ1:
78
0
  case AARCH64_REG_ZAQ2:
79
0
  case AARCH64_REG_ZAQ3:
80
0
  case AARCH64_REG_ZAQ4:
81
0
  case AARCH64_REG_ZAQ5:
82
0
  case AARCH64_REG_ZAQ6:
83
0
  case AARCH64_REG_ZAQ7:
84
0
  case AARCH64_REG_ZAQ8:
85
0
  case AARCH64_REG_ZAQ9:
86
0
  case AARCH64_REG_ZAQ10:
87
0
  case AARCH64_REG_ZAQ11:
88
0
  case AARCH64_REG_ZAQ12:
89
0
  case AARCH64_REG_ZAQ13:
90
0
  case AARCH64_REG_ZAQ14:
91
0
  case AARCH64_REG_ZAQ15:
92
0
    return AARCH64LAYOUT_VL_Q;
93
0
  case AARCH64_REG_ZA:
94
0
    return AARCH64LAYOUT_VL_COMPLETE;
95
0
  }
96
0
}
97
98
void AArch64_init_mri(MCRegisterInfo *MRI)
99
10.8k
{
100
10.8k
  MCRegisterInfo_InitMCRegisterInfo(
101
10.8k
    MRI, AArch64RegDesc, AARCH64_REG_ENDING, 0, 0,
102
10.8k
    AArch64MCRegisterClasses, ARR_SIZE(AArch64MCRegisterClasses), 0,
103
10.8k
    0, AArch64RegDiffLists, 0, AArch64SubRegIdxLists,
104
10.8k
    ARR_SIZE(AArch64SubRegIdxLists), 0);
105
10.8k
}
106
107
108
/// Sets up a new SME matrix operand at the currently active detail operand.
109
static void setup_sme_operand(MCInst *MI)
110
26.6k
{
111
26.6k
  if (!detail_is_set(MI))
112
0
    return;
113
114
26.6k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_SME;
115
26.6k
  AArch64_get_detail_op(MI, 0)->sme.type = AARCH64_SME_OP_INVALID;
116
26.6k
  AArch64_get_detail_op(MI, 0)->sme.tile = AARCH64_REG_INVALID;
117
26.6k
  AArch64_get_detail_op(MI, 0)->sme.slice_reg = AARCH64_REG_INVALID;
118
26.6k
  AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm = AARCH64_SLICE_IMM_INVALID;
119
26.6k
  AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.first = AARCH64_SLICE_IMM_RANGE_INVALID;
120
26.6k
  AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.offset = AARCH64_SLICE_IMM_RANGE_INVALID;
121
26.6k
}
122
123
static void setup_pred_operand(MCInst *MI)
124
61.2k
{
125
61.2k
  if (!detail_is_set(MI))
126
0
    return;
127
128
61.2k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_PRED;
129
61.2k
  AArch64_get_detail_op(MI, 0)->pred.imm_index = -1;
130
61.2k
}
131
132
const insn_map aarch64_insns[] = {
133
#include "AArch64GenCSMappingInsn.inc"
134
};
135
136
static const name_map insn_alias_mnem_map[] = {
137
#include "AArch64GenCSAliasMnemMap.inc"
138
  { AARCH64_INS_ALIAS_CFP, "cfp" },
139
  { AARCH64_INS_ALIAS_DVP, "dvp" },
140
  { AARCH64_INS_ALIAS_COSP, "cosp" },
141
  { AARCH64_INS_ALIAS_CPP, "cpp" },
142
  { AARCH64_INS_ALIAS_IC, "ic" },
143
  { AARCH64_INS_ALIAS_DC, "dc" },
144
  { AARCH64_INS_ALIAS_AT, "at" },
145
  { AARCH64_INS_ALIAS_TLBI, "tlbi" },
146
  { AARCH64_INS_ALIAS_TLBIP, "tlbip" },
147
  { AARCH64_INS_ALIAS_RPRFM, "rprfm" },
148
  { AARCH64_INS_ALIAS_LSL, "lsl" },
149
  { AARCH64_INS_ALIAS_SBFX, "sbfx" },
150
  { AARCH64_INS_ALIAS_UBFX, "ubfx" },
151
  { AARCH64_INS_ALIAS_SBFIZ, "sbfiz" },
152
  { AARCH64_INS_ALIAS_UBFIZ, "ubfiz" },
153
  { AARCH64_INS_ALIAS_BFC, "bfc" },
154
  { AARCH64_INS_ALIAS_BFI, "bfi" },
155
  { AARCH64_INS_ALIAS_BFXIL, "bfxil" },
156
  { AARCH64_INS_ALIAS_END, NULL },
157
};
158
159
static const char *get_custom_reg_alias(unsigned reg)
160
67.8k
{
161
67.8k
  switch (reg) {
162
680
  case AARCH64_REG_X29:
163
680
    return "fp";
164
2.23k
  case AARCH64_REG_X30:
165
2.23k
    return "lr";
166
67.8k
  }
167
64.9k
  return NULL;
168
67.8k
}
169
170
/// Very annoyingly LLVM hard codes the vector layout post-fixes into the asm string.
171
/// In this function we check for these cases and add the vectorlayout/arrangement
172
/// specifier.
173
void AArch64_add_vas(MCInst *MI, const SStream *OS)
174
306k
{
175
306k
  if (!detail_is_set(MI)) {
176
0
    return;
177
0
  }
178
179
306k
  if (AArch64_get_detail(MI)->op_count == 0) {
180
650
    return;
181
650
  }
182
305k
  if (MCInst_getOpcode(MI) == AArch64_MUL53HI || MCInst_getOpcode(MI) == AArch64_MUL53LO) {
183
    // Proprietary Apple instrucions.
184
0
    AArch64_get_detail(MI)->operands[0].vas = AARCH64LAYOUT_VL_2D;
185
0
    AArch64_get_detail(MI)->operands[1].vas = AARCH64LAYOUT_VL_2D;
186
0
    return;
187
0
  }
188
189
  // Search for r".[0-9]{1,2}[bhsdq]\W"
190
  // with poor mans regex
191
305k
  const char *vl_ptr = strchr(OS->buffer, '.');
192
653k
  while (vl_ptr) {
193
    // Number after dot?
194
348k
    unsigned num = 0;
195
348k
    if (strchr("1248", vl_ptr[1])) {
196
68.8k
      num = atoi(vl_ptr + 1);
197
68.8k
      vl_ptr = num > 9 ? vl_ptr + 3 : vl_ptr + 2;
198
279k
    } else {
199
279k
      vl_ptr++;
200
279k
    }
201
202
    // Layout letter
203
348k
    char letter = '\0';
204
348k
    if (strchr("bhsdq", vl_ptr[0])) {
205
336k
      letter = vl_ptr[0];
206
336k
    }
207
348k
    if (!letter) {
208
11.2k
      goto next_dot_continue;
209
11.2k
    }
210
211
336k
    AArch64Layout_VectorLayout vl = AARCH64LAYOUT_INVALID;
212
336k
    switch (letter) {
213
0
    default:
214
0
      CS_ASSERT_RET(0 && "Unhandled vector layout letter.");
215
0
      return;
216
76.2k
    case 'b':
217
76.2k
      vl = AARCH64LAYOUT_VL_B;
218
76.2k
      break;
219
83.2k
    case 'h':
220
83.2k
      vl = AARCH64LAYOUT_VL_H;
221
83.2k
      break;
222
79.5k
    case 's':
223
79.5k
      vl = AARCH64LAYOUT_VL_S;
224
79.5k
      break;
225
93.2k
    case 'd':
226
93.2k
      vl = AARCH64LAYOUT_VL_D;
227
93.2k
      break;
228
4.38k
    case 'q':
229
4.38k
      vl = AARCH64LAYOUT_VL_Q;
230
4.38k
      break;
231
336k
    }
232
336k
    vl |= (num << 8);
233
234
    // Determine op index by searching for trailing commata after op string
235
336k
    uint32_t op_idx = 0;
236
336k
    const char *comma_ptr = strchr(OS->buffer, ',');
237
336k
    ;
238
733k
    while (comma_ptr && comma_ptr < vl_ptr) {
239
396k
      ++op_idx;
240
396k
      comma_ptr = strchr(comma_ptr + 1, ',');
241
396k
    }
242
336k
    if (!comma_ptr) {
243
      // Last op doesn't have a trailing commata.
244
47.8k
      op_idx = AArch64_get_detail(MI)->op_count - 1;
245
47.8k
    }
246
336k
    if (op_idx >= AArch64_get_detail(MI)->op_count) {
247
      // A memory operand with a commata in [base, dist]
248
11.8k
      op_idx = AArch64_get_detail(MI)->op_count - 1;
249
11.8k
    }
250
251
    // Search for the operand this one belongs to.
252
336k
    cs_aarch64_op *op = &AArch64_get_detail(MI)->operands[op_idx];
253
336k
    if ((op->type != AARCH64_OP_REG &&
254
336k
         op->type != AARCH64_OP_SME) ||
255
336k
        op->vas != AARCH64LAYOUT_INVALID) {
256
280k
      goto next_dot_continue;
257
280k
    }
258
56.3k
    op->vas = vl;
259
260
348k
next_dot_continue:
261
348k
    vl_ptr = strchr(vl_ptr + 1, '.');
262
348k
  }
263
305k
}
264
265
const char *AArch64_reg_name(csh handle, unsigned int reg)
266
67.8k
{
267
67.8k
  int syntax_opt = ((cs_struct *)(uintptr_t)handle)->syntax;
268
67.8k
  const char *alias = get_custom_reg_alias(reg);
269
67.8k
  if ((syntax_opt & CS_OPT_SYNTAX_CS_REG_ALIAS) && alias)
270
0
    return alias;
271
272
67.8k
  if (((cs_struct *)(uintptr_t)handle)->syntax &
273
67.8k
      CS_OPT_SYNTAX_NOREGNAME) {
274
0
    return AArch64_LLVM_getRegisterName(reg, AArch64_NoRegAltName);
275
0
  }
276
  // TODO Add options for the other register names
277
67.8k
  return AArch64_LLVM_getRegisterName(reg, AArch64_NoRegAltName);
278
67.8k
}
279
280
void AArch64_setup_op(cs_aarch64_op *op)
281
5.03M
{
282
5.03M
  memset(op, 0, sizeof(cs_aarch64_op));
283
5.03M
  op->type = AARCH64_OP_INVALID;
284
5.03M
  op->vector_index = -1;
285
5.03M
}
286
287
void AArch64_init_cs_detail(MCInst *MI)
288
313k
{
289
313k
  if (detail_is_set(MI)) {
290
313k
    memset(get_detail(MI), 0,
291
313k
           offsetof(cs_detail, aarch64) + sizeof(cs_aarch64));
292
5.33M
    for (int i = 0; i < ARR_SIZE(AArch64_get_detail(MI)->operands);
293
5.01M
         i++)
294
5.01M
      AArch64_setup_op(&AArch64_get_detail(MI)->operands[i]);
295
313k
    AArch64_get_detail(MI)->cc = AArch64CC_Invalid;
296
313k
  }
297
313k
}
298
299
/// Unfortunately, the AARCH64 definitions do not indicate in any way
300
/// (exception are the instruction identifiers), if memory accesses
301
/// is post- or pre-indexed.
302
/// So the only generic way to determine, if the memory access is in
303
/// post-indexed addressing mode, is by search for "<membase>], #<memdisp>" in
304
/// @p OS.
305
/// Searching the asm string to determine such a property is enormously ugly
306
/// and wastes resources.
307
/// Sorry, I know and do feel bad about it. But for now it works.
308
static bool AArch64_check_post_index_am(const MCInst *MI, const SStream *OS)
309
306k
{
310
306k
  if (AArch64_get_detail(MI)->post_index) {
311
0
    return true;
312
0
  }
313
306k
  cs_aarch64_op *memop = NULL;
314
1.07M
  for (int i = 0; i < AArch64_get_detail(MI)->op_count; ++i) {
315
877k
    if (AArch64_get_detail(MI)->operands[i].type & CS_OP_MEM) {
316
108k
      memop = &AArch64_get_detail(MI)->operands[i];
317
108k
      break;
318
108k
    }
319
877k
  }
320
306k
  if (!memop)
321
198k
    return false;
322
108k
  if (memop->mem.base == AARCH64_REG_INVALID) {
323
    // Load/Store from/to label. Has no register base.
324
5.09k
    return false;
325
5.09k
  }
326
103k
  const char *membase = AArch64_LLVM_getRegisterName(
327
103k
    memop->mem.base, AArch64_NoRegAltName);
328
103k
  int64_t memdisp = memop->mem.disp;
329
103k
  SStream pattern = { 0 };
330
103k
  SStream_concat(&pattern, membase);
331
103k
  SStream_concat(&pattern, "], ");
332
103k
  printInt32Bang(&pattern, memdisp);
333
103k
  return strstr(OS->buffer, pattern.buffer) != NULL;
334
108k
}
335
336
static void AArch64_check_updates_flags(MCInst *MI)
337
306k
{
338
306k
#ifndef CAPSTONE_DIET
339
306k
  if (!detail_is_set(MI))
340
0
    return;
341
306k
  cs_detail *detail = get_detail(MI);
342
  // Implicitly written registers
343
338k
  for (int i = 0; i < detail->regs_write_count; ++i) {
344
48.7k
    if (detail->regs_write[i] == 0)
345
0
      break;
346
80.4k
    for (int j = 0; j < ARR_SIZE(aarch64_flag_regs); ++j) {
347
48.7k
      if (detail->regs_write[i] == aarch64_flag_regs[j]) {
348
16.9k
        detail->aarch64.update_flags = true;
349
16.9k
        return;
350
16.9k
      }
351
48.7k
    }
352
48.7k
  }
353
1.13M
  for (int i = 0; i < detail->aarch64.op_count; ++i) {
354
844k
    if (detail->aarch64.operands[i].type == AARCH64_OP_SYSREG &&
355
844k
        detail->aarch64.operands[i].sysop.sub_type ==
356
10.0k
          AARCH64_OP_REG_MSR) {
357
27.2k
      for (int j = 0; j < ARR_SIZE(aarch64_flag_sys_regs);
358
22.6k
           ++j)
359
22.7k
        if (detail->aarch64.operands[i]
360
22.7k
              .sysop.reg.sysreg ==
361
22.7k
            aarch64_flag_sys_regs[j]) {
362
91
          detail->aarch64.update_flags = true;
363
91
          return;
364
91
        }
365
840k
    } else if (detail->aarch64.operands[i].type == AARCH64_OP_REG &&
366
840k
         detail->aarch64.operands[i].access & CS_AC_WRITE) {
367
510k
      for (int j = 0; j < ARR_SIZE(aarch64_flag_regs); ++j)
368
255k
        if (detail->aarch64.operands[i].reg ==
369
255k
            aarch64_flag_regs[j]) {
370
0
          detail->aarch64.update_flags = true;
371
0
          return;
372
0
        }
373
255k
    }
374
844k
  }
375
289k
#endif // CAPSTONE_DIET
376
289k
}
377
378
230
static aarch64_shifter id_to_shifter(unsigned Opcode) {
379
230
  switch (Opcode) {
380
0
  default:
381
0
    return AARCH64_SFT_INVALID;
382
36
  case AArch64_RORVXr:
383
46
  case AArch64_RORVWr:
384
46
    return AARCH64_SFT_ROR_REG;
385
77
  case AArch64_LSRVXr:
386
95
  case AArch64_LSRVWr:
387
95
    return AARCH64_SFT_LSR_REG;
388
10
  case AArch64_LSLVXr:
389
60
  case AArch64_LSLVWr:
390
60
    return AARCH64_SFT_LSL_REG;
391
11
  case AArch64_ASRVXr:
392
29
  case AArch64_ASRVWr:
393
29
    return AARCH64_SFT_ASR_REG;
394
230
  }
395
230
}
396
397
static void add_non_alias_details(MCInst *MI)
398
261k
{
399
261k
  unsigned Opcode = MCInst_getOpcode(MI);
400
261k
  switch (Opcode) {
401
249k
  default:
402
249k
    break;
403
249k
  case AArch64_RORVXr:
404
46
  case AArch64_RORVWr:
405
123
  case AArch64_LSRVXr:
406
141
  case AArch64_LSRVWr:
407
151
  case AArch64_LSLVXr:
408
201
  case AArch64_LSLVWr:
409
212
  case AArch64_ASRVXr:
410
230
  case AArch64_ASRVWr:
411
230
    if (AArch64_get_detail(MI)->op_count != 3) {
412
0
      return;
413
0
    }
414
230
    CS_ASSERT_RET(AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_REG);
415
416
    // The shift by register instructions don't set the shift value properly.
417
    // Correct it here.
418
230
    uint64_t shift = AArch64_get_detail_op(MI, -1)->reg;
419
230
    cs_aarch64_op *op1 = AArch64_get_detail_op(MI, -2);
420
230
    op1->shift.type = id_to_shifter(Opcode);
421
230
    op1->shift.value = shift;
422
230
    AArch64_dec_op_count(MI);
423
230
    break;
424
148
  case AArch64_FCMPDri:
425
392
  case AArch64_FCMPEDri:
426
1.53k
  case AArch64_FCMPEHri:
427
1.57k
  case AArch64_FCMPESri:
428
2.53k
  case AArch64_FCMPHri:
429
2.68k
  case AArch64_FCMPSri:
430
2.68k
    AArch64_insert_detail_op_reg_at(MI, -1, AARCH64_REG_XZR,
431
2.68k
            CS_AC_READ);
432
2.68k
    break;
433
171
  case AArch64_CMEQv16i8rz:
434
238
  case AArch64_CMEQv1i64rz:
435
273
  case AArch64_CMEQv2i32rz:
436
310
  case AArch64_CMEQv2i64rz:
437
349
  case AArch64_CMEQv4i16rz:
438
391
  case AArch64_CMEQv4i32rz:
439
623
  case AArch64_CMEQv8i16rz:
440
656
  case AArch64_CMEQv8i8rz:
441
722
  case AArch64_CMGEv16i8rz:
442
767
  case AArch64_CMGEv1i64rz:
443
868
  case AArch64_CMGEv2i32rz:
444
1.22k
  case AArch64_CMGEv2i64rz:
445
1.29k
  case AArch64_CMGEv4i16rz:
446
1.30k
  case AArch64_CMGEv4i32rz:
447
1.40k
  case AArch64_CMGEv8i16rz:
448
1.60k
  case AArch64_CMGEv8i8rz:
449
1.72k
  case AArch64_CMGTv16i8rz:
450
1.80k
  case AArch64_CMGTv1i64rz:
451
1.81k
  case AArch64_CMGTv2i32rz:
452
2.29k
  case AArch64_CMGTv2i64rz:
453
2.30k
  case AArch64_CMGTv4i16rz:
454
2.40k
  case AArch64_CMGTv4i32rz:
455
2.52k
  case AArch64_CMGTv8i16rz:
456
2.69k
  case AArch64_CMGTv8i8rz:
457
2.82k
  case AArch64_CMLEv16i8rz:
458
2.86k
  case AArch64_CMLEv1i64rz:
459
2.92k
  case AArch64_CMLEv2i32rz:
460
2.94k
  case AArch64_CMLEv2i64rz:
461
2.97k
  case AArch64_CMLEv4i16rz:
462
3.02k
  case AArch64_CMLEv4i32rz:
463
3.10k
  case AArch64_CMLEv8i16rz:
464
3.16k
  case AArch64_CMLEv8i8rz:
465
3.23k
  case AArch64_CMLTv16i8rz:
466
3.25k
  case AArch64_CMLTv1i64rz:
467
3.32k
  case AArch64_CMLTv2i32rz:
468
3.88k
  case AArch64_CMLTv2i64rz:
469
3.89k
  case AArch64_CMLTv4i16rz:
470
3.96k
  case AArch64_CMLTv4i32rz:
471
4.04k
  case AArch64_CMLTv8i16rz:
472
4.08k
  case AArch64_CMLTv8i8rz:
473
4.08k
    AArch64_insert_detail_op_imm_at(MI, -1, 0);
474
4.08k
    break;
475
389
  case AArch64_FCMEQ_PPzZ0_D:
476
456
  case AArch64_FCMEQ_PPzZ0_H:
477
474
  case AArch64_FCMEQ_PPzZ0_S:
478
570
  case AArch64_FCMEQv1i16rz:
479
593
  case AArch64_FCMEQv1i32rz:
480
705
  case AArch64_FCMEQv1i64rz:
481
746
  case AArch64_FCMEQv2i32rz:
482
780
  case AArch64_FCMEQv2i64rz:
483
794
  case AArch64_FCMEQv4i16rz:
484
833
  case AArch64_FCMEQv4i32rz:
485
858
  case AArch64_FCMEQv8i16rz:
486
1.03k
  case AArch64_FCMGE_PPzZ0_D:
487
1.05k
  case AArch64_FCMGE_PPzZ0_H:
488
1.08k
  case AArch64_FCMGE_PPzZ0_S:
489
1.66k
  case AArch64_FCMGEv1i16rz:
490
1.67k
  case AArch64_FCMGEv1i32rz:
491
1.68k
  case AArch64_FCMGEv1i64rz:
492
2.26k
  case AArch64_FCMGEv2i32rz:
493
2.29k
  case AArch64_FCMGEv2i64rz:
494
2.33k
  case AArch64_FCMGEv4i16rz:
495
2.41k
  case AArch64_FCMGEv4i32rz:
496
2.42k
  case AArch64_FCMGEv8i16rz:
497
2.47k
  case AArch64_FCMGT_PPzZ0_D:
498
2.50k
  case AArch64_FCMGT_PPzZ0_H:
499
2.54k
  case AArch64_FCMGT_PPzZ0_S:
500
2.56k
  case AArch64_FCMGTv1i16rz:
501
2.69k
  case AArch64_FCMGTv1i32rz:
502
2.76k
  case AArch64_FCMGTv1i64rz:
503
2.92k
  case AArch64_FCMGTv2i32rz:
504
3.02k
  case AArch64_FCMGTv2i64rz:
505
3.03k
  case AArch64_FCMGTv4i16rz:
506
3.10k
  case AArch64_FCMGTv4i32rz:
507
3.22k
  case AArch64_FCMGTv8i16rz:
508
3.29k
  case AArch64_FCMLE_PPzZ0_D:
509
3.31k
  case AArch64_FCMLE_PPzZ0_H:
510
3.38k
  case AArch64_FCMLE_PPzZ0_S:
511
3.39k
  case AArch64_FCMLEv1i16rz:
512
3.40k
  case AArch64_FCMLEv1i32rz:
513
3.44k
  case AArch64_FCMLEv1i64rz:
514
3.71k
  case AArch64_FCMLEv2i32rz:
515
3.72k
  case AArch64_FCMLEv2i64rz:
516
3.73k
  case AArch64_FCMLEv4i16rz:
517
3.75k
  case AArch64_FCMLEv4i32rz:
518
3.77k
  case AArch64_FCMLEv8i16rz:
519
3.80k
  case AArch64_FCMLT_PPzZ0_D:
520
3.87k
  case AArch64_FCMLT_PPzZ0_H:
521
3.89k
  case AArch64_FCMLT_PPzZ0_S:
522
3.93k
  case AArch64_FCMLTv1i16rz:
523
3.94k
  case AArch64_FCMLTv1i32rz:
524
3.96k
  case AArch64_FCMLTv1i64rz:
525
4.01k
  case AArch64_FCMLTv2i32rz:
526
4.03k
  case AArch64_FCMLTv2i64rz:
527
4.10k
  case AArch64_FCMLTv4i16rz:
528
4.17k
  case AArch64_FCMLTv4i32rz:
529
4.40k
  case AArch64_FCMLTv8i16rz:
530
4.42k
  case AArch64_FCMNE_PPzZ0_D:
531
4.43k
  case AArch64_FCMNE_PPzZ0_H:
532
4.56k
  case AArch64_FCMNE_PPzZ0_S: {
533
4.56k
    aarch64_sysop sysop = { 0 };
534
4.56k
    sysop.imm.exactfpimm = AARCH64_EXACTFPIMM_ZERO;
535
4.56k
    sysop.sub_type = AARCH64_OP_EXACTFPIMM;
536
4.56k
    AArch64_insert_detail_op_sys(MI, -1, sysop, AARCH64_OP_SYSIMM);
537
4.56k
    break;
538
4.43k
  }
539
261k
  }
540
261k
}
541
542
#define ADD_ZA0_S \
543
288
      { aarch64_op_sme za0_op = { \
544
288
        .type = AARCH64_SME_OP_TILE, \
545
288
        .tile = AARCH64_REG_ZAS0, \
546
288
        .slice_reg = AARCH64_REG_INVALID, \
547
288
        .slice_offset = { -1 }, \
548
288
        .has_range_offset = false, \
549
288
        .is_vertical = false, \
550
288
      }; \
551
288
      AArch64_insert_detail_op_sme(MI, -1, za0_op); \
552
288
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
553
288
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
554
288
      }
555
#define ADD_ZA1_S \
556
815
      { aarch64_op_sme za1_op = { \
557
815
        .type = AARCH64_SME_OP_TILE, \
558
815
        .tile = AARCH64_REG_ZAS1, \
559
815
        .slice_reg = AARCH64_REG_INVALID, \
560
815
        .slice_offset = { -1 }, \
561
815
        .has_range_offset = false, \
562
815
        .is_vertical = false, \
563
815
      }; \
564
815
      AArch64_insert_detail_op_sme(MI, -1, za1_op); \
565
815
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
566
815
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
567
815
      }
568
#define ADD_ZA2_S \
569
926
      { aarch64_op_sme za2_op = { \
570
926
        .type = AARCH64_SME_OP_TILE, \
571
926
        .tile = AARCH64_REG_ZAS2, \
572
926
        .slice_reg = AARCH64_REG_INVALID, \
573
926
        .slice_offset = { -1 }, \
574
926
        .has_range_offset = false, \
575
926
        .is_vertical = false, \
576
926
      }; \
577
926
      AArch64_insert_detail_op_sme(MI, -1, za2_op); \
578
926
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
579
926
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
580
926
      }
581
#define ADD_ZA3_S \
582
687
      { aarch64_op_sme za3_op = { \
583
687
        .type = AARCH64_SME_OP_TILE, \
584
687
        .tile = AARCH64_REG_ZAS3, \
585
687
        .slice_reg = AARCH64_REG_INVALID, \
586
687
        .slice_offset = { -1 }, \
587
687
        .has_range_offset = false, \
588
687
        .is_vertical = false, \
589
687
      }; \
590
687
      AArch64_insert_detail_op_sme(MI, -1, za3_op); \
591
687
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
592
687
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
593
687
      }
594
#define ADD_ZA \
595
71
      { aarch64_op_sme za_op = \
596
71
        { \
597
71
        .type = AARCH64_SME_OP_TILE, \
598
71
        .tile = AARCH64_REG_ZA, \
599
71
        .slice_reg = AARCH64_REG_INVALID, \
600
71
        .slice_offset = { -1 }, \
601
71
        .has_range_offset = false, \
602
71
        .is_vertical = false, \
603
71
      }; \
604
71
      AArch64_insert_detail_op_sme(MI, -1, za_op); \
605
71
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
606
71
      }
607
608
static void AArch64_add_not_defined_ops(MCInst *MI, const SStream *OS)
609
306k
{
610
306k
  if (!detail_is_set(MI))
611
0
    return;
612
613
306k
  if (!MI->flat_insn->is_alias || !MI->flat_insn->usesAliasDetails) {
614
261k
    add_non_alias_details(MI);
615
261k
    return;
616
261k
  }
617
618
  // Alias details
619
45.3k
  switch (MI->flat_insn->alias_id) {
620
38.2k
  default:
621
38.2k
    return;
622
38.2k
  case AARCH64_INS_ALIAS_ROR:
623
119
    if (AArch64_get_detail(MI)->op_count != 3) {
624
0
      return;
625
0
    }
626
    // The ROR alias doesn't set the shift value properly.
627
    // Correct it here.
628
119
    bool reg_shift = AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_REG;
629
119
    uint64_t shift = reg_shift ? AArch64_get_detail_op(MI, -1)->reg : AArch64_get_detail_op(MI, -1)->imm;
630
119
    cs_aarch64_op *op1 = AArch64_get_detail_op(MI, -2);
631
119
    op1->shift.type = reg_shift ? AARCH64_SFT_ROR_REG : AARCH64_SFT_ROR;
632
119
    op1->shift.value = shift;
633
119
    AArch64_dec_op_count(MI);
634
119
    break;
635
85
  case AARCH64_INS_ALIAS_FMOV:
636
85
    if (AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_FP) {
637
85
      break;
638
85
    }
639
0
    AArch64_insert_detail_op_float_at(MI, -1, 0.0f, CS_AC_READ);
640
0
    break;
641
92
  case AARCH64_INS_ALIAS_LD1:
642
161
  case AARCH64_INS_ALIAS_LD1R:
643
943
  case AARCH64_INS_ALIAS_LD2:
644
1.36k
  case AARCH64_INS_ALIAS_LD2R:
645
1.64k
  case AARCH64_INS_ALIAS_LD3:
646
1.65k
  case AARCH64_INS_ALIAS_LD3R:
647
3.24k
  case AARCH64_INS_ALIAS_LD4:
648
3.52k
  case AARCH64_INS_ALIAS_LD4R:
649
3.98k
  case AARCH64_INS_ALIAS_ST1:
650
4.32k
  case AARCH64_INS_ALIAS_ST2:
651
4.34k
  case AARCH64_INS_ALIAS_ST3:
652
5.44k
  case AARCH64_INS_ALIAS_ST4: {
653
    // Add post-index disp
654
5.44k
    const char *disp_off = strrchr(OS->buffer, '#');
655
5.44k
    if (!disp_off)
656
0
      return;
657
5.44k
    unsigned disp = atoi(disp_off + 1);
658
5.44k
    AArch64_get_detail_op(MI, -1)->type = AARCH64_OP_MEM;
659
5.44k
    AArch64_get_detail_op(MI, -1)->mem.base =
660
5.44k
      AArch64_get_detail_op(MI, -1)->reg;
661
5.44k
    AArch64_get_detail_op(MI, -1)->mem.disp = disp;
662
5.44k
    AArch64_get_detail(MI)->post_index = true;
663
5.44k
    break;
664
5.44k
  }
665
6
  case AARCH64_INS_ALIAS_GCSB:
666
    // TODO
667
    // Only CSYNC is defined in LLVM. So we need to add it.
668
    //     /* 2825 */ "gcsb dsync\0"
669
6
    break;
670
240
  case AARCH64_INS_ALIAS_SMSTART:
671
288
  case AARCH64_INS_ALIAS_SMSTOP: {
672
288
    const char *disp_off = NULL;
673
288
    disp_off = strstr(OS->buffer, "smstart\tza");
674
288
    if (disp_off) {
675
219
      aarch64_sysop sysop = { 0 };
676
219
      sysop.alias.svcr = AARCH64_SVCR_SVCRZA;
677
219
      sysop.sub_type = AARCH64_OP_SVCR;
678
219
      AArch64_insert_detail_op_sys(MI, -1, sysop,
679
219
              AARCH64_OP_SYSALIAS);
680
219
      return;
681
219
    }
682
69
    disp_off = strstr(OS->buffer, "smstart\tsm");
683
69
    if (disp_off) {
684
19
      aarch64_sysop sysop = { 0 };
685
19
      sysop.alias.svcr = AARCH64_SVCR_SVCRSM;
686
19
      sysop.sub_type = AARCH64_OP_SVCR;
687
19
      AArch64_insert_detail_op_sys(MI, -1, sysop,
688
19
              AARCH64_OP_SYSALIAS);
689
19
      return;
690
19
    }
691
50
    break;
692
69
  }
693
1.20k
  case AARCH64_INS_ALIAS_ZERO: {
694
    // It is ugly, but the hard coded search patterns do it for now.
695
1.20k
    const char *disp_off = NULL;
696
697
1.20k
    disp_off = strstr(OS->buffer, "{za}");
698
1.20k
    if (disp_off) {
699
71
      ADD_ZA;
700
71
      return;
701
71
    }
702
1.12k
    disp_off = strstr(OS->buffer, "{za1.h}");
703
1.12k
    if (disp_off) {
704
40
      aarch64_op_sme op =
705
40
        {
706
40
        .type = AARCH64_SME_OP_TILE,
707
40
        .tile = AARCH64_REG_ZAH1,
708
40
        .slice_reg = AARCH64_REG_INVALID,
709
40
        .slice_offset = { -1 },
710
40
        .has_range_offset = false,
711
40
        .is_vertical = false,
712
40
      };
713
40
      AArch64_insert_detail_op_sme(MI, -1, op);
714
40
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_H;
715
40
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE;
716
40
      return;
717
40
    }
718
1.08k
    disp_off = strstr(OS->buffer, "{za0.h}");
719
1.08k
    if (disp_off) {
720
10
      aarch64_op_sme op =
721
10
        {
722
10
        .type = AARCH64_SME_OP_TILE,
723
10
        .tile = AARCH64_REG_ZAH0,
724
10
        .slice_reg = AARCH64_REG_INVALID,
725
10
        .slice_offset = { -1 },
726
10
        .has_range_offset = false,
727
10
        .is_vertical = false,
728
10
      };
729
10
      AArch64_insert_detail_op_sme(MI, -1, op);
730
10
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_H;
731
10
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE;
732
10
      return;
733
10
    }
734
1.07k
    disp_off = strstr(OS->buffer, "{za0.s}");
735
1.07k
    if (disp_off) {
736
37
      ADD_ZA0_S;
737
37
      return;
738
37
    }
739
1.04k
    disp_off = strstr(OS->buffer, "{za1.s}");
740
1.04k
    if (disp_off) {
741
18
      ADD_ZA1_S;
742
18
      return;
743
18
    }
744
1.02k
    disp_off = strstr(OS->buffer, "{za2.s}");
745
1.02k
    if (disp_off) {
746
67
      ADD_ZA2_S;
747
67
      return;
748
67
    }
749
957
    disp_off = strstr(OS->buffer, "{za3.s}");
750
957
    if (disp_off) {
751
10
      ADD_ZA3_S;
752
10
      return;
753
10
    }
754
947
    disp_off = strstr(OS->buffer, "{za0.s,za1.s}");
755
947
    if (disp_off) {
756
50
      ADD_ZA0_S;
757
50
      ADD_ZA1_S;
758
50
      return;
759
50
    }
760
897
    disp_off = strstr(OS->buffer, "{za0.s,za3.s}");
761
897
    if (disp_off) {
762
20
      ADD_ZA0_S;
763
20
      ADD_ZA3_S;
764
20
      return;
765
20
    }
766
877
    disp_off = strstr(OS->buffer, "{za1.s,za2.s}");
767
877
    if (disp_off) {
768
117
      ADD_ZA1_S;
769
117
      ADD_ZA2_S;
770
117
      return;
771
117
    }
772
760
    disp_off = strstr(OS->buffer, "{za2.s,za3.s}");
773
760
    if (disp_off) {
774
70
      ADD_ZA2_S;
775
70
      ADD_ZA3_S;
776
70
      return;
777
70
    }
778
690
    disp_off = strstr(OS->buffer, "{za0.s,za1.s,za2.s}");
779
690
    if (disp_off) {
780
103
      ADD_ZA0_S;
781
103
      ADD_ZA1_S;
782
103
      ADD_ZA2_S;
783
103
      return;
784
103
    }
785
587
    disp_off = strstr(OS->buffer, "{za0.s,za1.s,za3.s}");
786
587
    if (disp_off) {
787
18
      ADD_ZA0_S;
788
18
      ADD_ZA1_S;
789
18
      ADD_ZA3_S;
790
18
      return;
791
18
    }
792
569
    disp_off = strstr(OS->buffer, "{za0.s,za2.s,za3.s}");
793
569
    if (disp_off) {
794
60
      ADD_ZA0_S;
795
60
      ADD_ZA2_S;
796
60
      ADD_ZA3_S;
797
60
      return;
798
60
    }
799
509
    disp_off = strstr(OS->buffer, "{za1.s,za2.s,za3.s}");
800
509
    if (disp_off) {
801
509
      ADD_ZA1_S;
802
509
      ADD_ZA2_S;
803
509
      ADD_ZA3_S;
804
509
      return;
805
509
    }
806
0
    break;
807
509
  }
808
45.3k
  }
809
45.3k
}
810
811
void AArch64_set_instr_map_data(MCInst *MI)
812
313k
{
813
313k
  map_cs_id(MI, aarch64_insns, ARR_SIZE(aarch64_insns));
814
313k
  map_implicit_reads(MI, aarch64_insns);
815
313k
  map_implicit_writes(MI, aarch64_insns);
816
313k
  map_groups(MI, aarch64_insns);
817
313k
}
818
819
bool AArch64_getInstruction(csh handle, const uint8_t *code, size_t code_len,
820
          MCInst *MI, uint16_t *size, uint64_t address,
821
          void *info)
822
313k
{
823
313k
  AArch64_init_cs_detail(MI);
824
313k
  DecodeStatus Result = AArch64_LLVM_getInstruction(handle, code, code_len, MI,
825
313k
              size, address,
826
313k
              info);
827
313k
  AArch64_set_instr_map_data(MI);
828
313k
  if (Result == MCDisassembler_SoftFail) {
829
7.33k
    MCInst_setSoftFail(MI);
830
7.33k
  }
831
313k
  return Result != MCDisassembler_Fail;
832
313k
}
833
834
/// Patches the register names with Capstone specific alias.
835
/// Those are common alias for registers (e.g. r15 = pc)
836
/// which are not set in LLVM.
837
static void patch_cs_reg_alias(char *asm_str)
838
0
{
839
0
  bool skip_sub = false;
840
0
  char *x29 = strstr(asm_str, "x29");
841
0
  if (x29 > asm_str && strstr(asm_str, "0x29") == (x29 - 1)) {
842
    // Check for hex prefix
843
0
    skip_sub = true;
844
0
  }
845
0
  while (x29 && !skip_sub) {
846
0
    x29[0] = 'f';
847
0
    x29[1] = 'p';
848
0
    memmove(x29 + 2, x29 + 3, strlen(x29 + 3));
849
0
    asm_str[strlen(asm_str) - 1] = '\0';
850
0
    x29 = strstr(asm_str, "x29");
851
0
  }
852
0
  skip_sub = false;
853
0
  char *x30 = strstr(asm_str, "x30");
854
0
  if (x30 > asm_str && strstr(asm_str, "0x30") == (x30 - 1)) {
855
    // Check for hex prefix
856
0
    skip_sub = true;
857
0
  }
858
0
  while (x30 && !skip_sub) {
859
0
    x30[0] = 'l';
860
0
    x30[1] = 'r';
861
0
    memmove(x30 + 2, x30 + 3, strlen(x30 + 3));
862
0
    asm_str[strlen(asm_str) - 1] = '\0';
863
0
    x30 = strstr(asm_str, "x30");
864
0
  }
865
0
}
866
867
/// Adds group to the instruction which are not defined in LLVM.
868
static void AArch64_add_cs_groups(MCInst *MI)
869
306k
{
870
306k
  unsigned Opcode = MI->flat_insn->id;
871
306k
  switch (Opcode) {
872
298k
  default:
873
298k
    return;
874
298k
  case AARCH64_INS_SVC:
875
12
    add_group(MI, AARCH64_GRP_INT);
876
12
    break;
877
81
  case AARCH64_INS_SMC:
878
6.12k
  case AARCH64_INS_MSR:
879
7.54k
  case AARCH64_INS_MRS:
880
7.54k
    add_group(MI, AARCH64_GRP_PRIVILEGE);
881
7.54k
    break;
882
41
  case AARCH64_INS_RET:
883
90
  case AARCH64_INS_RETAA:
884
244
  case AARCH64_INS_RETAB:
885
244
    add_group(MI, AARCH64_GRP_RET);
886
244
    break;
887
306k
  }
888
306k
}
889
890
306k
static void AArch64_correct_mem_access(MCInst *MI) {
891
306k
  if (!detail_is_set(MI))
892
0
    return;
893
306k
  cs_ac_type access = aarch64_insns[MI->Opcode].suppl_info.aarch64.mem_acc;
894
306k
  if (access == CS_AC_INVALID) {
895
204k
    return;
896
204k
  }
897
216k
  for (int i = 0; i < AArch64_get_detail(MI)->op_count; ++i) {
898
214k
    if (AArch64_get_detail_op(MI, -i)->type == AARCH64_OP_MEM) {
899
99.8k
      AArch64_get_detail_op(MI, -i)->access = access;
900
99.8k
      return;
901
99.8k
    }
902
214k
  }
903
102k
}
904
905
void AArch64_printer(MCInst *MI, SStream *O, void * /* MCRegisterInfo* */ info)
906
306k
{
907
306k
  MCRegisterInfo *MRI = (MCRegisterInfo *)info;
908
306k
  MI->MRI = MRI;
909
306k
  MI->fillDetailOps = detail_is_set(MI);
910
306k
  MI->flat_insn->usesAliasDetails = map_use_alias_details(MI);
911
306k
  AArch64_LLVM_printInstruction(MI, O, info);
912
306k
  if (detail_is_set(MI)) {
913
306k
    if (AArch64_get_detail(MI)->is_doing_sme) {
914
      // Last operand still needs to be closed.
915
4.89k
      AArch64_get_detail(MI)->is_doing_sme = false;
916
4.89k
      AArch64_inc_op_count(MI);
917
4.89k
    }
918
306k
    AArch64_get_detail(MI)->post_index =
919
306k
      AArch64_check_post_index_am(MI, O);
920
306k
  }
921
306k
  AArch64_check_updates_flags(MI);
922
306k
  map_set_alias_id(MI, O, insn_alias_mnem_map,
923
306k
       ARR_SIZE(insn_alias_mnem_map) - 1);
924
306k
  int syntax_opt = MI->csh->syntax;
925
306k
  if (syntax_opt & CS_OPT_SYNTAX_CS_REG_ALIAS)
926
0
    patch_cs_reg_alias(O->buffer);
927
306k
  AArch64_add_not_defined_ops(MI, O);
928
306k
  AArch64_add_cs_groups(MI);
929
306k
  AArch64_add_vas(MI, O);
930
306k
  AArch64_correct_mem_access(MI);
931
306k
}
932
933
// given internal insn id, return public instruction info
934
void AArch64_get_insn_id(cs_struct *h, cs_insn *insn, unsigned int id)
935
306k
{
936
  // Done after disassembly
937
306k
  return;
938
306k
}
939
940
static const char *const insn_name_maps[] = {
941
#include "AArch64GenCSMappingInsnName.inc"
942
};
943
944
const char *AArch64_insn_name(csh handle, unsigned int id)
945
306k
{
946
306k
#ifndef CAPSTONE_DIET
947
306k
  if (id < AARCH64_INS_ALIAS_END && id > AARCH64_INS_ALIAS_BEGIN) {
948
0
    if (id - AARCH64_INS_ALIAS_BEGIN >=
949
0
        ARR_SIZE(insn_alias_mnem_map))
950
0
      return NULL;
951
952
0
    return insn_alias_mnem_map[id - AARCH64_INS_ALIAS_BEGIN - 1]
953
0
      .name;
954
0
  }
955
306k
  if (id >= AARCH64_INS_ENDING)
956
0
    return NULL;
957
958
306k
  if (id < ARR_SIZE(insn_name_maps))
959
306k
    return insn_name_maps[id];
960
961
  // not found
962
0
  return NULL;
963
#else
964
  return NULL;
965
#endif
966
306k
}
967
968
#ifndef CAPSTONE_DIET
969
static const name_map group_name_maps[] = {
970
  // generic groups
971
  { AARCH64_GRP_INVALID, NULL },
972
  { AARCH64_GRP_JUMP, "jump" },
973
  { AARCH64_GRP_CALL, "call" },
974
  { AARCH64_GRP_RET, "return" },
975
  { AARCH64_GRP_PRIVILEGE, "privilege" },
976
  { AARCH64_GRP_INT, "int" },
977
  { AARCH64_GRP_BRANCH_RELATIVE, "branch_relative" },
978
979
// architecture-specific groups
980
#include "AArch64GenCSFeatureName.inc"
981
};
982
#endif
983
984
const char *AArch64_group_name(csh handle, unsigned int id)
985
244k
{
986
244k
#ifndef CAPSTONE_DIET
987
244k
  return id2name(group_name_maps, ARR_SIZE(group_name_maps), id);
988
#else
989
  return NULL;
990
#endif
991
244k
}
992
993
// map instruction name to public instruction ID
994
aarch64_insn AArch64_map_insn(const char *name)
995
0
{
996
0
  unsigned int i;
997
998
0
  for (i = 1; i < ARR_SIZE(insn_name_maps); i++) {
999
0
    if (!strcmp(name, insn_name_maps[i]))
1000
0
      return i;
1001
0
  }
1002
1003
  // not found
1004
0
  return AARCH64_INS_INVALID;
1005
0
}
1006
1007
#ifndef CAPSTONE_DIET
1008
1009
static const map_insn_ops insn_operands[] = {
1010
#include "AArch64GenCSMappingInsnOp.inc"
1011
};
1012
1013
void AArch64_reg_access(const cs_insn *insn, cs_regs regs_read,
1014
      uint8_t *regs_read_count, cs_regs regs_write,
1015
      uint8_t *regs_write_count)
1016
0
{
1017
0
  uint8_t i;
1018
0
  uint8_t read_count, write_count;
1019
0
  cs_aarch64 *aarch64 = &(insn->detail->aarch64);
1020
1021
0
  read_count = insn->detail->regs_read_count;
1022
0
  write_count = insn->detail->regs_write_count;
1023
1024
  // implicit registers
1025
0
  memcpy(regs_read, insn->detail->regs_read,
1026
0
         read_count * sizeof(insn->detail->regs_read[0]));
1027
0
  memcpy(regs_write, insn->detail->regs_write,
1028
0
         write_count * sizeof(insn->detail->regs_write[0]));
1029
1030
  // explicit registers
1031
0
  for (i = 0; i < aarch64->op_count; i++) {
1032
0
    cs_aarch64_op *op = &(aarch64->operands[i]);
1033
0
    switch ((int)op->type) {
1034
0
    case AARCH64_OP_REG:
1035
0
      if ((op->access & CS_AC_READ) &&
1036
0
          !arr_exist(regs_read, read_count, op->reg)) {
1037
0
        regs_read[read_count] = (uint16_t)op->reg;
1038
0
        read_count++;
1039
0
      }
1040
0
      if ((op->access & CS_AC_WRITE) &&
1041
0
          !arr_exist(regs_write, write_count, op->reg)) {
1042
0
        regs_write[write_count] = (uint16_t)op->reg;
1043
0
        write_count++;
1044
0
      }
1045
0
      break;
1046
0
    case AARCH64_OP_MEM:
1047
      // registers appeared in memory references always being read
1048
0
      if ((op->mem.base != AARCH64_REG_INVALID) &&
1049
0
          !arr_exist(regs_read, read_count, op->mem.base)) {
1050
0
        regs_read[read_count] = (uint16_t)op->mem.base;
1051
0
        read_count++;
1052
0
      }
1053
0
      if ((op->mem.index != AARCH64_REG_INVALID) &&
1054
0
          !arr_exist(regs_read, read_count, op->mem.index)) {
1055
0
        regs_read[read_count] = (uint16_t)op->mem.index;
1056
0
        read_count++;
1057
0
      }
1058
0
      if ((insn->detail->writeback) &&
1059
0
          (op->mem.base != AARCH64_REG_INVALID) &&
1060
0
          !arr_exist(regs_write, write_count, op->mem.base)) {
1061
0
        regs_write[write_count] =
1062
0
          (uint16_t)op->mem.base;
1063
0
        write_count++;
1064
0
      }
1065
0
      break;
1066
0
    case AARCH64_OP_SME:
1067
0
        if ((op->access & CS_AC_READ) &&
1068
0
            (op->sme.tile != AARCH64_REG_INVALID) &&
1069
0
            !arr_exist(regs_read, read_count, op->sme.tile)) {
1070
0
          regs_read[read_count] = (uint16_t)op->sme.tile;
1071
0
          read_count++;
1072
0
        }
1073
0
        if ((op->access & CS_AC_WRITE) &&
1074
0
            (op->sme.tile != AARCH64_REG_INVALID) &&
1075
0
            !arr_exist(regs_write, write_count, op->sme.tile)) {
1076
0
          regs_write[write_count] = (uint16_t)op->sme.tile;
1077
0
          write_count++;
1078
0
        }
1079
0
        if ((op->sme.slice_reg != AARCH64_REG_INVALID) &&
1080
0
            !arr_exist(regs_read, read_count, op->sme.slice_reg)) {
1081
0
          regs_read[read_count] = (uint16_t)op->sme.slice_reg;
1082
0
          read_count++;
1083
0
        }
1084
0
        break;
1085
0
    case AARCH64_OP_PRED:
1086
0
      if ((op->access & CS_AC_READ) &&
1087
0
          (op->pred.reg != AARCH64_REG_INVALID) &&
1088
0
          !arr_exist(regs_read, read_count, op->pred.reg)) {
1089
0
        regs_read[read_count] = (uint16_t)op->pred.reg;
1090
0
        read_count++;
1091
0
      }
1092
0
      if ((op->access & CS_AC_WRITE) &&
1093
0
          (op->pred.reg != AARCH64_REG_INVALID) &&
1094
0
          !arr_exist(regs_write, write_count, op->pred.reg)) {
1095
0
        regs_write[write_count] = (uint16_t)op->pred.reg;
1096
0
        write_count++;
1097
0
      }
1098
0
      if ((op->pred.vec_select != AARCH64_REG_INVALID) &&
1099
0
          !arr_exist(regs_read, read_count, op->pred.vec_select)) {
1100
0
        regs_read[read_count] = (uint16_t)op->pred.vec_select;
1101
0
        read_count++;
1102
0
      }
1103
0
      break;
1104
0
    default:
1105
0
      break;
1106
0
    }
1107
0
    if (op->shift.type >= AARCH64_SFT_LSL_REG) {
1108
0
      if (!arr_exist(regs_read, read_count, op->shift.value)) {
1109
0
        regs_read[read_count] = (uint16_t)op->shift.value;
1110
0
        read_count++;
1111
0
      }
1112
0
    }
1113
0
  }
1114
1115
0
  switch (insn->alias_id) {
1116
0
  default:
1117
0
    break;
1118
0
  case AARCH64_INS_ALIAS_RET:
1119
0
    regs_read[read_count] = AARCH64_REG_X30;
1120
0
    read_count++;
1121
0
    break;
1122
0
  }
1123
1124
0
  *regs_read_count = read_count;
1125
0
  *regs_write_count = write_count;
1126
0
}
1127
#endif
1128
1129
static AArch64Layout_VectorLayout get_vl_by_suffix(const char suffix)
1130
195k
{
1131
195k
  switch (suffix) {
1132
54.2k
  default:
1133
54.2k
    return AARCH64LAYOUT_INVALID;
1134
33.9k
  case 'b':
1135
33.9k
  case 'B':
1136
33.9k
    return AARCH64LAYOUT_VL_B;
1137
31.9k
  case 'h':
1138
31.9k
  case 'H':
1139
31.9k
    return AARCH64LAYOUT_VL_H;
1140
33.1k
  case 's':
1141
33.1k
  case 'S':
1142
33.1k
    return AARCH64LAYOUT_VL_S;
1143
39.4k
  case 'd':
1144
39.4k
  case 'D':
1145
39.4k
    return AARCH64LAYOUT_VL_D;
1146
2.61k
  case 'q':
1147
2.61k
  case 'Q':
1148
2.61k
    return AARCH64LAYOUT_VL_Q;
1149
195k
  }
1150
195k
}
1151
1152
static unsigned get_vec_list_num_regs(MCInst *MI, unsigned Reg)
1153
59.4k
{
1154
  // Work out how many registers there are in the list (if there is an actual
1155
  // list).
1156
59.4k
  unsigned NumRegs = 1;
1157
59.4k
  if (MCRegisterClass_contains(
1158
59.4k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_DDRegClassID),
1159
59.4k
        Reg) ||
1160
59.4k
      MCRegisterClass_contains(
1161
58.5k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_ZPR2RegClassID),
1162
58.5k
        Reg) ||
1163
59.4k
      MCRegisterClass_contains(
1164
49.1k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_QQRegClassID),
1165
49.1k
        Reg) ||
1166
59.4k
      MCRegisterClass_contains(
1167
42.7k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_PPR2RegClassID),
1168
42.7k
        Reg) ||
1169
59.4k
      MCRegisterClass_contains(
1170
42.4k
        MCRegisterInfo_getRegClass(MI->MRI,
1171
42.4k
                 AArch64_ZPR2StridedRegClassID),
1172
42.4k
        Reg))
1173
18.2k
    NumRegs = 2;
1174
41.1k
  else if (MCRegisterClass_contains(
1175
41.1k
       MCRegisterInfo_getRegClass(MI->MRI,
1176
41.1k
                AArch64_DDDRegClassID),
1177
41.1k
       Reg) ||
1178
41.1k
     MCRegisterClass_contains(
1179
39.7k
       MCRegisterInfo_getRegClass(MI->MRI,
1180
39.7k
                AArch64_ZPR3RegClassID),
1181
39.7k
       Reg) ||
1182
41.1k
     MCRegisterClass_contains(
1183
39.4k
       MCRegisterInfo_getRegClass(MI->MRI,
1184
39.4k
                AArch64_QQQRegClassID),
1185
39.4k
       Reg))
1186
6.64k
    NumRegs = 3;
1187
34.5k
  else if (MCRegisterClass_contains(
1188
34.5k
       MCRegisterInfo_getRegClass(MI->MRI,
1189
34.5k
                AArch64_DDDDRegClassID),
1190
34.5k
       Reg) ||
1191
34.5k
     MCRegisterClass_contains(
1192
33.8k
       MCRegisterInfo_getRegClass(MI->MRI,
1193
33.8k
                AArch64_ZPR4RegClassID),
1194
33.8k
       Reg) ||
1195
34.5k
     MCRegisterClass_contains(
1196
26.4k
       MCRegisterInfo_getRegClass(MI->MRI,
1197
26.4k
                AArch64_QQQQRegClassID),
1198
26.4k
       Reg) ||
1199
34.5k
     MCRegisterClass_contains(
1200
19.3k
       MCRegisterInfo_getRegClass(
1201
19.3k
         MI->MRI, AArch64_ZPR4StridedRegClassID),
1202
19.3k
       Reg))
1203
15.9k
    NumRegs = 4;
1204
59.4k
  return NumRegs;
1205
59.4k
}
1206
1207
static unsigned get_vec_list_stride(MCInst *MI, unsigned Reg)
1208
59.4k
{
1209
59.4k
  unsigned Stride = 1;
1210
59.4k
  if (MCRegisterClass_contains(
1211
59.4k
        MCRegisterInfo_getRegClass(MI->MRI,
1212
59.4k
                 AArch64_ZPR2StridedRegClassID),
1213
59.4k
        Reg))
1214
1.28k
    Stride = 8;
1215
58.1k
  else if (MCRegisterClass_contains(
1216
58.1k
       MCRegisterInfo_getRegClass(
1217
58.1k
         MI->MRI, AArch64_ZPR4StridedRegClassID),
1218
58.1k
       Reg))
1219
807
    Stride = 4;
1220
59.4k
  return Stride;
1221
59.4k
}
1222
1223
static unsigned get_vec_list_first_reg(MCInst *MI, unsigned RegL)
1224
59.4k
{
1225
59.4k
  unsigned Reg = RegL;
1226
  // Now forget about the list and find out what the first register is.
1227
59.4k
  if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_dsub0))
1228
2.93k
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_dsub0);
1229
56.4k
  else if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_qsub0))
1230
18.3k
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_qsub0);
1231
38.1k
  else if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_zsub0))
1232
19.2k
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_zsub0);
1233
18.8k
  else if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_psub0))
1234
304
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_psub0);
1235
1236
  // If it's a D-reg, we need to promote it to the equivalent Q-reg before
1237
  // printing (otherwise getRegisterName fails).
1238
59.4k
  if (MCRegisterClass_contains(MCRegisterInfo_getRegClass(
1239
59.4k
               MI->MRI, AArch64_FPR64RegClassID),
1240
59.4k
             Reg)) {
1241
3.06k
    const MCRegisterClass *FPR128RC = MCRegisterInfo_getRegClass(
1242
3.06k
      MI->MRI, AArch64_FPR128RegClassID);
1243
3.06k
    Reg = MCRegisterInfo_getMatchingSuperReg(
1244
3.06k
      MI->MRI, Reg, AArch64_dsub, FPR128RC);
1245
3.06k
  }
1246
59.4k
  return Reg;
1247
59.4k
}
1248
1249
static bool is_vector_reg(unsigned Reg)
1250
182k
{
1251
182k
  if ((Reg >= AArch64_Q0) && (Reg <= AArch64_Q31))
1252
70.5k
    return true;
1253
111k
  else if ((Reg >= AArch64_Z0) && (Reg <= AArch64_Z31))
1254
110k
    return true;
1255
626
  else if ((Reg >= AArch64_P0) && (Reg <= AArch64_P15))
1256
626
    return true;
1257
0
  return false;
1258
182k
}
1259
1260
static unsigned getNextVectorRegister(unsigned Reg, unsigned Stride /* = 1 */)
1261
139k
{
1262
321k
  while (Stride--) {
1263
182k
    if (!is_vector_reg(Reg)) {
1264
0
      CS_ASSERT(0 && "Vector register expected!");
1265
0
      return 0;
1266
0
    }
1267
    // Vector lists can wrap around.
1268
182k
    else if (Reg == AArch64_Q31)
1269
2.58k
      Reg = AArch64_Q0;
1270
    // Vector lists can wrap around.
1271
179k
    else if (Reg == AArch64_Z31)
1272
1.39k
      Reg = AArch64_Z0;
1273
    // Vector lists can wrap around.
1274
178k
    else if (Reg == AArch64_P15)
1275
36
      Reg = AArch64_P0;
1276
177k
    else
1277
      // Assume ordered registers
1278
177k
      ++Reg;
1279
182k
  }
1280
139k
  return Reg;
1281
139k
}
1282
1283
static aarch64_extender llvm_to_cs_ext(AArch64_AM_ShiftExtendType ExtType)
1284
15.9k
{
1285
15.9k
  switch (ExtType) {
1286
12.8k
  default:
1287
12.8k
    return AARCH64_EXT_INVALID;
1288
431
  case AArch64_AM_UXTB:
1289
431
    return AARCH64_EXT_UXTB;
1290
577
  case AArch64_AM_UXTH:
1291
577
    return AARCH64_EXT_UXTH;
1292
794
  case AArch64_AM_UXTW:
1293
794
    return AARCH64_EXT_UXTW;
1294
544
  case AArch64_AM_UXTX:
1295
544
    return AARCH64_EXT_UXTX;
1296
342
  case AArch64_AM_SXTB:
1297
342
    return AARCH64_EXT_SXTB;
1298
83
  case AArch64_AM_SXTH:
1299
83
    return AARCH64_EXT_SXTH;
1300
69
  case AArch64_AM_SXTW:
1301
69
    return AARCH64_EXT_SXTW;
1302
249
  case AArch64_AM_SXTX:
1303
249
    return AARCH64_EXT_SXTX;
1304
15.9k
  }
1305
15.9k
}
1306
1307
static aarch64_shifter llvm_to_cs_shift(AArch64_AM_ShiftExtendType ShiftExtType)
1308
12.8k
{
1309
12.8k
  switch (ShiftExtType) {
1310
0
  default:
1311
0
    return AARCH64_SFT_INVALID;
1312
6.79k
  case AArch64_AM_LSL:
1313
6.79k
    return AARCH64_SFT_LSL;
1314
2.19k
  case AArch64_AM_LSR:
1315
2.19k
    return AARCH64_SFT_LSR;
1316
1.28k
  case AArch64_AM_ASR:
1317
1.28k
    return AARCH64_SFT_ASR;
1318
1.91k
  case AArch64_AM_ROR:
1319
1.91k
    return AARCH64_SFT_ROR;
1320
679
  case AArch64_AM_MSL:
1321
679
    return AARCH64_SFT_MSL;
1322
12.8k
  }
1323
12.8k
}
1324
1325
/// Initializes or finishes a memory operand of Capstone (depending on \p
1326
/// status). A memory operand in Capstone can be assembled by two LLVM operands.
1327
/// E.g. the base register and the immediate disponent.
1328
void AArch64_set_mem_access(MCInst *MI, bool status)
1329
354k
{
1330
354k
  if (!detail_is_set(MI))
1331
0
    return;
1332
354k
  set_doing_mem(MI, status);
1333
354k
  if (status) {
1334
177k
    if (AArch64_get_detail(MI)->op_count > 0 &&
1335
177k
        AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_MEM &&
1336
177k
        AArch64_get_detail_op(MI, -1)->mem.index ==
1337
69.0k
          AARCH64_REG_INVALID &&
1338
177k
        AArch64_get_detail_op(MI, -1)->mem.disp == 0) {
1339
      // Previous memory operand not done yet. Select it.
1340
67.8k
      AArch64_dec_op_count(MI);
1341
67.8k
      return;
1342
67.8k
    }
1343
1344
    // Init a new one.
1345
109k
    AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_MEM;
1346
109k
    AArch64_get_detail_op(MI, 0)->mem.base = AARCH64_REG_INVALID;
1347
109k
    AArch64_get_detail_op(MI, 0)->mem.index = AARCH64_REG_INVALID;
1348
109k
    AArch64_get_detail_op(MI, 0)->mem.disp = 0;
1349
1350
109k
#ifndef CAPSTONE_DIET
1351
109k
    uint8_t access =
1352
109k
      map_get_op_access(MI, AArch64_get_detail(MI)->op_count);
1353
109k
    AArch64_get_detail_op(MI, 0)->access = access;
1354
109k
#endif
1355
177k
  } else {
1356
    // done, select the next operand slot
1357
177k
    AArch64_inc_op_count(MI);
1358
177k
  }
1359
354k
}
1360
1361
/// Common prefix for all AArch64_add_cs_detail_* functions
1362
static bool add_cs_detail_begin(MCInst *MI, unsigned op_num)
1363
956k
{
1364
956k
  if (!detail_is_set(MI) || !map_fill_detail_ops(MI))
1365
0
    return false;
1366
1367
956k
  if (AArch64_get_detail(MI)->is_doing_sme) {
1368
    // Unset the flag if there is no bound operand anymore.
1369
104k
    if (!(map_get_op_type(MI, op_num) & CS_OP_BOUND)) {
1370
73.7k
      AArch64_get_detail(MI)->is_doing_sme = false;
1371
73.7k
      AArch64_inc_op_count(MI);
1372
73.7k
    }
1373
104k
  }
1374
956k
  return true;
1375
956k
}
1376
1377
/// Fills cs_detail with the data of the operand.
1378
/// This function handles operands which's original printer function has no
1379
/// specialities.
1380
void AArch64_add_cs_detail_0(MCInst *MI, aarch64_op_group op_group,
1381
          unsigned OpNum)
1382
561k
{
1383
561k
  if (!add_cs_detail_begin(MI, OpNum))
1384
0
    return;
1385
1386
  // Fill cs_detail
1387
561k
  switch (op_group) {
1388
0
  default:
1389
0
    printf("ERROR: Operand group %d not handled!\n", op_group);
1390
0
    CS_ASSERT_RET(0);
1391
403k
  case AArch64_OP_GROUP_Operand: {
1392
403k
    cs_op_type primary_op_type = map_get_op_type(MI, OpNum) &
1393
403k
               ~(CS_OP_MEM | CS_OP_BOUND);
1394
403k
    switch (primary_op_type) {
1395
0
    default:
1396
0
      printf("Unhandled operand type 0x%x\n",
1397
0
             primary_op_type);
1398
0
      CS_ASSERT_RET(0);
1399
338k
    case AARCH64_OP_REG:
1400
338k
      AArch64_set_detail_op_reg(MI, OpNum,
1401
338k
              MCInst_getOpVal(MI, OpNum));
1402
338k
      break;
1403
64.7k
    case AARCH64_OP_IMM:
1404
64.7k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1405
64.7k
              MCInst_getOpVal(MI, OpNum));
1406
64.7k
      break;
1407
653
    case AARCH64_OP_FP: {
1408
      // printOperand does not handle FP operands. But sometimes
1409
      // is used to print FP operands as normal immediate.
1410
653
      AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_IMM;
1411
653
      AArch64_get_detail_op(MI, 0)->imm =
1412
653
        MCInst_getOpVal(MI, OpNum);
1413
653
      AArch64_get_detail_op(MI, 0)->access =
1414
653
        map_get_op_access(MI, OpNum);
1415
653
      AArch64_inc_op_count(MI);
1416
653
      break;
1417
0
    }
1418
403k
    }
1419
403k
    break;
1420
403k
  }
1421
403k
  case AArch64_OP_GROUP_AddSubImm: {
1422
4.39k
    unsigned Val = (MCInst_getOpVal(MI, OpNum) & 0xfff);
1423
4.39k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1424
    // Shift is added in printShifter()
1425
4.39k
    break;
1426
403k
  }
1427
0
  case AArch64_OP_GROUP_AdrLabel: {
1428
0
    if (MCOperand_isImm(MCInst_getOperand(MI, OpNum))) {
1429
0
      int64_t Offset = MCInst_getOpVal(MI, OpNum);
1430
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1431
0
              (MI->address & -4) + Offset);
1432
0
    } else {
1433
      // Expression
1434
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1435
0
              MCOperand_isImm(MCInst_getOperand(MI, OpNum)));
1436
0
    }
1437
0
    break;
1438
403k
  }
1439
0
  case AArch64_OP_GROUP_AdrpLabel: {
1440
0
    if (MCOperand_isImm(MCInst_getOperand(MI, OpNum))) {
1441
0
      int64_t Offset = MCInst_getOpVal(MI, OpNum) * 4096;
1442
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1443
0
              (MI->address & -4096) + Offset);
1444
0
    } else {
1445
      // Expression
1446
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1447
0
              MCOperand_isImm(MCInst_getOperand(MI, OpNum)));
1448
0
    }
1449
0
    break;
1450
403k
  }
1451
5.10k
  case AArch64_OP_GROUP_AdrAdrpLabel: {
1452
5.10k
    if (!MCOperand_isImm(MCInst_getOperand(MI, OpNum))) {
1453
      // Expression
1454
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1455
0
              MCOperand_isImm(MCInst_getOperand(MI, OpNum)));
1456
0
      break;
1457
0
    }
1458
5.10k
    int64_t Offset = MCInst_getOpVal(MI, OpNum);
1459
5.10k
    uint64_t Address = MI->address;
1460
5.10k
    if (MCInst_getOpcode(MI) == AArch64_ADRP) {
1461
2.74k
      Offset = Offset * 4096;
1462
2.74k
      Address = Address & -4096;
1463
2.74k
    }
1464
5.10k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1465
5.10k
            Address + Offset);
1466
5.10k
    break;
1467
5.10k
  }
1468
13.5k
  case AArch64_OP_GROUP_AlignedLabel: {
1469
13.5k
    if (MCOperand_isImm(MCInst_getOperand(MI, OpNum))) {
1470
13.3k
      int64_t Offset = MCInst_getOpVal(MI, OpNum) * 4;
1471
13.3k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1472
13.3k
              MI->address + Offset);
1473
13.3k
    } else {
1474
      // Expression
1475
211
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1476
211
              MCOperand_isImm(MCInst_getOperand(MI, OpNum)));
1477
211
    }
1478
13.5k
    break;
1479
5.10k
  }
1480
0
  case AArch64_OP_GROUP_AMNoIndex: {
1481
0
    AArch64_set_detail_op_mem(MI, OpNum,
1482
0
            MCInst_getOpVal(MI, OpNum));
1483
0
    break;
1484
5.10k
  }
1485
3.08k
  case AArch64_OP_GROUP_ArithExtend: {
1486
3.08k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1487
3.08k
    AArch64_AM_ShiftExtendType ExtType =
1488
3.08k
      AArch64_AM_getArithExtendType(Val);
1489
3.08k
    unsigned ShiftVal = AArch64_AM_getArithShiftValue(Val);
1490
1491
3.08k
    AArch64_get_detail_op(MI, -1)->ext = llvm_to_cs_ext(ExtType);
1492
3.08k
    AArch64_get_detail_op(MI, -1)->shift.value = ShiftVal;
1493
3.08k
    AArch64_get_detail_op(MI, -1)->shift.type = AARCH64_SFT_LSL;
1494
3.08k
    break;
1495
5.10k
  }
1496
96
  case AArch64_OP_GROUP_BarriernXSOption: {
1497
96
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1498
96
    aarch64_sysop sysop = { 0 };
1499
96
    const AArch64DBnXS_DBnXS *DB =
1500
96
      AArch64DBnXS_lookupDBnXSByEncoding(Val);
1501
96
    if (DB)
1502
96
      sysop.imm.dbnxs = (aarch64_dbnxs) DB->SysImm.dbnxs;
1503
0
    else
1504
0
      sysop.imm.raw_val = Val;
1505
96
    sysop.sub_type = AARCH64_OP_DBNXS;
1506
96
    AArch64_set_detail_op_sys(MI, OpNum, sysop, AARCH64_OP_SYSIMM);
1507
96
    break;
1508
5.10k
  }
1509
184
  case AArch64_OP_GROUP_AppleSysBarrierOption: {
1510
    // Proprietary stuff. We just add the
1511
    // immediate here.
1512
184
    unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1513
184
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1514
184
    break;
1515
5.10k
  }
1516
135
  case AArch64_OP_GROUP_BarrierOption: {
1517
135
    unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1518
135
    unsigned Opcode = MCInst_getOpcode(MI);
1519
135
    aarch64_sysop sysop = { 0 };
1520
1521
135
    if (Opcode == AArch64_ISB) {
1522
17
      const AArch64ISB_ISB *ISB =
1523
17
        AArch64ISB_lookupISBByEncoding(Val);
1524
17
      if (ISB)
1525
0
        sysop.alias.isb = (aarch64_isb) ISB->SysAlias.isb;
1526
17
      else
1527
17
        sysop.alias.raw_val = Val;
1528
17
      sysop.sub_type = AARCH64_OP_ISB;
1529
17
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1530
17
              AARCH64_OP_SYSALIAS);
1531
118
    } else if (Opcode == AArch64_TSB) {
1532
37
      const AArch64TSB_TSB *TSB =
1533
37
        AArch64TSB_lookupTSBByEncoding(Val);
1534
37
      if (TSB)
1535
37
        sysop.alias.tsb = (aarch64_tsb) TSB->SysAlias.tsb;
1536
0
      else
1537
0
        sysop.alias.raw_val = Val;
1538
37
      sysop.sub_type = AARCH64_OP_TSB;
1539
37
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1540
37
              AARCH64_OP_SYSALIAS);
1541
81
    } else {
1542
81
      const AArch64DB_DB *DB =
1543
81
        AArch64DB_lookupDBByEncoding(Val);
1544
81
      if (DB)
1545
42
        sysop.alias.db = (aarch64_db) DB->SysAlias.db;
1546
39
      else
1547
39
        sysop.alias.raw_val = Val;
1548
81
      sysop.sub_type = AARCH64_OP_DB;
1549
81
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1550
81
              AARCH64_OP_SYSALIAS);
1551
81
    }
1552
135
    break;
1553
5.10k
  }
1554
396
  case AArch64_OP_GROUP_BTIHintOp: {
1555
396
    aarch64_sysop sysop = { 0 };
1556
396
    unsigned btihintop = MCInst_getOpVal(MI, OpNum) ^ 32;
1557
396
    const AArch64BTIHint_BTI *BTI =
1558
396
      AArch64BTIHint_lookupBTIByEncoding(btihintop);
1559
396
    if (BTI)
1560
396
      sysop.alias.bti = (aarch64_bti) BTI->SysAlias.bti;
1561
0
    else
1562
0
      sysop.alias.raw_val = btihintop;
1563
396
    sysop.sub_type = AARCH64_OP_BTI;
1564
396
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1565
396
            AARCH64_OP_SYSALIAS);
1566
396
    break;
1567
5.10k
  }
1568
2.92k
  case AArch64_OP_GROUP_CondCode: {
1569
2.92k
    AArch64_get_detail(MI)->cc = MCInst_getOpVal(MI, OpNum);
1570
2.92k
    break;
1571
5.10k
  }
1572
2.00k
  case AArch64_OP_GROUP_ExtendedRegister: {
1573
2.00k
    AArch64_set_detail_op_reg(MI, OpNum,
1574
2.00k
            MCInst_getOpVal(MI, OpNum));
1575
2.00k
    break;
1576
5.10k
  }
1577
206
  case AArch64_OP_GROUP_FPImmOperand: {
1578
206
    MCOperand *MO = MCInst_getOperand(MI, (OpNum));
1579
206
    float FPImm =
1580
206
      MCOperand_isDFPImm(MO) ?
1581
0
        BitsToDouble(MCOperand_getImm(MO)) :
1582
206
        AArch64_AM_getFPImmFloat(MCOperand_getImm(MO));
1583
206
    AArch64_set_detail_op_float(MI, OpNum, FPImm);
1584
206
    break;
1585
5.10k
  }
1586
5.64k
  case AArch64_OP_GROUP_GPR64as32: {
1587
5.64k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
1588
5.64k
    AArch64_set_detail_op_reg(MI, OpNum, getWRegFromXReg(Reg));
1589
5.64k
    break;
1590
5.10k
  }
1591
143
  case AArch64_OP_GROUP_GPR64x8: {
1592
143
    unsigned Reg = MCInst_getOpVal(MI, (OpNum));
1593
143
    Reg = MCRegisterInfo_getSubReg(MI->MRI, Reg, AArch64_x8sub_0);
1594
143
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1595
143
    break;
1596
5.10k
  }
1597
5.03k
  case AArch64_OP_GROUP_Imm:
1598
5.12k
  case AArch64_OP_GROUP_ImmHex:
1599
5.12k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1600
5.12k
            MCInst_getOpVal(MI, OpNum));
1601
5.12k
    break;
1602
0
  case AArch64_OP_GROUP_ImplicitlyTypedVectorList:
1603
    // The TypedVectorList implements the logic of implicitly typed operand.
1604
0
    AArch64_add_cs_detail_2(MI, AArch64_OP_GROUP_TypedVectorList_0_b, OpNum,
1605
0
            0, 0);
1606
0
    break;
1607
84
  case AArch64_OP_GROUP_InverseCondCode: {
1608
84
    AArch64CC_CondCode CC = (AArch64CC_CondCode)MCOperand_getImm(
1609
84
      MCInst_getOperand(MI, (OpNum)));
1610
84
    AArch64_get_detail(MI)->cc = AArch64CC_getInvertedCondCode(CC);
1611
84
    break;
1612
5.03k
  }
1613
2.10k
  case AArch64_OP_GROUP_MatrixTile: {
1614
2.10k
    const char *RegName = AArch64_LLVM_getRegisterName(
1615
2.10k
      MCInst_getOpVal(MI, OpNum), AArch64_NoRegAltName);
1616
2.10k
    const char *Dot = strstr(RegName, ".");
1617
2.10k
    AArch64Layout_VectorLayout vas = AARCH64LAYOUT_INVALID;
1618
2.10k
    if (!Dot) {
1619
      // The matrix dimensions are machine dependent.
1620
      // Currently we do not support differentiation of machines.
1621
      // So we just indicate the use of the complete matrix.
1622
0
      vas = sme_reg_to_vas(MCInst_getOpVal(MI, OpNum));
1623
0
    } else
1624
2.10k
      vas = get_vl_by_suffix(Dot[1]);
1625
2.10k
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
1626
2.10k
            vas);
1627
2.10k
    break;
1628
5.03k
  }
1629
581
  case AArch64_OP_GROUP_MatrixTileList: {
1630
581
    unsigned MaxRegs = 8;
1631
581
    unsigned RegMask = MCInst_getOpVal(MI, (OpNum));
1632
1633
5.22k
    for (unsigned I = 0; I < MaxRegs; ++I) {
1634
4.64k
      unsigned Reg = RegMask & (1 << I);
1635
4.64k
      if (Reg == 0)
1636
1.95k
        continue;
1637
2.69k
      AArch64_get_detail_op(MI, 0)->is_list_member = true;
1638
2.69k
      AArch64_set_detail_op_sme(MI, OpNum,
1639
2.69k
              AARCH64_SME_MATRIX_TILE_LIST,
1640
2.69k
              AARCH64LAYOUT_VL_D,
1641
2.69k
              (int) (AARCH64_REG_ZAD0 + I));
1642
2.69k
      AArch64_inc_op_count(MI);
1643
2.69k
    }
1644
581
    AArch64_get_detail(MI)->is_doing_sme = false;
1645
581
    break;
1646
5.03k
  }
1647
2.04k
  case AArch64_OP_GROUP_MRSSystemRegister:
1648
6.65k
  case AArch64_OP_GROUP_MSRSystemRegister: {
1649
6.65k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1650
6.65k
    const AArch64SysReg_SysReg *Reg =
1651
6.65k
      AArch64SysReg_lookupSysRegByEncoding(Val);
1652
6.65k
    bool Read = (op_group == AArch64_OP_GROUP_MRSSystemRegister) ?
1653
2.04k
            true :
1654
6.65k
            false;
1655
1656
6.65k
    bool isValidSysReg =
1657
6.65k
      (Reg && (Read ? Reg->Readable : Reg->Writeable) &&
1658
6.65k
       AArch64_testFeatureList(MI->csh->mode,
1659
382
             Reg->FeaturesRequired));
1660
1661
6.65k
    if (Reg && !isValidSysReg)
1662
1.25k
      Reg = AArch64SysReg_lookupSysRegByName(Reg->AltName);
1663
6.65k
    aarch64_sysop sysop = { 0 };
1664
    // If Reg is NULL it is a generic system register.
1665
6.65k
    if (Reg)
1666
1.62k
      sysop.reg.sysreg = (aarch64_sysreg) Reg->SysReg.sysreg;
1667
5.02k
    else {
1668
5.02k
      sysop.reg.raw_val = Val;
1669
5.02k
    }
1670
6.65k
    aarch64_op_type type =
1671
6.65k
      (op_group == AArch64_OP_GROUP_MRSSystemRegister) ?
1672
2.04k
        AARCH64_OP_REG_MRS :
1673
6.65k
        AARCH64_OP_REG_MSR;
1674
6.65k
    sysop.sub_type = type;
1675
6.65k
    AArch64_set_detail_op_sys(MI, OpNum, sysop, AARCH64_OP_SYSREG);
1676
6.65k
    break;
1677
2.04k
  }
1678
279
  case AArch64_OP_GROUP_PSBHintOp: {
1679
279
    unsigned psbhintop = MCInst_getOpVal(MI, OpNum);
1680
279
    const AArch64PSBHint_PSB *PSB =
1681
279
      AArch64PSBHint_lookupPSBByEncoding(psbhintop);
1682
279
    aarch64_sysop sysop = { 0 };
1683
279
    if (PSB)
1684
279
      sysop.alias.psb = (aarch64_psb) PSB->SysAlias.psb;
1685
0
    else
1686
0
      sysop.alias.raw_val = psbhintop;
1687
279
    sysop.sub_type = AARCH64_OP_PSB;
1688
279
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1689
279
            AARCH64_OP_SYSALIAS);
1690
279
    break;
1691
2.04k
  }
1692
918
  case AArch64_OP_GROUP_RPRFMOperand: {
1693
918
    unsigned prfop = MCInst_getOpVal(MI, OpNum);
1694
918
    const AArch64PRFM_PRFM *PRFM =
1695
918
      AArch64PRFM_lookupPRFMByEncoding(prfop);
1696
918
    aarch64_sysop sysop = { 0 };
1697
918
    if (PRFM)
1698
819
      sysop.alias.prfm = (aarch64_prfm) PRFM->SysAlias.prfm;
1699
99
    else
1700
99
      sysop.alias.raw_val = prfop;
1701
918
    sysop.sub_type = AARCH64_OP_PRFM;
1702
918
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1703
918
            AARCH64_OP_SYSALIAS);
1704
918
    break;
1705
2.04k
  }
1706
6.99k
  case AArch64_OP_GROUP_ShiftedRegister: {
1707
6.99k
    AArch64_set_detail_op_reg(MI, OpNum,
1708
6.99k
            MCInst_getOpVal(MI, OpNum));
1709
    // Shift part is handled in printShifter()
1710
6.99k
    break;
1711
2.04k
  }
1712
12.8k
  case AArch64_OP_GROUP_Shifter: {
1713
12.8k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1714
12.8k
    AArch64_AM_ShiftExtendType ShExtType =
1715
12.8k
      AArch64_AM_getShiftType(Val);
1716
12.8k
    AArch64_get_detail_op(MI, -1)->ext = llvm_to_cs_ext(ShExtType);
1717
12.8k
    AArch64_get_detail_op(MI, -1)->shift.type =
1718
12.8k
      llvm_to_cs_shift(ShExtType);
1719
12.8k
    AArch64_get_detail_op(MI, -1)->shift.value =
1720
12.8k
      AArch64_AM_getShiftValue(Val);
1721
12.8k
    break;
1722
2.04k
  }
1723
1.55k
  case AArch64_OP_GROUP_SIMDType10Operand: {
1724
1.55k
    unsigned RawVal = MCInst_getOpVal(MI, OpNum);
1725
1.55k
    uint64_t Val = AArch64_AM_decodeAdvSIMDModImmType10(RawVal);
1726
1.55k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1727
1.55k
    break;
1728
2.04k
  }
1729
0
  case AArch64_OP_GROUP_SVCROp: {
1730
0
    unsigned svcrop = MCInst_getOpVal(MI, OpNum);
1731
0
    const AArch64SVCR_SVCR *SVCR =
1732
0
      AArch64SVCR_lookupSVCRByEncoding(svcrop);
1733
0
    aarch64_sysop sysop = { 0 };
1734
0
    if (SVCR)
1735
0
      sysop.alias.svcr = (aarch64_svcr) SVCR->SysAlias.svcr;
1736
0
    else
1737
0
      sysop.alias.raw_val = svcrop;
1738
0
    sysop.sub_type = AARCH64_OP_SVCR;
1739
0
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1740
0
            AARCH64_OP_SYSALIAS);
1741
0
    break;
1742
2.04k
  }
1743
6.80k
  case AArch64_OP_GROUP_SVEPattern: {
1744
6.80k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1745
6.80k
    const AArch64SVEPredPattern_SVEPREDPAT *Pat =
1746
6.80k
      AArch64SVEPredPattern_lookupSVEPREDPATByEncoding(Val);
1747
6.80k
    if (!Pat) {
1748
2.15k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1749
2.15k
      break;
1750
2.15k
    }
1751
4.65k
    aarch64_sysop sysop = { 0 };
1752
4.65k
    sysop.alias = Pat->SysAlias;
1753
4.65k
    sysop.sub_type = AARCH64_OP_SVEPREDPAT;
1754
4.65k
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1755
4.65k
            AARCH64_OP_SYSALIAS);
1756
4.65k
    break;
1757
6.80k
  }
1758
1.61k
  case AArch64_OP_GROUP_SVEVecLenSpecifier: {
1759
1.61k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1760
    // Pattern has only 1 bit
1761
1.61k
    if (Val > 1)
1762
0
      CS_ASSERT_RET(0 && "Invalid vector length specifier");
1763
1.61k
    const AArch64SVEVecLenSpecifier_SVEVECLENSPECIFIER *Pat =
1764
1.61k
      AArch64SVEVecLenSpecifier_lookupSVEVECLENSPECIFIERByEncoding(
1765
1.61k
        Val);
1766
1.61k
    if (!Pat)
1767
0
      break;
1768
1.61k
    aarch64_sysop sysop = { 0 };
1769
1.61k
    sysop.alias = Pat->SysAlias;
1770
1.61k
    sysop.sub_type = AARCH64_OP_SVEVECLENSPECIFIER;
1771
1.61k
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1772
1.61k
            AARCH64_OP_SYSALIAS);
1773
1.61k
    break;
1774
1.61k
  }
1775
15.3k
  case AArch64_OP_GROUP_SysCROperand: {
1776
15.3k
    uint64_t cimm = MCInst_getOpVal(MI, OpNum);
1777
15.3k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_CIMM, cimm);
1778
15.3k
    break;
1779
1.61k
  }
1780
1.45k
  case AArch64_OP_GROUP_SyspXzrPair: {
1781
1.45k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
1782
1.45k
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1783
1.45k
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1784
1.45k
    break;
1785
1.61k
  }
1786
1.27k
  case AArch64_OP_GROUP_SystemPStateField: {
1787
1.27k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1788
1789
1.27k
    aarch64_sysop sysop = { 0 };
1790
1.27k
    const AArch64PState_PStateImm0_15 *PStateImm15 =
1791
1.27k
      AArch64PState_lookupPStateImm0_15ByEncoding(Val);
1792
1.27k
    const AArch64PState_PStateImm0_1 *PStateImm1 =
1793
1.27k
      AArch64PState_lookupPStateImm0_1ByEncoding(Val);
1794
1.27k
    if (PStateImm15 &&
1795
1.27k
        AArch64_testFeatureList(MI->csh->mode,
1796
1.20k
              PStateImm15->FeaturesRequired)) {
1797
1.20k
      sysop.alias = PStateImm15->SysAlias;
1798
1.20k
      sysop.sub_type = AARCH64_OP_PSTATEIMM0_15;
1799
1.20k
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1800
1.20k
              AARCH64_OP_SYSALIAS);
1801
1.20k
    } else if (PStateImm1 &&
1802
68
         AArch64_testFeatureList(
1803
68
           MI->csh->mode,
1804
68
           PStateImm1->FeaturesRequired)) {
1805
68
      sysop.alias = PStateImm1->SysAlias;
1806
68
      sysop.sub_type = AARCH64_OP_PSTATEIMM0_1;
1807
68
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1808
68
              AARCH64_OP_SYSALIAS);
1809
68
    } else {
1810
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1811
0
              Val);
1812
0
    }
1813
1.27k
    break;
1814
1.61k
  }
1815
56.5k
  case AArch64_OP_GROUP_VRegOperand: {
1816
56.5k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
1817
56.5k
    AArch64_get_detail_op(MI, 0)->is_vreg = true;
1818
56.5k
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1819
56.5k
    break;
1820
1.61k
  }
1821
561k
  }
1822
561k
}
1823
1824
/// Fills cs_detail with the data of the operand.
1825
/// This function handles operands which original printer function is a template
1826
/// with one argument.
1827
void AArch64_add_cs_detail_1(MCInst *MI, aarch64_op_group op_group,
1828
             unsigned OpNum, uint64_t temp_arg_0)
1829
305k
{
1830
305k
  if (!add_cs_detail_begin(MI, OpNum))
1831
0
    return;
1832
305k
  switch (op_group) {
1833
0
  default:
1834
0
    printf("ERROR: Operand group %d not handled!\n", op_group);
1835
0
    CS_ASSERT_RET(0);
1836
132
  case AArch64_OP_GROUP_GPRSeqPairsClassOperand_32:
1837
2.85k
  case AArch64_OP_GROUP_GPRSeqPairsClassOperand_64: {
1838
2.85k
    unsigned size = temp_arg_0;
1839
2.85k
    unsigned Reg = MCInst_getOpVal(MI, (OpNum));
1840
1841
2.85k
    unsigned Sube = (size == 32) ? AArch64_sube32 : AArch64_sube64;
1842
2.85k
    unsigned Subo = (size == 32) ? AArch64_subo32 : AArch64_subo64;
1843
1844
2.85k
    unsigned Even = MCRegisterInfo_getSubReg(MI->MRI, Reg, Sube);
1845
2.85k
    unsigned Odd = MCRegisterInfo_getSubReg(MI->MRI, Reg, Subo);
1846
2.85k
    AArch64_set_detail_op_reg(MI, OpNum, Even);
1847
2.85k
    AArch64_set_detail_op_reg(MI, OpNum, Odd);
1848
2.85k
    break;
1849
132
  }
1850
580
  case AArch64_OP_GROUP_Imm8OptLsl_int16_t:
1851
915
  case AArch64_OP_GROUP_Imm8OptLsl_int32_t:
1852
1.21k
  case AArch64_OP_GROUP_Imm8OptLsl_int64_t:
1853
1.51k
  case AArch64_OP_GROUP_Imm8OptLsl_int8_t:
1854
2.00k
  case AArch64_OP_GROUP_Imm8OptLsl_uint16_t:
1855
2.76k
  case AArch64_OP_GROUP_Imm8OptLsl_uint32_t:
1856
3.37k
  case AArch64_OP_GROUP_Imm8OptLsl_uint64_t:
1857
3.50k
  case AArch64_OP_GROUP_Imm8OptLsl_uint8_t: {
1858
3.50k
    unsigned UnscaledVal = MCInst_getOpVal(MI, (OpNum));
1859
3.50k
    unsigned Shift = MCInst_getOpVal(MI, (OpNum + 1));
1860
1861
3.50k
    if ((UnscaledVal == 0) &&
1862
3.50k
        (AArch64_AM_getShiftValue(Shift) != 0)) {
1863
342
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1864
342
              UnscaledVal);
1865
      // Shift is handled in printShifter()
1866
342
      break;
1867
342
    }
1868
1869
3.16k
  #define SCALE_SET(T) \
1870
3.16k
    do { \
1871
3.16k
      T Val; \
1872
3.16k
      if (CHAR(T) == 'i') /* Signed */ \
1873
3.16k
        Val = (int8_t)UnscaledVal * \
1874
1.44k
              (1 << AArch64_AM_getShiftValue(Shift)); \
1875
3.16k
      else \
1876
3.16k
        Val = (uint8_t)UnscaledVal * \
1877
1.71k
              (1 << AArch64_AM_getShiftValue(Shift)); \
1878
3.16k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val); \
1879
3.16k
    } while (0)
1880
1881
3.16k
    switch (op_group) {
1882
0
    default:
1883
0
      CS_ASSERT_RET(0 &&
1884
0
             "Operand group for Imm8OptLsl not handled.");
1885
542
    case AArch64_OP_GROUP_Imm8OptLsl_int16_t: {
1886
542
      SCALE_SET(int16_t);
1887
542
      break;
1888
0
    }
1889
316
    case AArch64_OP_GROUP_Imm8OptLsl_int32_t: {
1890
316
      SCALE_SET(int32_t);
1891
316
      break;
1892
0
    }
1893
287
    case AArch64_OP_GROUP_Imm8OptLsl_int64_t: {
1894
287
      SCALE_SET(int64_t);
1895
287
      break;
1896
0
    }
1897
298
    case AArch64_OP_GROUP_Imm8OptLsl_int8_t: {
1898
298
      SCALE_SET(int8_t);
1899
298
      break;
1900
0
    }
1901
425
    case AArch64_OP_GROUP_Imm8OptLsl_uint16_t: {
1902
425
      SCALE_SET(uint16_t);
1903
425
      break;
1904
0
    }
1905
726
    case AArch64_OP_GROUP_Imm8OptLsl_uint32_t: {
1906
726
      SCALE_SET(uint32_t);
1907
726
      break;
1908
0
    }
1909
441
    case AArch64_OP_GROUP_Imm8OptLsl_uint64_t: {
1910
441
      SCALE_SET(uint64_t);
1911
441
      break;
1912
0
    }
1913
126
    case AArch64_OP_GROUP_Imm8OptLsl_uint8_t: {
1914
126
      SCALE_SET(uint8_t);
1915
126
      break;
1916
0
    }
1917
3.16k
    }
1918
3.16k
    break;
1919
3.16k
  }
1920
3.16k
  case AArch64_OP_GROUP_ImmScale_16:
1921
4.72k
  case AArch64_OP_GROUP_ImmScale_2:
1922
4.83k
  case AArch64_OP_GROUP_ImmScale_3:
1923
4.87k
  case AArch64_OP_GROUP_ImmScale_32:
1924
14.5k
  case AArch64_OP_GROUP_ImmScale_4:
1925
21.4k
  case AArch64_OP_GROUP_ImmScale_8: {
1926
21.4k
    unsigned Scale = temp_arg_0;
1927
21.4k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1928
21.4k
            Scale * MCInst_getOpVal(MI, OpNum));
1929
21.4k
    break;
1930
14.5k
  }
1931
937
  case AArch64_OP_GROUP_LogicalImm_int16_t:
1932
4.75k
  case AArch64_OP_GROUP_LogicalImm_int32_t:
1933
8.06k
  case AArch64_OP_GROUP_LogicalImm_int64_t:
1934
12.2k
  case AArch64_OP_GROUP_LogicalImm_int8_t: {
1935
12.2k
    unsigned TypeSize = temp_arg_0;
1936
12.2k
    uint64_t Val = AArch64_AM_decodeLogicalImmediate(
1937
12.2k
      MCInst_getOpVal(MI, OpNum), 8 * TypeSize);
1938
12.2k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1939
12.2k
    break;
1940
8.06k
  }
1941
167
  case AArch64_OP_GROUP_Matrix_0:
1942
2.17k
  case AArch64_OP_GROUP_Matrix_16:
1943
5.95k
  case AArch64_OP_GROUP_Matrix_32:
1944
8.62k
  case AArch64_OP_GROUP_Matrix_64: {
1945
8.62k
    unsigned EltSize = temp_arg_0;
1946
8.62k
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
1947
8.62k
            (AArch64Layout_VectorLayout)EltSize);
1948
8.62k
    break;
1949
5.95k
  }
1950
0
  case AArch64_OP_GROUP_MatrixIndex_0:
1951
10.0k
  case AArch64_OP_GROUP_MatrixIndex_1:
1952
10.5k
  case AArch64_OP_GROUP_MatrixIndex_8: {
1953
10.5k
    unsigned scale = temp_arg_0;
1954
10.5k
    if (AArch64_get_detail_op(MI, 0)->type ==
1955
10.5k
        AARCH64_OP_SME) {
1956
        // The index is part of an SME matrix
1957
9.73k
        AArch64_set_detail_op_sme(MI, OpNum,
1958
9.73k
                AARCH64_SME_MATRIX_SLICE_OFF,
1959
9.73k
                AARCH64LAYOUT_INVALID,
1960
9.73k
                (uint32_t) (MCInst_getOpVal(MI, OpNum) * scale));
1961
9.73k
    } else if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_PRED) {
1962
      // The index is part of a predicate
1963
234
      AArch64_set_detail_op_pred(MI, OpNum);
1964
604
    } else {
1965
      // The index is used for an SVE2 instruction.
1966
604
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1967
604
              scale * MCInst_getOpVal(MI, OpNum));
1968
604
    }
1969
10.5k
    break;
1970
10.0k
  }
1971
2.59k
  case AArch64_OP_GROUP_MatrixTileVector_0:
1972
6.61k
  case AArch64_OP_GROUP_MatrixTileVector_1: {
1973
6.61k
    bool isVertical = temp_arg_0;
1974
6.61k
    const char *RegName = AArch64_LLVM_getRegisterName(
1975
6.61k
      MCInst_getOpVal(MI, OpNum), AArch64_NoRegAltName);
1976
6.61k
    const char *Dot = strstr(RegName, ".");
1977
6.61k
    AArch64Layout_VectorLayout vas = AARCH64LAYOUT_INVALID;
1978
6.61k
    if (!Dot) {
1979
      // The matrix dimensions are machine dependent.
1980
      // Currently we do not support differentiation of machines.
1981
      // So we just indicate the use of the complete matrix.
1982
0
      vas = sme_reg_to_vas(MCInst_getOpVal(MI, OpNum));
1983
0
    } else
1984
6.61k
      vas = get_vl_by_suffix(Dot[1]);
1985
6.61k
    setup_sme_operand(MI);
1986
6.61k
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
1987
6.61k
            vas);
1988
6.61k
    AArch64_get_detail_op(MI, 0)->sme.is_vertical = isVertical;
1989
6.61k
    break;
1990
2.59k
  }
1991
1.26k
  case AArch64_OP_GROUP_PostIncOperand_1:
1992
1.37k
  case AArch64_OP_GROUP_PostIncOperand_12:
1993
3.06k
  case AArch64_OP_GROUP_PostIncOperand_16:
1994
3.67k
  case AArch64_OP_GROUP_PostIncOperand_2:
1995
5.71k
  case AArch64_OP_GROUP_PostIncOperand_24:
1996
6.06k
  case AArch64_OP_GROUP_PostIncOperand_3:
1997
6.83k
  case AArch64_OP_GROUP_PostIncOperand_32:
1998
7.39k
  case AArch64_OP_GROUP_PostIncOperand_4:
1999
7.92k
  case AArch64_OP_GROUP_PostIncOperand_48:
2000
8.91k
  case AArch64_OP_GROUP_PostIncOperand_6:
2001
8.97k
  case AArch64_OP_GROUP_PostIncOperand_64:
2002
10.5k
  case AArch64_OP_GROUP_PostIncOperand_8: {
2003
10.5k
    uint64_t Imm = temp_arg_0;
2004
10.5k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
2005
10.5k
    if (Reg == AArch64_XZR) {
2006
0
      AArch64_get_detail_op(MI, -1)->mem.disp = Imm;
2007
0
      AArch64_get_detail(MI)->post_index = true;
2008
0
      AArch64_inc_op_count(MI);
2009
0
    } else
2010
10.5k
      AArch64_set_detail_op_reg(MI, OpNum, Reg);
2011
10.5k
    break;
2012
8.97k
  }
2013
4.16k
  case AArch64_OP_GROUP_PredicateAsCounter_0:
2014
4.23k
  case AArch64_OP_GROUP_PredicateAsCounter_16:
2015
4.33k
  case AArch64_OP_GROUP_PredicateAsCounter_32:
2016
5.28k
  case AArch64_OP_GROUP_PredicateAsCounter_64:
2017
5.79k
  case AArch64_OP_GROUP_PredicateAsCounter_8: {
2018
5.79k
    unsigned EltSize = temp_arg_0;
2019
5.79k
    AArch64_get_detail_op(MI, 0)->vas = EltSize;
2020
5.79k
    AArch64_set_detail_op_reg(
2021
5.79k
      MI, OpNum, MCInst_getOpVal(MI, OpNum));
2022
5.79k
    break;
2023
5.28k
  }
2024
1.60k
  case AArch64_OP_GROUP_PrefetchOp_0:
2025
5.95k
  case AArch64_OP_GROUP_PrefetchOp_1: {
2026
5.95k
    bool IsSVEPrefetch = (bool)temp_arg_0;
2027
5.95k
    unsigned prfop = MCInst_getOpVal(MI, (OpNum));
2028
5.95k
    aarch64_sysop sysop = { 0 };
2029
5.95k
    if (IsSVEPrefetch) {
2030
4.34k
      const AArch64SVEPRFM_SVEPRFM *PRFM =
2031
4.34k
        AArch64SVEPRFM_lookupSVEPRFMByEncoding(prfop);
2032
4.34k
      if (PRFM) {
2033
3.49k
        sysop.alias = PRFM->SysAlias;
2034
3.49k
        sysop.sub_type = AARCH64_OP_SVEPRFM;
2035
3.49k
        AArch64_set_detail_op_sys(MI, OpNum, sysop,
2036
3.49k
                AARCH64_OP_SYSALIAS);
2037
3.49k
        break;
2038
3.49k
      }
2039
4.34k
    } else {
2040
1.60k
      const AArch64PRFM_PRFM *PRFM =
2041
1.60k
        AArch64PRFM_lookupPRFMByEncoding(prfop);
2042
1.60k
      if (PRFM &&
2043
1.60k
          AArch64_testFeatureList(MI->csh->mode,
2044
1.04k
                PRFM->FeaturesRequired)) {
2045
1.04k
        sysop.alias = PRFM->SysAlias;
2046
1.04k
        sysop.sub_type = AARCH64_OP_PRFM;
2047
1.04k
        AArch64_set_detail_op_sys(MI, OpNum, sysop,
2048
1.04k
                AARCH64_OP_SYSALIAS);
2049
1.04k
        break;
2050
1.04k
      }
2051
1.60k
    }
2052
1.40k
    AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_IMM;
2053
1.40k
    AArch64_get_detail_op(MI, 0)->imm = prfop;
2054
1.40k
    AArch64_get_detail_op(MI, 0)->access =
2055
1.40k
      map_get_op_access(MI, OpNum);
2056
1.40k
    AArch64_inc_op_count(MI);
2057
1.40k
    break;
2058
5.95k
  }
2059
391
  case AArch64_OP_GROUP_SImm_16:
2060
678
  case AArch64_OP_GROUP_SImm_8: {
2061
678
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
2062
678
            MCInst_getOpVal(MI, OpNum));
2063
678
    break;
2064
391
  }
2065
607
  case AArch64_OP_GROUP_SVELogicalImm_int16_t:
2066
2.49k
  case AArch64_OP_GROUP_SVELogicalImm_int32_t:
2067
3.64k
  case AArch64_OP_GROUP_SVELogicalImm_int64_t: {
2068
    // General issue here that we do not save the operand type
2069
    // for each operand. So we choose the largest type.
2070
3.64k
    uint64_t Val = MCInst_getOpVal(MI, OpNum);
2071
3.64k
    uint64_t DecodedVal =
2072
3.64k
      AArch64_AM_decodeLogicalImmediate(Val, 64);
2073
3.64k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
2074
3.64k
            DecodedVal);
2075
3.64k
    break;
2076
2.49k
  }
2077
48.1k
  case AArch64_OP_GROUP_SVERegOp_0:
2078
80.4k
  case AArch64_OP_GROUP_SVERegOp_b:
2079
108k
  case AArch64_OP_GROUP_SVERegOp_d:
2080
139k
  case AArch64_OP_GROUP_SVERegOp_h:
2081
141k
  case AArch64_OP_GROUP_SVERegOp_q:
2082
170k
  case AArch64_OP_GROUP_SVERegOp_s: {
2083
170k
    char Suffix = (char)temp_arg_0;
2084
170k
    AArch64_get_detail_op(MI, 0)->vas = get_vl_by_suffix(Suffix);
2085
170k
    AArch64_set_detail_op_reg(MI, OpNum,
2086
170k
            MCInst_getOpVal(MI, OpNum));
2087
170k
    break;
2088
141k
  }
2089
1.86k
  case AArch64_OP_GROUP_UImm12Offset_1:
2090
2.17k
  case AArch64_OP_GROUP_UImm12Offset_16:
2091
3.53k
  case AArch64_OP_GROUP_UImm12Offset_2:
2092
4.44k
  case AArch64_OP_GROUP_UImm12Offset_4:
2093
4.94k
  case AArch64_OP_GROUP_UImm12Offset_8: {
2094
    // Otherwise it is an expression. For which we only add the immediate
2095
4.94k
    unsigned Scale = MCOperand_isImm(MCInst_getOperand(MI, OpNum)) ? temp_arg_0 : 1;
2096
4.94k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
2097
4.94k
            Scale * MCInst_getOpVal(MI, OpNum));
2098
4.94k
    break;
2099
4.44k
  }
2100
36.2k
  case AArch64_OP_GROUP_VectorIndex_1:
2101
36.2k
  case AArch64_OP_GROUP_VectorIndex_8: {
2102
36.2k
    CS_ASSERT_RET(AArch64_get_detail(MI)->op_count > 0);
2103
36.2k
    unsigned Scale = temp_arg_0;
2104
36.2k
    unsigned VIndex = Scale * MCInst_getOpVal(MI, OpNum);
2105
    // The index can either be for one operand, or for each operand of a list.
2106
36.2k
    if (!AArch64_get_detail_op(MI, -1)->is_list_member) {
2107
15.1k
      AArch64_get_detail_op(MI, -1)->vector_index = VIndex;
2108
15.1k
      break;
2109
15.1k
    }
2110
73.7k
    for (int i = AArch64_get_detail(MI)->op_count - 1; i >= 0;
2111
52.7k
         --i) {
2112
52.7k
      if (!AArch64_get_detail(MI)->operands[i].is_list_member)
2113
0
        break;
2114
52.7k
      AArch64_get_detail(MI)->operands[i].vector_index =
2115
52.7k
        VIndex;
2116
52.7k
    }
2117
21.0k
    break;
2118
36.2k
  }
2119
22
  case AArch64_OP_GROUP_ZPRasFPR_128:
2120
383
  case AArch64_OP_GROUP_ZPRasFPR_16:
2121
610
  case AArch64_OP_GROUP_ZPRasFPR_32:
2122
1.01k
  case AArch64_OP_GROUP_ZPRasFPR_64:
2123
1.06k
  case AArch64_OP_GROUP_ZPRasFPR_8: {
2124
1.06k
    unsigned Base = AArch64_NoRegister;
2125
1.06k
    unsigned Width = temp_arg_0;
2126
1.06k
    switch (Width) {
2127
52
    case 8:
2128
52
      Base = AArch64_B0;
2129
52
      break;
2130
361
    case 16:
2131
361
      Base = AArch64_H0;
2132
361
      break;
2133
227
    case 32:
2134
227
      Base = AArch64_S0;
2135
227
      break;
2136
402
    case 64:
2137
402
      Base = AArch64_D0;
2138
402
      break;
2139
22
    case 128:
2140
22
      Base = AArch64_Q0;
2141
22
      break;
2142
0
    default:
2143
0
      CS_ASSERT_RET(0 && "Unsupported width");
2144
1.06k
    }
2145
1.06k
    unsigned Reg = MCInst_getOpVal(MI, (OpNum));
2146
1.06k
    AArch64_set_detail_op_reg(MI, OpNum, Reg - AArch64_Z0 + Base);
2147
1.06k
    break;
2148
1.06k
  }
2149
305k
  }
2150
305k
}
2151
2152
/// Fills cs_detail with the data of the operand.
2153
/// This function handles operands which original printer function is a template
2154
/// with two arguments.
2155
void AArch64_add_cs_detail_2(MCInst *MI, aarch64_op_group op_group,
2156
             unsigned OpNum, uint64_t temp_arg_0,
2157
             uint64_t temp_arg_1)
2158
73.8k
{
2159
73.8k
  if (!add_cs_detail_begin(MI, OpNum))
2160
0
    return;
2161
73.8k
  switch (op_group) {
2162
0
  default:
2163
0
    printf("ERROR: Operand group %d not handled!\n", op_group);
2164
0
    CS_ASSERT_RET(0);
2165
600
  case AArch64_OP_GROUP_ComplexRotationOp_180_90:
2166
2.65k
  case AArch64_OP_GROUP_ComplexRotationOp_90_0: {
2167
2.65k
    unsigned Angle = temp_arg_0;
2168
2.65k
    unsigned Remainder = temp_arg_1;
2169
2.65k
    unsigned Imm = (MCInst_getOpVal(MI, OpNum) * Angle) + Remainder;
2170
2.65k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Imm);
2171
2.65k
    break;
2172
600
  }
2173
156
  case AArch64_OP_GROUP_ExactFPImm_AArch64ExactFPImm_half_AArch64ExactFPImm_one:
2174
1.87k
  case AArch64_OP_GROUP_ExactFPImm_AArch64ExactFPImm_half_AArch64ExactFPImm_two:
2175
2.67k
  case AArch64_OP_GROUP_ExactFPImm_AArch64ExactFPImm_zero_AArch64ExactFPImm_one: {
2176
2.67k
    aarch64_exactfpimm ImmIs0 = temp_arg_0;
2177
2.67k
    aarch64_exactfpimm ImmIs1 = temp_arg_1;
2178
2.67k
    const AArch64ExactFPImm_ExactFPImm *Imm0Desc =
2179
2.67k
      AArch64ExactFPImm_lookupExactFPImmByEnum(ImmIs0);
2180
2.67k
    const AArch64ExactFPImm_ExactFPImm *Imm1Desc =
2181
2.67k
      AArch64ExactFPImm_lookupExactFPImmByEnum(ImmIs1);
2182
2.67k
    unsigned Val = MCInst_getOpVal(MI, (OpNum));
2183
2.67k
    aarch64_sysop sysop = { 0 };
2184
2.67k
    sysop.imm = Val ? Imm1Desc->SysImm : Imm0Desc->SysImm;
2185
2.67k
    sysop.sub_type = AARCH64_OP_EXACTFPIMM;
2186
2.67k
    AArch64_set_detail_op_sys(MI, OpNum, sysop, AARCH64_OP_SYSIMM);
2187
2.67k
    break;
2188
1.87k
  }
2189
2.68k
  case AArch64_OP_GROUP_ImmRangeScale_2_1:
2190
5.49k
  case AArch64_OP_GROUP_ImmRangeScale_4_3: {
2191
5.49k
    uint64_t Scale = temp_arg_0;
2192
5.49k
    uint64_t Offset = temp_arg_1;
2193
5.49k
    unsigned FirstImm = Scale * MCInst_getOpVal(MI, (OpNum));
2194
5.49k
    AArch64_set_detail_op_imm_range(MI, OpNum, FirstImm, FirstImm + Offset);
2195
5.49k
    break;
2196
2.68k
  }
2197
34
  case AArch64_OP_GROUP_MemExtend_w_128:
2198
550
  case AArch64_OP_GROUP_MemExtend_w_16:
2199
620
  case AArch64_OP_GROUP_MemExtend_w_32:
2200
874
  case AArch64_OP_GROUP_MemExtend_w_64:
2201
1.68k
  case AArch64_OP_GROUP_MemExtend_w_8:
2202
1.70k
  case AArch64_OP_GROUP_MemExtend_x_128:
2203
2.13k
  case AArch64_OP_GROUP_MemExtend_x_16:
2204
2.34k
  case AArch64_OP_GROUP_MemExtend_x_32:
2205
3.16k
  case AArch64_OP_GROUP_MemExtend_x_64:
2206
3.59k
  case AArch64_OP_GROUP_MemExtend_x_8: {
2207
3.59k
    char SrcRegKind = (char)temp_arg_0;
2208
3.59k
    unsigned ExtWidth = temp_arg_1;
2209
3.59k
    bool SignExtend = MCInst_getOpVal(MI, OpNum);
2210
3.59k
    bool DoShift = MCInst_getOpVal(MI, OpNum + 1);
2211
3.59k
    AArch64_set_detail_shift_ext(MI, OpNum, SignExtend, DoShift,
2212
3.59k
               ExtWidth, SrcRegKind);
2213
3.59k
    break;
2214
3.16k
  }
2215
12.1k
  case AArch64_OP_GROUP_TypedVectorList_0_b:
2216
30.0k
  case AArch64_OP_GROUP_TypedVectorList_0_d:
2217
41.7k
  case AArch64_OP_GROUP_TypedVectorList_0_h:
2218
42.7k
  case AArch64_OP_GROUP_TypedVectorList_0_q:
2219
52.7k
  case AArch64_OP_GROUP_TypedVectorList_0_s:
2220
52.8k
  case AArch64_OP_GROUP_TypedVectorList_0_0:
2221
54.3k
  case AArch64_OP_GROUP_TypedVectorList_16_b:
2222
55.1k
  case AArch64_OP_GROUP_TypedVectorList_1_d:
2223
56.2k
  case AArch64_OP_GROUP_TypedVectorList_2_d:
2224
57.0k
  case AArch64_OP_GROUP_TypedVectorList_2_s:
2225
57.8k
  case AArch64_OP_GROUP_TypedVectorList_4_h:
2226
58.1k
  case AArch64_OP_GROUP_TypedVectorList_4_s:
2227
58.7k
  case AArch64_OP_GROUP_TypedVectorList_8_b:
2228
59.4k
  case AArch64_OP_GROUP_TypedVectorList_8_h: {
2229
59.4k
    uint8_t NumLanes = (uint8_t)temp_arg_0;
2230
59.4k
    char LaneKind = (char)temp_arg_1;
2231
59.4k
    uint16_t Pair = ((NumLanes << 8) | LaneKind);
2232
2233
59.4k
    AArch64Layout_VectorLayout vas = AARCH64LAYOUT_INVALID;
2234
59.4k
    switch (Pair) {
2235
0
    default:
2236
0
      printf("Typed vector list with NumLanes = %d and LaneKind = %c not handled.\n",
2237
0
             NumLanes, LaneKind);
2238
0
      CS_ASSERT_RET(0);
2239
554
    case ((8 << 8) | 'b'):
2240
554
      vas = AARCH64LAYOUT_VL_8B;
2241
554
      break;
2242
838
    case ((4 << 8) | 'h'):
2243
838
      vas = AARCH64LAYOUT_VL_4H;
2244
838
      break;
2245
848
    case ((2 << 8) | 's'):
2246
848
      vas = AARCH64LAYOUT_VL_2S;
2247
848
      break;
2248
825
    case ((1 << 8) | 'd'):
2249
825
      vas = AARCH64LAYOUT_VL_1D;
2250
825
      break;
2251
1.54k
    case ((16 << 8) | 'b'):
2252
1.54k
      vas = AARCH64LAYOUT_VL_16B;
2253
1.54k
      break;
2254
702
    case ((8 << 8) | 'h'):
2255
702
      vas = AARCH64LAYOUT_VL_8H;
2256
702
      break;
2257
253
    case ((4 << 8) | 's'):
2258
253
      vas = AARCH64LAYOUT_VL_4S;
2259
253
      break;
2260
1.03k
    case ((2 << 8) | 'd'):
2261
1.03k
      vas = AARCH64LAYOUT_VL_2D;
2262
1.03k
      break;
2263
12.1k
    case 'b':
2264
12.1k
      vas = AARCH64LAYOUT_VL_B;
2265
12.1k
      break;
2266
11.7k
    case 'h':
2267
11.7k
      vas = AARCH64LAYOUT_VL_H;
2268
11.7k
      break;
2269
9.97k
    case 's':
2270
9.97k
      vas = AARCH64LAYOUT_VL_S;
2271
9.97k
      break;
2272
17.8k
    case 'd':
2273
17.8k
      vas = AARCH64LAYOUT_VL_D;
2274
17.8k
      break;
2275
959
    case 'q':
2276
959
      vas = AARCH64LAYOUT_VL_Q;
2277
959
      break;
2278
73
    case '0':
2279
      // Implicitly Typed register
2280
73
      break;
2281
59.4k
    }
2282
2283
59.4k
    unsigned Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
2284
59.4k
    unsigned NumRegs = get_vec_list_num_regs(MI, Reg);
2285
59.4k
    unsigned Stride = get_vec_list_stride(MI, Reg);
2286
59.4k
    Reg = get_vec_list_first_reg(MI, Reg);
2287
2288
59.4k
    if ((MCRegisterClass_contains(
2289
59.4k
           MCRegisterInfo_getRegClass(MI->MRI,
2290
59.4k
              AArch64_ZPRRegClassID),
2291
59.4k
           Reg) ||
2292
59.4k
         MCRegisterClass_contains(
2293
27.9k
           MCRegisterInfo_getRegClass(MI->MRI,
2294
27.9k
              AArch64_PPRRegClassID),
2295
27.9k
           Reg)) &&
2296
59.4k
        NumRegs > 1 && Stride == 1 &&
2297
59.4k
        Reg < getNextVectorRegister(Reg, NumRegs - 1)) {
2298
17.1k
      AArch64_get_detail_op(MI, 0)->is_list_member = true;
2299
17.1k
      AArch64_get_detail_op(MI, 0)->vas = vas;
2300
17.1k
      AArch64_set_detail_op_reg(MI, OpNum, Reg);
2301
17.1k
      if (NumRegs > 1) {
2302
        // Add all registers of the list to the details.
2303
49.0k
        for (size_t i = 0; i < NumRegs - 1; ++i) {
2304
31.8k
          AArch64_get_detail_op(MI, 0)->is_list_member =
2305
31.8k
            true;
2306
31.8k
          AArch64_get_detail_op(MI, 0)->vas = vas;
2307
31.8k
          AArch64_set_detail_op_reg(
2308
31.8k
            MI, OpNum,
2309
31.8k
            getNextVectorRegister(Reg + i, 1));
2310
31.8k
        }
2311
17.1k
      }
2312
42.2k
    } else {
2313
131k
      for (unsigned i = 0; i < NumRegs;
2314
89.7k
           ++i, Reg = getNextVectorRegister(Reg, Stride)) {
2315
89.7k
        if (!(MCRegisterClass_contains(
2316
89.7k
            MCRegisterInfo_getRegClass(
2317
89.7k
              MI->MRI, AArch64_ZPRRegClassID),
2318
89.7k
            Reg) ||
2319
89.7k
          MCRegisterClass_contains(
2320
70.6k
            MCRegisterInfo_getRegClass(
2321
70.6k
              MI->MRI, AArch64_PPRRegClassID),
2322
70.6k
            Reg))) {
2323
70.5k
          AArch64_get_detail_op(MI, 0)->is_vreg = true;
2324
70.5k
        }
2325
89.7k
        AArch64_get_detail_op(MI, 0)->is_list_member =
2326
89.7k
          true;
2327
89.7k
        AArch64_get_detail_op(MI, 0)->vas = vas;
2328
89.7k
        AArch64_set_detail_op_reg(MI, OpNum, Reg);
2329
89.7k
      }
2330
42.2k
    }
2331
59.4k
  }
2332
73.8k
  }
2333
73.8k
}
2334
2335
/// Fills cs_detail with the data of the operand.
2336
/// This function handles operands which original printer function is a template
2337
/// with four arguments.
2338
void AArch64_add_cs_detail_4(MCInst *MI, aarch64_op_group op_group,
2339
             unsigned OpNum, uint64_t temp_arg_0,
2340
             uint64_t temp_arg_1, uint64_t temp_arg_2,
2341
             uint64_t temp_arg_3)
2342
16.0k
{
2343
16.0k
  if (!add_cs_detail_begin(MI, OpNum))
2344
0
    return;
2345
16.0k
  switch (op_group) {
2346
0
  default:
2347
0
    printf("ERROR: Operand group %d not handled!\n", op_group);
2348
0
    CS_ASSERT_RET(0);
2349
402
  case AArch64_OP_GROUP_RegWithShiftExtend_0_128_x_0:
2350
633
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_w_d:
2351
816
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_w_s:
2352
1.76k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_x_0:
2353
2.02k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_x_d:
2354
2.05k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_x_s:
2355
2.45k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_w_d:
2356
2.54k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_w_s:
2357
3.47k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_x_0:
2358
3.95k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_x_d:
2359
4.14k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_x_s:
2360
4.64k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_w_d:
2361
4.66k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_w_s:
2362
6.22k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_x_0:
2363
6.41k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_x_d:
2364
6.46k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_x_s:
2365
8.30k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_w_d:
2366
8.86k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_w_s:
2367
11.0k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_x_0:
2368
11.9k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_x_d:
2369
12.0k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_x_s:
2370
12.5k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_16_w_d:
2371
12.6k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_16_w_s:
2372
13.2k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_32_w_d:
2373
13.3k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_32_w_s:
2374
13.6k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_64_w_d:
2375
13.8k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_64_w_s:
2376
15.3k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_8_w_d:
2377
16.0k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_8_w_s: {
2378
    // signed (s) and unsigned (u) extend
2379
16.0k
    bool SignExtend = (bool)temp_arg_0;
2380
    // Extend width
2381
16.0k
    int ExtWidth = (int)temp_arg_1;
2382
    // w = word, x = doubleword
2383
16.0k
    char SrcRegKind = (char)temp_arg_2;
2384
    // Vector register element/arrangement specifier:
2385
    // B = 8bit, H = 16bit, S = 32bit, D = 64bit, Q = 128bit
2386
    // No suffix = complete register
2387
    // According to: ARM Reference manual supplement, doc number: DDI 0584
2388
16.0k
    char Suffix = (char)temp_arg_3;
2389
2390
    // Register will be added in printOperand() afterwards. Here we only handle
2391
    // shift and extend.
2392
16.0k
    AArch64_get_detail_op(MI, -1)->vas = get_vl_by_suffix(Suffix);
2393
2394
16.0k
    bool DoShift = ExtWidth != 8;
2395
16.0k
    if (!(SignExtend || DoShift || SrcRegKind == 'w'))
2396
3.13k
      return;
2397
2398
12.8k
    AArch64_set_detail_shift_ext(MI, OpNum, SignExtend, DoShift,
2399
12.8k
               ExtWidth, SrcRegKind);
2400
12.8k
    break;
2401
16.0k
  }
2402
16.0k
  }
2403
16.0k
}
2404
2405
/// Adds a register AArch64 operand at position OpNum and increases the op_count by
2406
/// one.
2407
void AArch64_set_detail_op_reg(MCInst *MI, unsigned OpNum, aarch64_reg Reg)
2408
755k
{
2409
755k
  if (!detail_is_set(MI))
2410
0
    return;
2411
755k
  AArch64_check_safe_inc(MI);
2412
2413
755k
  if (Reg == AARCH64_REG_ZA ||
2414
755k
      (Reg >= AARCH64_REG_ZAB0 && Reg < AARCH64_REG_ZT0)) {
2415
    // A tile register should be treated as SME operand.
2416
0
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
2417
0
            sme_reg_to_vas(Reg));
2418
0
    return;
2419
755k
  } else if (((Reg >= AARCH64_REG_P0) && (Reg <= AARCH64_REG_P15)) ||
2420
755k
             ((Reg >= AARCH64_REG_PN0) && (Reg <= AARCH64_REG_PN15))) {
2421
    // SME/SVE predicate register.
2422
61.5k
    AArch64_set_detail_op_pred(MI, OpNum);
2423
61.5k
    return;
2424
693k
  } else if (AArch64_get_detail(MI)->is_doing_sme) {
2425
15.4k
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) & CS_OP_BOUND);
2426
15.4k
    if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_SME) {
2427
15.2k
      AArch64_set_detail_op_sme(MI, OpNum,
2428
15.2k
              AARCH64_SME_MATRIX_SLICE_REG,
2429
15.2k
              AARCH64LAYOUT_INVALID);
2430
15.2k
    } else if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_PRED) {
2431
234
      AArch64_set_detail_op_pred(MI, OpNum);
2432
234
    } else {
2433
0
      CS_ASSERT_RET(0 && "Unkown SME/SVE operand type");
2434
0
    }
2435
15.4k
    return;
2436
15.4k
  }
2437
678k
  if (map_get_op_type(MI, OpNum) & CS_OP_MEM) {
2438
127k
    AArch64_set_detail_op_mem(MI, OpNum, Reg);
2439
127k
    return;
2440
127k
  }
2441
2442
551k
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_BOUND));
2443
551k
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_MEM));
2444
551k
  CS_ASSERT_RET(map_get_op_type(MI, OpNum) == CS_OP_REG);
2445
2446
551k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_REG;
2447
551k
  AArch64_get_detail_op(MI, 0)->reg = Reg;
2448
551k
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2449
551k
  AArch64_inc_op_count(MI);
2450
551k
}
2451
2452
/// Check if the previous operand is a memory operand
2453
/// with only the base register set AND if this base register
2454
/// is write-back.
2455
/// This indicates the following immediate is a post-indexed
2456
/// memory offset.
2457
127k
static bool prev_is_membase_wb(MCInst *MI) {
2458
127k
  return AArch64_get_detail(MI)->op_count > 0 &&
2459
127k
         AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_MEM &&
2460
127k
         AArch64_get_detail_op(MI, -1)->mem.disp == 0 &&
2461
127k
         get_detail(MI)->writeback;
2462
127k
}
2463
2464
/// Adds an immediate AArch64 operand at position OpNum and increases the op_count
2465
/// by one.
2466
void AArch64_set_detail_op_imm(MCInst *MI, unsigned OpNum,
2467
             aarch64_op_type ImmType, int64_t Imm)
2468
169k
{
2469
169k
  if (!detail_is_set(MI))
2470
0
    return;
2471
169k
  AArch64_check_safe_inc(MI);
2472
2473
169k
  if (AArch64_get_detail(MI)->is_doing_sme) {
2474
0
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) & CS_OP_BOUND);
2475
0
    if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_SME) {
2476
0
      AArch64_set_detail_op_sme(MI, OpNum,
2477
0
              AARCH64_SME_MATRIX_SLICE_OFF,
2478
0
              AARCH64LAYOUT_INVALID, (uint32_t) 1);
2479
0
    } else if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_PRED) {
2480
0
      AArch64_set_detail_op_pred(MI, OpNum);
2481
0
    } else {
2482
0
      CS_ASSERT_RET(0 && "Unkown SME operand type");
2483
0
    }
2484
0
    return;
2485
0
  }
2486
169k
  if (map_get_op_type(MI, OpNum) & CS_OP_MEM || prev_is_membase_wb(MI)) {
2487
49.8k
    AArch64_set_detail_op_mem(MI, OpNum, Imm);
2488
49.8k
    return;
2489
49.8k
  }
2490
2491
119k
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_MEM));
2492
119k
  CS_ASSERT_RET((map_get_op_type(MI, OpNum) & ~CS_OP_BOUND) == CS_OP_IMM);
2493
119k
  CS_ASSERT_RET(ImmType == AARCH64_OP_IMM || ImmType == AARCH64_OP_CIMM);
2494
2495
119k
  AArch64_get_detail_op(MI, 0)->type = ImmType;
2496
119k
  AArch64_get_detail_op(MI, 0)->imm = Imm;
2497
119k
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2498
119k
  AArch64_inc_op_count(MI);
2499
119k
}
2500
2501
void AArch64_set_detail_op_imm_range(MCInst *MI, unsigned OpNum,
2502
             uint32_t FirstImm, uint32_t Offset)
2503
5.49k
{
2504
5.49k
  if (!detail_is_set(MI))
2505
0
    return;
2506
5.49k
  AArch64_check_safe_inc(MI);
2507
2508
5.49k
  if (AArch64_get_detail(MI)->is_doing_sme) {
2509
5.49k
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) & CS_OP_BOUND);
2510
5.49k
    if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_SME) {
2511
5.49k
      AArch64_set_detail_op_sme(MI, OpNum,
2512
5.49k
              AARCH64_SME_MATRIX_SLICE_OFF_RANGE,
2513
5.49k
              AARCH64LAYOUT_INVALID, (uint32_t) FirstImm,
2514
5.49k
              (uint32_t) Offset);
2515
5.49k
    } else if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_PRED) {
2516
0
      CS_ASSERT_RET(0 && "Unkown SME predicate imm range type");
2517
0
    } else {
2518
0
      CS_ASSERT_RET(0 && "Unkown SME operand type");
2519
0
    }
2520
5.49k
    return;
2521
5.49k
  }
2522
2523
0
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_MEM));
2524
0
  CS_ASSERT_RET(map_get_op_type(MI, OpNum) == CS_OP_IMM);
2525
2526
0
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_IMM_RANGE;
2527
0
  AArch64_get_detail_op(MI, 0)->imm_range.first = FirstImm;
2528
0
  AArch64_get_detail_op(MI, 0)->imm_range.offset = Offset;
2529
0
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2530
0
  AArch64_inc_op_count(MI);
2531
0
}
2532
2533
/// Adds a memory AARCH64 operand at position OpNum. op_count is *not* increased by
2534
/// one. This is done by set_mem_access().
2535
void AArch64_set_detail_op_mem(MCInst *MI, unsigned OpNum, uint64_t Val)
2536
177k
{
2537
177k
  if (!detail_is_set(MI))
2538
0
    return;
2539
177k
  AArch64_check_safe_inc(MI);
2540
2541
177k
  AArch64_set_mem_access(MI, true);
2542
2543
177k
  cs_op_type secondary_type = map_get_op_type(MI, OpNum) & ~CS_OP_MEM;
2544
177k
  switch (secondary_type) {
2545
0
  default:
2546
0
    CS_ASSERT_RET(0 && "Secondary type not supported yet.");
2547
127k
  case CS_OP_REG: {
2548
127k
    bool is_index_reg = AArch64_get_detail_op(MI, 0)->mem.base !=
2549
127k
            AARCH64_REG_INVALID;
2550
127k
    if (is_index_reg)
2551
23.8k
      AArch64_get_detail_op(MI, 0)->mem.index = Val;
2552
103k
    else {
2553
103k
      AArch64_get_detail_op(MI, 0)->mem.base = Val;
2554
103k
    }
2555
2556
127k
    if (MCInst_opIsTying(MI, OpNum)) {
2557
      // Especially base registers can be writeback registers.
2558
      // For this they tie an MC operand which has write
2559
      // access. But this one is never processed in the printer
2560
      // (because it is never emitted). Therefor it is never
2561
      // added to the modified list.
2562
      // Here we check for this case and add the memory register
2563
      // to the modified list.
2564
29.8k
      map_add_implicit_write(MI, MCInst_getOpVal(MI, OpNum));
2565
29.8k
    }
2566
127k
    break;
2567
0
  }
2568
49.8k
  case CS_OP_IMM: {
2569
49.8k
    AArch64_get_detail_op(MI, 0)->mem.disp = Val;
2570
49.8k
    break;
2571
0
  }
2572
177k
  }
2573
2574
177k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_MEM;
2575
177k
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2576
177k
  AArch64_set_mem_access(MI, false);
2577
177k
}
2578
2579
/// Adds the shift and sign extend info to the previous operand.
2580
/// op_count is *not* incremented by one.
2581
void AArch64_set_detail_shift_ext(MCInst *MI, unsigned OpNum, bool SignExtend,
2582
          bool DoShift, unsigned ExtWidth,
2583
          char SrcRegKind)
2584
16.4k
{
2585
16.4k
  bool IsLSL = !SignExtend && SrcRegKind == 'x';
2586
16.4k
  if (IsLSL)
2587
5.86k
    AArch64_get_detail_op(MI, -1)->shift.type = AARCH64_SFT_LSL;
2588
10.6k
  else {
2589
10.6k
    aarch64_extender ext = SignExtend ? AARCH64_EXT_SXTB :
2590
10.6k
                AARCH64_EXT_UXTB;
2591
10.6k
    switch (SrcRegKind) {
2592
0
    default:
2593
0
      CS_ASSERT_RET(0 && "Extender not handled\n");
2594
0
    case 'b':
2595
0
      ext += 0;
2596
0
      break;
2597
0
    case 'h':
2598
0
      ext += 1;
2599
0
      break;
2600
9.53k
    case 'w':
2601
9.53k
      ext += 2;
2602
9.53k
      break;
2603
1.06k
    case 'x':
2604
1.06k
      ext += 3;
2605
1.06k
      break;
2606
10.6k
    }
2607
10.6k
    AArch64_get_detail_op(MI, -1)->ext = ext;
2608
10.6k
  }
2609
16.4k
  if (DoShift || IsLSL) {
2610
10.4k
    unsigned ShiftAmount = DoShift ? Log2_32(ExtWidth / 8) : 0;
2611
10.4k
    AArch64_get_detail_op(MI, -1)->shift.type = AARCH64_SFT_LSL;
2612
10.4k
    AArch64_get_detail_op(MI, -1)->shift.value = ShiftAmount;
2613
10.4k
  }
2614
16.4k
}
2615
2616
/// Transforms the immediate of the operand to a float and stores it.
2617
/// Increments the op_counter by one.
2618
void AArch64_set_detail_op_float(MCInst *MI, unsigned OpNum, float Val)
2619
206
{
2620
206
  if (!detail_is_set(MI))
2621
0
    return;
2622
206
  AArch64_check_safe_inc(MI);
2623
2624
206
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_FP;
2625
206
  AArch64_get_detail_op(MI, 0)->fp = Val;
2626
206
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2627
206
  AArch64_inc_op_count(MI);
2628
206
}
2629
2630
/// Adds a the system operand and increases the op_count by
2631
/// one.
2632
void AArch64_set_detail_op_sys(MCInst *MI, unsigned OpNum, aarch64_sysop sys_op,
2633
             aarch64_op_type type)
2634
23.2k
{
2635
23.2k
  if (!detail_is_set(MI))
2636
0
    return;
2637
23.2k
  AArch64_check_safe_inc(MI);
2638
2639
23.2k
  AArch64_get_detail_op(MI, 0)->type = type;
2640
23.2k
  AArch64_get_detail_op(MI, 0)->sysop = sys_op;
2641
23.2k
  if (sys_op.sub_type == AARCH64_OP_EXACTFPIMM) {
2642
2.67k
    AArch64_get_detail_op(MI, 0)->fp = aarch64_exact_fp_to_fp(sys_op.imm.exactfpimm);
2643
2.67k
  }
2644
23.2k
  AArch64_inc_op_count(MI);
2645
23.2k
}
2646
2647
62.0k
void AArch64_set_detail_op_pred(MCInst *MI, unsigned OpNum) {
2648
62.0k
  if (!detail_is_set(MI))
2649
0
    return;
2650
62.0k
  AArch64_check_safe_inc(MI);
2651
2652
62.0k
  if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_INVALID) {
2653
61.2k
    setup_pred_operand(MI);
2654
61.2k
  }
2655
62.0k
  aarch64_op_pred *p = &AArch64_get_detail_op(MI, 0)->pred;
2656
62.0k
  if (p->reg == AARCH64_REG_INVALID) {
2657
61.2k
    p->reg = MCInst_getOpVal(MI, OpNum);
2658
61.2k
    AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2659
61.2k
    AArch64_get_detail(MI)->is_doing_sme = true;
2660
61.2k
    return;
2661
61.2k
  } else if (p->vec_select == AARCH64_REG_INVALID) {
2662
538
    p->vec_select = MCInst_getOpVal(MI, OpNum);
2663
538
    return;
2664
538
  } else if (p->imm_index == -1) {
2665
234
    p->imm_index = MCInst_getOpVal(MI, OpNum);
2666
234
    return;
2667
234
  }
2668
0
  CS_ASSERT_RET(0 && "Should not be reached.");
2669
0
}
2670
2671
/// Adds a SME matrix component to a SME operand.
2672
void AArch64_set_detail_op_sme(MCInst *MI, unsigned OpNum,
2673
             aarch64_sme_op_part part,
2674
             AArch64Layout_VectorLayout vas, ...)
2675
50.4k
{
2676
50.4k
  if (!detail_is_set(MI))
2677
0
    return;
2678
50.4k
  AArch64_check_safe_inc(MI);
2679
2680
50.4k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_SME;
2681
50.4k
  switch (part) {
2682
0
  default:
2683
0
    printf("Unhandled SME operand part %d\n", part);
2684
0
    CS_ASSERT_RET(0);
2685
2.69k
  case AARCH64_SME_MATRIX_TILE_LIST: {
2686
2.69k
    setup_sme_operand(MI);
2687
2.69k
    va_list args;
2688
2.69k
    va_start(args, vas);
2689
2.69k
    int Tile = va_arg(args, int); // NOLINT(clang-analyzer-valist.Uninitialized)
2690
2.69k
    va_end(args);
2691
2.69k
    AArch64_get_detail_op(MI, 0)->sme.type = AARCH64_SME_OP_TILE;
2692
2.69k
    AArch64_get_detail_op(MI, 0)->sme.tile = Tile;
2693
2.69k
    AArch64_get_detail_op(MI, 0)->vas = vas;
2694
2.69k
    AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2695
2.69k
    AArch64_get_detail(MI)->is_doing_sme = true;
2696
2.69k
    break;
2697
0
  }
2698
17.3k
  case AARCH64_SME_MATRIX_TILE:
2699
17.3k
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) == CS_OP_REG);
2700
2701
17.3k
    setup_sme_operand(MI);
2702
17.3k
    AArch64_get_detail_op(MI, 0)->sme.type = AARCH64_SME_OP_TILE;
2703
17.3k
    AArch64_get_detail_op(MI, 0)->sme.tile =
2704
17.3k
      MCInst_getOpVal(MI, OpNum);
2705
17.3k
    AArch64_get_detail_op(MI, 0)->vas = vas;
2706
17.3k
    AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2707
17.3k
    AArch64_get_detail(MI)->is_doing_sme = true;
2708
17.3k
    break;
2709
15.2k
  case AARCH64_SME_MATRIX_SLICE_REG:
2710
15.2k
    CS_ASSERT_RET((map_get_op_type(MI, OpNum) & ~(CS_OP_MEM | CS_OP_BOUND)) == CS_OP_REG);
2711
15.2k
    CS_ASSERT_RET(AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_SME);
2712
2713
    // SME operand already present. Add the slice to it.
2714
15.2k
    AArch64_get_detail_op(MI, 0)->sme.type =
2715
15.2k
      AARCH64_SME_OP_TILE_VEC;
2716
15.2k
    AArch64_get_detail_op(MI, 0)->sme.slice_reg =
2717
15.2k
      MCInst_getOpVal(MI, OpNum);
2718
15.2k
    break;
2719
9.73k
  case AARCH64_SME_MATRIX_SLICE_OFF: {
2720
9.73k
    CS_ASSERT_RET((map_get_op_type(MI, OpNum) & ~(CS_OP_MEM | CS_OP_BOUND)) == CS_OP_IMM);
2721
    // Because we took care of the slice register before, the op at -1 must be a SME operand.
2722
9.73k
    CS_ASSERT_RET(AArch64_get_detail_op(MI, 0)->type ==
2723
9.73k
           AARCH64_OP_SME);
2724
9.73k
    CS_ASSERT_RET(AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm ==
2725
9.73k
           AARCH64_SLICE_IMM_INVALID);
2726
9.73k
    va_list args;
2727
9.73k
    va_start(args, vas);
2728
9.73k
    uint16_t offset = va_arg(args, uint32_t); // NOLINT(clang-analyzer-valist.Uninitialized)
2729
9.73k
    va_end(args);
2730
9.73k
    AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm =
2731
9.73k
      offset;
2732
9.73k
    break;
2733
0
  }
2734
5.49k
  case AARCH64_SME_MATRIX_SLICE_OFF_RANGE: {
2735
5.49k
    va_list args;
2736
5.49k
    va_start(args, vas);
2737
5.49k
    uint8_t First = va_arg(args, uint32_t); // NOLINT(clang-analyzer-valist.Uninitialized)
2738
5.49k
    uint8_t Offset = va_arg(args, uint32_t); // NOLINT(clang-analyzer-valist.Uninitialized)
2739
5.49k
    va_end(args);
2740
5.49k
    AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.first =
2741
5.49k
      First;
2742
5.49k
    AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.offset =
2743
5.49k
      Offset;
2744
5.49k
    AArch64_get_detail_op(MI, 0)->sme.has_range_offset = true;
2745
5.49k
    break;
2746
0
  }
2747
50.4k
  }
2748
50.4k
}
2749
2750
static void insert_op(MCInst *MI, unsigned index, cs_aarch64_op op)
2751
14.4k
{
2752
14.4k
  if (!detail_is_set(MI)) {
2753
0
    return;
2754
0
  }
2755
2756
14.4k
  AArch64_check_safe_inc(MI);
2757
14.4k
  cs_aarch64_op *ops = AArch64_get_detail(MI)->operands;
2758
14.4k
  int i = AArch64_get_detail(MI)->op_count;
2759
14.4k
  if (index == -1) {
2760
14.4k
    ops[i] = op;
2761
14.4k
    AArch64_inc_op_count(MI);
2762
14.4k
    return;
2763
14.4k
  }
2764
0
  for (; i > 0 && i > index; --i) {
2765
0
    ops[i] = ops[i - 1];
2766
0
  }
2767
0
  ops[index] = op;
2768
0
  AArch64_inc_op_count(MI);
2769
0
}
2770
2771
/// Inserts a float to the detail operands at @index.
2772
/// If @index == -1, it pushes the operand to the end of the ops array.
2773
/// Already present operands are moved.
2774
void AArch64_insert_detail_op_float_at(MCInst *MI, unsigned index, double val,
2775
               cs_ac_type access)
2776
0
{
2777
0
  if (!detail_is_set(MI))
2778
0
    return;
2779
2780
0
  AArch64_check_safe_inc(MI);
2781
2782
0
  cs_aarch64_op op;
2783
0
  AArch64_setup_op(&op);
2784
0
  op.type = AARCH64_OP_FP;
2785
0
  op.fp = val;
2786
0
  op.access = access;
2787
2788
0
  insert_op(MI, index, op);
2789
0
}
2790
2791
/// Inserts a register to the detail operands at @index.
2792
/// If @index == -1, it pushes the operand to the end of the ops array.
2793
/// Already present operands are moved.
2794
void AArch64_insert_detail_op_reg_at(MCInst *MI, unsigned index,
2795
             aarch64_reg Reg, cs_ac_type access)
2796
2.68k
{
2797
2.68k
  if (!detail_is_set(MI))
2798
0
    return;
2799
2800
2.68k
  AArch64_check_safe_inc(MI);
2801
2802
2.68k
  cs_aarch64_op op;
2803
2.68k
  AArch64_setup_op(&op);
2804
2.68k
  op.type = AARCH64_OP_REG;
2805
2.68k
  op.reg = Reg;
2806
2.68k
  op.access = access;
2807
2808
2.68k
  insert_op(MI, index, op);
2809
2.68k
}
2810
2811
/// Inserts a immediate to the detail operands at @index.
2812
/// If @index == -1, it pushes the operand to the end of the ops array.
2813
/// Already present operands are moved.
2814
void AArch64_insert_detail_op_imm_at(MCInst *MI, unsigned index, int64_t Imm)
2815
4.08k
{
2816
4.08k
  if (!detail_is_set(MI))
2817
0
    return;
2818
4.08k
  AArch64_check_safe_inc(MI);
2819
2820
4.08k
  cs_aarch64_op op;
2821
4.08k
  AArch64_setup_op(&op);
2822
4.08k
  op.type = AARCH64_OP_IMM;
2823
4.08k
  op.imm = Imm;
2824
4.08k
  op.access = CS_AC_READ;
2825
2826
4.08k
  insert_op(MI, index, op);
2827
4.08k
}
2828
2829
void AArch64_insert_detail_op_sys(MCInst *MI, unsigned index, aarch64_sysop sys_op,
2830
             aarch64_op_type type)
2831
4.80k
{
2832
4.80k
  if (!detail_is_set(MI))
2833
0
    return;
2834
4.80k
  AArch64_check_safe_inc(MI);
2835
2836
4.80k
  cs_aarch64_op op;
2837
4.80k
  AArch64_setup_op(&op);
2838
4.80k
  op.type = type;
2839
4.80k
  op.sysop = sys_op;
2840
4.80k
  if (op.sysop.sub_type == AARCH64_OP_EXACTFPIMM) {
2841
4.56k
    op.fp = aarch64_exact_fp_to_fp(op.sysop.imm.exactfpimm);
2842
4.56k
  }
2843
4.80k
  insert_op(MI, index, op);
2844
4.80k
}
2845
2846
2847
void AArch64_insert_detail_op_sme(MCInst *MI, unsigned index, aarch64_op_sme sme_op)
2848
2.83k
{
2849
2.83k
  if (!detail_is_set(MI))
2850
0
    return;
2851
2.83k
  AArch64_check_safe_inc(MI);
2852
2853
2.83k
  cs_aarch64_op op;
2854
2.83k
  AArch64_setup_op(&op);
2855
2.83k
  op.type = AARCH64_OP_SME;
2856
2.83k
  op.sme = sme_op;
2857
2.83k
  insert_op(MI, index, op);
2858
2.83k
}
2859
2860
#endif