Coverage Report

Created: 2025-07-11 06:32

/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.77k
#define CHAR(c) #c[0]
23
24
7.52k
static float aarch64_exact_fp_to_fp(aarch64_exactfpimm exact) {
25
7.52k
  switch (exact) {
26
0
  default:
27
0
    CS_ASSERT(0 && "Not handled.");
28
0
    return 999.0;
29
456
  case AARCH64_EXACTFPIMM_HALF:
30
456
    return 0.5;
31
1.13k
  case AARCH64_EXACTFPIMM_ONE:
32
1.13k
    return 1.0;
33
996
  case AARCH64_EXACTFPIMM_TWO:
34
996
    return 2.0;
35
4.93k
  case AARCH64_EXACTFPIMM_ZERO:
36
4.93k
    return 0.0;
37
7.52k
  }
38
7.52k
}
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.4k
{
100
10.4k
  MCRegisterInfo_InitMCRegisterInfo(
101
10.4k
    MRI, AArch64RegDesc, AARCH64_REG_ENDING, 0, 0,
102
10.4k
    AArch64MCRegisterClasses, ARR_SIZE(AArch64MCRegisterClasses), 0,
103
10.4k
    0, AArch64RegDiffLists, 0, AArch64SubRegIdxLists,
104
10.4k
    ARR_SIZE(AArch64SubRegIdxLists), 0);
105
10.4k
}
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
27.3k
{
111
27.3k
  if (!detail_is_set(MI))
112
0
    return;
113
114
27.3k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_SME;
115
27.3k
  AArch64_get_detail_op(MI, 0)->sme.type = AARCH64_SME_OP_INVALID;
116
27.3k
  AArch64_get_detail_op(MI, 0)->sme.tile = AARCH64_REG_INVALID;
117
27.3k
  AArch64_get_detail_op(MI, 0)->sme.slice_reg = AARCH64_REG_INVALID;
118
27.3k
  AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm = AARCH64_SLICE_IMM_INVALID;
119
27.3k
  AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.first = AARCH64_SLICE_IMM_RANGE_INVALID;
120
27.3k
  AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.offset = AARCH64_SLICE_IMM_RANGE_INVALID;
121
27.3k
}
122
123
static void setup_pred_operand(MCInst *MI)
124
53.0k
{
125
53.0k
  if (!detail_is_set(MI))
126
0
    return;
127
128
53.0k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_PRED;
129
53.0k
  AArch64_get_detail_op(MI, 0)->pred.imm_index = -1;
130
53.0k
}
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
64.0k
{
161
64.0k
  switch (reg) {
162
679
  case AARCH64_REG_X29:
163
679
    return "fp";
164
2.13k
  case AARCH64_REG_X30:
165
2.13k
    return "lr";
166
64.0k
  }
167
61.2k
  return NULL;
168
64.0k
}
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
282k
{
175
282k
  if (!detail_is_set(MI)) {
176
0
    return;
177
0
  }
178
179
282k
  if (AArch64_get_detail(MI)->op_count == 0) {
180
633
    return;
181
633
  }
182
282k
  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
282k
  const char *vl_ptr = strchr(OS->buffer, '.');
192
624k
  while (vl_ptr) {
193
    // Number after dot?
194
341k
    unsigned num = 0;
195
341k
    if (strchr("1248", vl_ptr[1])) {
196
70.8k
      num = atoi(vl_ptr + 1);
197
70.8k
      vl_ptr = num > 9 ? vl_ptr + 3 : vl_ptr + 2;
198
271k
    } else {
199
271k
      vl_ptr++;
200
271k
    }
201
202
    // Layout letter
203
341k
    char letter = '\0';
204
341k
    if (strchr("bhsdq", vl_ptr[0])) {
205
330k
      letter = vl_ptr[0];
206
330k
    }
207
341k
    if (!letter) {
208
11.7k
      goto next_dot_continue;
209
11.7k
    }
210
211
330k
    AArch64Layout_VectorLayout vl = AARCH64LAYOUT_INVALID;
212
330k
    switch (letter) {
213
0
    default:
214
0
      CS_ASSERT_RET(0 && "Unhandled vector layout letter.");
215
0
      return;
216
74.5k
    case 'b':
217
74.5k
      vl = AARCH64LAYOUT_VL_B;
218
74.5k
      break;
219
89.5k
    case 'h':
220
89.5k
      vl = AARCH64LAYOUT_VL_H;
221
89.5k
      break;
222
76.0k
    case 's':
223
76.0k
      vl = AARCH64LAYOUT_VL_S;
224
76.0k
      break;
225
85.5k
    case 'd':
226
85.5k
      vl = AARCH64LAYOUT_VL_D;
227
85.5k
      break;
228
4.27k
    case 'q':
229
4.27k
      vl = AARCH64LAYOUT_VL_Q;
230
4.27k
      break;
231
330k
    }
232
330k
    vl |= (num << 8);
233
234
    // Determine op index by searching for trailing commata after op string
235
330k
    uint32_t op_idx = 0;
236
330k
    const char *comma_ptr = strchr(OS->buffer, ',');
237
330k
    ;
238
751k
    while (comma_ptr && comma_ptr < vl_ptr) {
239
421k
      ++op_idx;
240
421k
      comma_ptr = strchr(comma_ptr + 1, ',');
241
421k
    }
242
330k
    if (!comma_ptr) {
243
      // Last op doesn't have a trailing commata.
244
50.3k
      op_idx = AArch64_get_detail(MI)->op_count - 1;
245
50.3k
    }
246
330k
    if (op_idx >= AArch64_get_detail(MI)->op_count) {
247
      // A memory operand with a commata in [base, dist]
248
12.2k
      op_idx = AArch64_get_detail(MI)->op_count - 1;
249
12.2k
    }
250
251
    // Search for the operand this one belongs to.
252
330k
    cs_aarch64_op *op = &AArch64_get_detail(MI)->operands[op_idx];
253
330k
    if ((op->type != AARCH64_OP_REG &&
254
330k
         op->type != AARCH64_OP_SME) ||
255
330k
        op->vas != AARCH64LAYOUT_INVALID) {
256
276k
      goto next_dot_continue;
257
276k
    }
258
53.9k
    op->vas = vl;
259
260
341k
next_dot_continue:
261
341k
    vl_ptr = strchr(vl_ptr + 1, '.');
262
341k
  }
263
282k
}
264
265
const char *AArch64_reg_name(csh handle, unsigned int reg)
266
64.0k
{
267
64.0k
  int syntax_opt = ((cs_struct *)(uintptr_t)handle)->syntax;
268
64.0k
  const char *alias = get_custom_reg_alias(reg);
269
64.0k
  if ((syntax_opt & CS_OPT_SYNTAX_CS_REG_ALIAS) && alias)
270
0
    return alias;
271
272
64.0k
  if (((cs_struct *)(uintptr_t)handle)->syntax &
273
64.0k
      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
64.0k
  return AArch64_LLVM_getRegisterName(reg, AArch64_NoRegAltName);
278
64.0k
}
279
280
void AArch64_setup_op(cs_aarch64_op *op)
281
4.63M
{
282
4.63M
  memset(op, 0, sizeof(cs_aarch64_op));
283
4.63M
  op->type = AARCH64_OP_INVALID;
284
4.63M
  op->vector_index = -1;
285
4.63M
}
286
287
void AArch64_init_cs_detail(MCInst *MI)
288
288k
{
289
288k
  if (detail_is_set(MI)) {
290
288k
    memset(get_detail(MI), 0,
291
288k
           offsetof(cs_detail, aarch64) + sizeof(cs_aarch64));
292
4.91M
    for (int i = 0; i < ARR_SIZE(AArch64_get_detail(MI)->operands);
293
4.62M
         i++)
294
4.62M
      AArch64_setup_op(&AArch64_get_detail(MI)->operands[i]);
295
288k
    AArch64_get_detail(MI)->cc = AArch64CC_Invalid;
296
288k
  }
297
288k
}
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
282k
{
310
282k
  if (AArch64_get_detail(MI)->post_index) {
311
0
    return true;
312
0
  }
313
282k
  cs_aarch64_op *memop = NULL;
314
995k
  for (int i = 0; i < AArch64_get_detail(MI)->op_count; ++i) {
315
813k
    if (AArch64_get_detail(MI)->operands[i].type & CS_OP_MEM) {
316
100k
      memop = &AArch64_get_detail(MI)->operands[i];
317
100k
      break;
318
100k
    }
319
813k
  }
320
282k
  if (!memop)
321
181k
    return false;
322
100k
  if (memop->mem.base == AARCH64_REG_INVALID) {
323
    // Load/Store from/to label. Has no register base.
324
5.32k
    return false;
325
5.32k
  }
326
95.5k
  const char *membase = AArch64_LLVM_getRegisterName(
327
95.5k
    memop->mem.base, AArch64_NoRegAltName);
328
95.5k
  int64_t memdisp = memop->mem.disp;
329
95.5k
  SStream pattern = { 0 };
330
95.5k
  SStream_concat(&pattern, membase);
331
95.5k
  SStream_concat(&pattern, "], ");
332
95.5k
  printInt32Bang(&pattern, memdisp);
333
95.5k
  return strstr(OS->buffer, pattern.buffer) != NULL;
334
100k
}
335
336
static void AArch64_check_updates_flags(MCInst *MI)
337
282k
{
338
282k
#ifndef CAPSTONE_DIET
339
282k
  if (!detail_is_set(MI))
340
0
    return;
341
282k
  cs_detail *detail = get_detail(MI);
342
  // Implicitly written registers
343
312k
  for (int i = 0; i < detail->regs_write_count; ++i) {
344
45.3k
    if (detail->regs_write[i] == 0)
345
0
      break;
346
75.4k
    for (int j = 0; j < ARR_SIZE(aarch64_flag_regs); ++j) {
347
45.3k
      if (detail->regs_write[i] == aarch64_flag_regs[j]) {
348
15.2k
        detail->aarch64.update_flags = true;
349
15.2k
        return;
350
15.2k
      }
351
45.3k
    }
352
45.3k
  }
353
1.05M
  for (int i = 0; i < detail->aarch64.op_count; ++i) {
354
787k
    if (detail->aarch64.operands[i].type == AARCH64_OP_SYSREG &&
355
787k
        detail->aarch64.operands[i].sysop.sub_type ==
356
7.04k
          AARCH64_OP_REG_MSR) {
357
23.3k
      for (int j = 0; j < ARR_SIZE(aarch64_flag_sys_regs);
358
19.3k
           ++j)
359
19.4k
        if (detail->aarch64.operands[i]
360
19.4k
              .sysop.reg.sysreg ==
361
19.4k
            aarch64_flag_sys_regs[j]) {
362
109
          detail->aarch64.update_flags = true;
363
109
          return;
364
109
        }
365
783k
    } else if (detail->aarch64.operands[i].type == AARCH64_OP_REG &&
366
783k
         detail->aarch64.operands[i].access & CS_AC_WRITE) {
367
458k
      for (int j = 0; j < ARR_SIZE(aarch64_flag_regs); ++j)
368
229k
        if (detail->aarch64.operands[i].reg ==
369
229k
            aarch64_flag_regs[j]) {
370
0
          detail->aarch64.update_flags = true;
371
0
          return;
372
0
        }
373
229k
    }
374
787k
  }
375
267k
#endif // CAPSTONE_DIET
376
267k
}
377
378
257
static aarch64_shifter id_to_shifter(unsigned Opcode) {
379
257
  switch (Opcode) {
380
0
  default:
381
0
    return AARCH64_SFT_INVALID;
382
78
  case AArch64_RORVXr:
383
88
  case AArch64_RORVWr:
384
88
    return AARCH64_SFT_ROR_REG;
385
66
  case AArch64_LSRVXr:
386
76
  case AArch64_LSRVWr:
387
76
    return AARCH64_SFT_LSR_REG;
388
18
  case AArch64_LSLVXr:
389
39
  case AArch64_LSLVWr:
390
39
    return AARCH64_SFT_LSL_REG;
391
20
  case AArch64_ASRVXr:
392
54
  case AArch64_ASRVWr:
393
54
    return AARCH64_SFT_ASR_REG;
394
257
  }
395
257
}
396
397
static void add_non_alias_details(MCInst *MI)
398
243k
{
399
243k
  unsigned Opcode = MCInst_getOpcode(MI);
400
243k
  switch (Opcode) {
401
232k
  default:
402
232k
    break;
403
232k
  case AArch64_RORVXr:
404
88
  case AArch64_RORVWr:
405
154
  case AArch64_LSRVXr:
406
164
  case AArch64_LSRVWr:
407
182
  case AArch64_LSLVXr:
408
203
  case AArch64_LSLVWr:
409
223
  case AArch64_ASRVXr:
410
257
  case AArch64_ASRVWr:
411
257
    if (AArch64_get_detail(MI)->op_count != 3) {
412
0
      return;
413
0
    }
414
257
    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
257
    uint64_t shift = AArch64_get_detail_op(MI, -1)->reg;
419
257
    cs_aarch64_op *op1 = AArch64_get_detail_op(MI, -2);
420
257
    op1->shift.type = id_to_shifter(Opcode);
421
257
    op1->shift.value = shift;
422
257
    AArch64_dec_op_count(MI);
423
257
    break;
424
146
  case AArch64_FCMPDri:
425
422
  case AArch64_FCMPEDri:
426
1.65k
  case AArch64_FCMPEHri:
427
1.71k
  case AArch64_FCMPESri:
428
2.69k
  case AArch64_FCMPHri:
429
2.85k
  case AArch64_FCMPSri:
430
2.85k
    AArch64_insert_detail_op_reg_at(MI, -1, AARCH64_REG_XZR,
431
2.85k
            CS_AC_READ);
432
2.85k
    break;
433
147
  case AArch64_CMEQv16i8rz:
434
213
  case AArch64_CMEQv1i64rz:
435
231
  case AArch64_CMEQv2i32rz:
436
250
  case AArch64_CMEQv2i64rz:
437
288
  case AArch64_CMEQv4i16rz:
438
381
  case AArch64_CMEQv4i32rz:
439
520
  case AArch64_CMEQv8i16rz:
440
573
  case AArch64_CMEQv8i8rz:
441
641
  case AArch64_CMGEv16i8rz:
442
687
  case AArch64_CMGEv1i64rz:
443
777
  case AArch64_CMGEv2i32rz:
444
856
  case AArch64_CMGEv2i64rz:
445
875
  case AArch64_CMGEv4i16rz:
446
887
  case AArch64_CMGEv4i32rz:
447
918
  case AArch64_CMGEv8i16rz:
448
1.37k
  case AArch64_CMGEv8i8rz:
449
1.43k
  case AArch64_CMGTv16i8rz:
450
1.48k
  case AArch64_CMGTv1i64rz:
451
1.49k
  case AArch64_CMGTv2i32rz:
452
1.69k
  case AArch64_CMGTv2i64rz:
453
1.71k
  case AArch64_CMGTv4i16rz:
454
1.74k
  case AArch64_CMGTv4i32rz:
455
1.80k
  case AArch64_CMGTv8i16rz:
456
2.02k
  case AArch64_CMGTv8i8rz:
457
2.07k
  case AArch64_CMLEv16i8rz:
458
2.08k
  case AArch64_CMLEv1i64rz:
459
2.09k
  case AArch64_CMLEv2i32rz:
460
2.10k
  case AArch64_CMLEv2i64rz:
461
2.12k
  case AArch64_CMLEv4i16rz:
462
2.14k
  case AArch64_CMLEv4i32rz:
463
2.36k
  case AArch64_CMLEv8i16rz:
464
2.46k
  case AArch64_CMLEv8i8rz:
465
2.53k
  case AArch64_CMLTv16i8rz:
466
2.55k
  case AArch64_CMLTv1i64rz:
467
2.57k
  case AArch64_CMLTv2i32rz:
468
2.87k
  case AArch64_CMLTv2i64rz:
469
2.89k
  case AArch64_CMLTv4i16rz:
470
2.93k
  case AArch64_CMLTv4i32rz:
471
3.19k
  case AArch64_CMLTv8i16rz:
472
3.23k
  case AArch64_CMLTv8i8rz:
473
3.23k
    AArch64_insert_detail_op_imm_at(MI, -1, 0);
474
3.23k
    break;
475
68
  case AArch64_FCMEQ_PPzZ0_D:
476
140
  case AArch64_FCMEQ_PPzZ0_H:
477
178
  case AArch64_FCMEQ_PPzZ0_S:
478
733
  case AArch64_FCMEQv1i16rz:
479
748
  case AArch64_FCMEQv1i32rz:
480
841
  case AArch64_FCMEQv1i64rz:
481
877
  case AArch64_FCMEQv2i32rz:
482
943
  case AArch64_FCMEQv2i64rz:
483
977
  case AArch64_FCMEQv4i16rz:
484
997
  case AArch64_FCMEQv4i32rz:
485
1.10k
  case AArch64_FCMEQv8i16rz:
486
1.17k
  case AArch64_FCMGE_PPzZ0_D:
487
1.19k
  case AArch64_FCMGE_PPzZ0_H:
488
1.39k
  case AArch64_FCMGE_PPzZ0_S:
489
1.43k
  case AArch64_FCMGEv1i16rz:
490
1.44k
  case AArch64_FCMGEv1i32rz:
491
1.45k
  case AArch64_FCMGEv1i64rz:
492
1.61k
  case AArch64_FCMGEv2i32rz:
493
1.64k
  case AArch64_FCMGEv2i64rz:
494
1.73k
  case AArch64_FCMGEv4i16rz:
495
1.76k
  case AArch64_FCMGEv4i32rz:
496
1.78k
  case AArch64_FCMGEv8i16rz:
497
1.81k
  case AArch64_FCMGT_PPzZ0_D:
498
1.82k
  case AArch64_FCMGT_PPzZ0_H:
499
2.00k
  case AArch64_FCMGT_PPzZ0_S:
500
2.04k
  case AArch64_FCMGTv1i16rz:
501
2.11k
  case AArch64_FCMGTv1i32rz:
502
2.17k
  case AArch64_FCMGTv1i64rz:
503
2.25k
  case AArch64_FCMGTv2i32rz:
504
2.32k
  case AArch64_FCMGTv2i64rz:
505
2.35k
  case AArch64_FCMGTv4i16rz:
506
2.40k
  case AArch64_FCMGTv4i32rz:
507
2.49k
  case AArch64_FCMGTv8i16rz:
508
2.52k
  case AArch64_FCMLE_PPzZ0_D:
509
2.54k
  case AArch64_FCMLE_PPzZ0_H:
510
2.75k
  case AArch64_FCMLE_PPzZ0_S:
511
2.76k
  case AArch64_FCMLEv1i16rz:
512
2.77k
  case AArch64_FCMLEv1i32rz:
513
2.79k
  case AArch64_FCMLEv1i64rz:
514
3.05k
  case AArch64_FCMLEv2i32rz:
515
3.09k
  case AArch64_FCMLEv2i64rz:
516
3.11k
  case AArch64_FCMLEv4i16rz:
517
3.13k
  case AArch64_FCMLEv4i32rz:
518
3.14k
  case AArch64_FCMLEv8i16rz:
519
3.18k
  case AArch64_FCMLT_PPzZ0_D:
520
3.25k
  case AArch64_FCMLT_PPzZ0_H:
521
3.27k
  case AArch64_FCMLT_PPzZ0_S:
522
3.46k
  case AArch64_FCMLTv1i16rz:
523
3.48k
  case AArch64_FCMLTv1i32rz:
524
3.74k
  case AArch64_FCMLTv1i64rz:
525
3.79k
  case AArch64_FCMLTv2i32rz:
526
3.82k
  case AArch64_FCMLTv2i64rz:
527
3.86k
  case AArch64_FCMLTv4i16rz:
528
3.93k
  case AArch64_FCMLTv4i32rz:
529
4.75k
  case AArch64_FCMLTv8i16rz:
530
4.77k
  case AArch64_FCMNE_PPzZ0_D:
531
4.81k
  case AArch64_FCMNE_PPzZ0_H:
532
4.82k
  case AArch64_FCMNE_PPzZ0_S: {
533
4.82k
    aarch64_sysop sysop = { 0 };
534
4.82k
    sysop.imm.exactfpimm = AARCH64_EXACTFPIMM_ZERO;
535
4.82k
    sysop.sub_type = AARCH64_OP_EXACTFPIMM;
536
4.82k
    AArch64_insert_detail_op_sys(MI, -1, sysop, AARCH64_OP_SYSIMM);
537
4.82k
    break;
538
4.81k
  }
539
243k
  }
540
243k
}
541
542
#define ADD_ZA0_S \
543
444
      { aarch64_op_sme za0_op = { \
544
444
        .type = AARCH64_SME_OP_TILE, \
545
444
        .tile = AARCH64_REG_ZAS0, \
546
444
        .slice_reg = AARCH64_REG_INVALID, \
547
444
        .slice_offset = { -1 }, \
548
444
        .has_range_offset = false, \
549
444
        .is_vertical = false, \
550
444
      }; \
551
444
      AArch64_insert_detail_op_sme(MI, -1, za0_op); \
552
444
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
553
444
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
554
444
      }
555
#define ADD_ZA1_S \
556
1.03k
      { aarch64_op_sme za1_op = { \
557
1.03k
        .type = AARCH64_SME_OP_TILE, \
558
1.03k
        .tile = AARCH64_REG_ZAS1, \
559
1.03k
        .slice_reg = AARCH64_REG_INVALID, \
560
1.03k
        .slice_offset = { -1 }, \
561
1.03k
        .has_range_offset = false, \
562
1.03k
        .is_vertical = false, \
563
1.03k
      }; \
564
1.03k
      AArch64_insert_detail_op_sme(MI, -1, za1_op); \
565
1.03k
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
566
1.03k
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
567
1.03k
      }
568
#define ADD_ZA2_S \
569
964
      { aarch64_op_sme za2_op = { \
570
964
        .type = AARCH64_SME_OP_TILE, \
571
964
        .tile = AARCH64_REG_ZAS2, \
572
964
        .slice_reg = AARCH64_REG_INVALID, \
573
964
        .slice_offset = { -1 }, \
574
964
        .has_range_offset = false, \
575
964
        .is_vertical = false, \
576
964
      }; \
577
964
      AArch64_insert_detail_op_sme(MI, -1, za2_op); \
578
964
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
579
964
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
580
964
      }
581
#define ADD_ZA3_S \
582
744
      { aarch64_op_sme za3_op = { \
583
744
        .type = AARCH64_SME_OP_TILE, \
584
744
        .tile = AARCH64_REG_ZAS3, \
585
744
        .slice_reg = AARCH64_REG_INVALID, \
586
744
        .slice_offset = { -1 }, \
587
744
        .has_range_offset = false, \
588
744
        .is_vertical = false, \
589
744
      }; \
590
744
      AArch64_insert_detail_op_sme(MI, -1, za3_op); \
591
744
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_S; \
592
744
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
593
744
      }
594
#define ADD_ZA \
595
37
      { aarch64_op_sme za_op = \
596
37
        { \
597
37
        .type = AARCH64_SME_OP_TILE, \
598
37
        .tile = AARCH64_REG_ZA, \
599
37
        .slice_reg = AARCH64_REG_INVALID, \
600
37
        .slice_offset = { -1 }, \
601
37
        .has_range_offset = false, \
602
37
        .is_vertical = false, \
603
37
      }; \
604
37
      AArch64_insert_detail_op_sme(MI, -1, za_op); \
605
37
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE; \
606
37
      }
607
608
static void AArch64_add_not_defined_ops(MCInst *MI, const SStream *OS)
609
282k
{
610
282k
  if (!detail_is_set(MI))
611
0
    return;
612
613
282k
  if (!MI->flat_insn->is_alias || !MI->flat_insn->usesAliasDetails) {
614
243k
    add_non_alias_details(MI);
615
243k
    return;
616
243k
  }
617
618
  // Alias details
619
39.1k
  switch (MI->flat_insn->alias_id) {
620
32.9k
  default:
621
32.9k
    return;
622
32.9k
  case AARCH64_INS_ALIAS_ROR:
623
195
    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
195
    bool reg_shift = AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_REG;
629
195
    uint64_t shift = reg_shift ? AArch64_get_detail_op(MI, -1)->reg : AArch64_get_detail_op(MI, -1)->imm;
630
195
    cs_aarch64_op *op1 = AArch64_get_detail_op(MI, -2);
631
195
    op1->shift.type = reg_shift ? AARCH64_SFT_ROR_REG : AARCH64_SFT_ROR;
632
195
    op1->shift.value = shift;
633
195
    AArch64_dec_op_count(MI);
634
195
    break;
635
41
  case AARCH64_INS_ALIAS_FMOV:
636
41
    if (AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_FP) {
637
41
      break;
638
41
    }
639
0
    AArch64_insert_detail_op_float_at(MI, -1, 0.0f, CS_AC_READ);
640
0
    break;
641
78
  case AARCH64_INS_ALIAS_LD1:
642
162
  case AARCH64_INS_ALIAS_LD1R:
643
888
  case AARCH64_INS_ALIAS_LD2:
644
1.19k
  case AARCH64_INS_ALIAS_LD2R:
645
1.45k
  case AARCH64_INS_ALIAS_LD3:
646
1.46k
  case AARCH64_INS_ALIAS_LD3R:
647
2.57k
  case AARCH64_INS_ALIAS_LD4:
648
2.79k
  case AARCH64_INS_ALIAS_LD4R:
649
3.27k
  case AARCH64_INS_ALIAS_ST1:
650
3.66k
  case AARCH64_INS_ALIAS_ST2:
651
3.68k
  case AARCH64_INS_ALIAS_ST3:
652
4.30k
  case AARCH64_INS_ALIAS_ST4: {
653
    // Add post-index disp
654
4.30k
    const char *disp_off = strrchr(OS->buffer, '#');
655
4.30k
    if (!disp_off)
656
0
      return;
657
4.30k
    unsigned disp = atoi(disp_off + 1);
658
4.30k
    AArch64_get_detail_op(MI, -1)->type = AARCH64_OP_MEM;
659
4.30k
    AArch64_get_detail_op(MI, -1)->mem.base =
660
4.30k
      AArch64_get_detail_op(MI, -1)->reg;
661
4.30k
    AArch64_get_detail_op(MI, -1)->mem.disp = disp;
662
4.30k
    AArch64_get_detail(MI)->post_index = true;
663
4.30k
    break;
664
4.30k
  }
665
3
  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
3
    break;
670
127
  case AARCH64_INS_ALIAS_SMSTART:
671
221
  case AARCH64_INS_ALIAS_SMSTOP: {
672
221
    const char *disp_off = NULL;
673
221
    disp_off = strstr(OS->buffer, "smstart\tza");
674
221
    if (disp_off) {
675
117
      aarch64_sysop sysop = { 0 };
676
117
      sysop.alias.svcr = AARCH64_SVCR_SVCRZA;
677
117
      sysop.sub_type = AARCH64_OP_SVCR;
678
117
      AArch64_insert_detail_op_sys(MI, -1, sysop,
679
117
              AARCH64_OP_SYSALIAS);
680
117
      return;
681
117
    }
682
104
    disp_off = strstr(OS->buffer, "smstart\tsm");
683
104
    if (disp_off) {
684
10
      aarch64_sysop sysop = { 0 };
685
10
      sysop.alias.svcr = AARCH64_SVCR_SVCRSM;
686
10
      sysop.sub_type = AARCH64_OP_SVCR;
687
10
      AArch64_insert_detail_op_sys(MI, -1, sysop,
688
10
              AARCH64_OP_SYSALIAS);
689
10
      return;
690
10
    }
691
94
    break;
692
104
  }
693
1.38k
  case AARCH64_INS_ALIAS_ZERO: {
694
    // It is ugly, but the hard coded search patterns do it for now.
695
1.38k
    const char *disp_off = NULL;
696
697
1.38k
    disp_off = strstr(OS->buffer, "{za}");
698
1.38k
    if (disp_off) {
699
37
      ADD_ZA;
700
37
      return;
701
37
    }
702
1.35k
    disp_off = strstr(OS->buffer, "{za1.h}");
703
1.35k
    if (disp_off) {
704
49
      aarch64_op_sme op =
705
49
        {
706
49
        .type = AARCH64_SME_OP_TILE,
707
49
        .tile = AARCH64_REG_ZAH1,
708
49
        .slice_reg = AARCH64_REG_INVALID,
709
49
        .slice_offset = { -1 },
710
49
        .has_range_offset = false,
711
49
        .is_vertical = false,
712
49
      };
713
49
      AArch64_insert_detail_op_sme(MI, -1, op);
714
49
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_H;
715
49
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE;
716
49
      return;
717
49
    }
718
1.30k
    disp_off = strstr(OS->buffer, "{za0.h}");
719
1.30k
    if (disp_off) {
720
19
      aarch64_op_sme op =
721
19
        {
722
19
        .type = AARCH64_SME_OP_TILE,
723
19
        .tile = AARCH64_REG_ZAH0,
724
19
        .slice_reg = AARCH64_REG_INVALID,
725
19
        .slice_offset = { -1 },
726
19
        .has_range_offset = false,
727
19
        .is_vertical = false,
728
19
      };
729
19
      AArch64_insert_detail_op_sme(MI, -1, op);
730
19
      AArch64_get_detail_op(MI, -1)->vas = AARCH64LAYOUT_VL_H;
731
19
      AArch64_get_detail_op(MI, -1)->access = CS_AC_WRITE;
732
19
      return;
733
19
    }
734
1.28k
    disp_off = strstr(OS->buffer, "{za0.s}");
735
1.28k
    if (disp_off) {
736
71
      ADD_ZA0_S;
737
71
      return;
738
71
    }
739
1.21k
    disp_off = strstr(OS->buffer, "{za1.s}");
740
1.21k
    if (disp_off) {
741
18
      ADD_ZA1_S;
742
18
      return;
743
18
    }
744
1.19k
    disp_off = strstr(OS->buffer, "{za2.s}");
745
1.19k
    if (disp_off) {
746
11
      ADD_ZA2_S;
747
11
      return;
748
11
    }
749
1.18k
    disp_off = strstr(OS->buffer, "{za3.s}");
750
1.18k
    if (disp_off) {
751
35
      ADD_ZA3_S;
752
35
      return;
753
35
    }
754
1.14k
    disp_off = strstr(OS->buffer, "{za0.s,za1.s}");
755
1.14k
    if (disp_off) {
756
95
      ADD_ZA0_S;
757
95
      ADD_ZA1_S;
758
95
      return;
759
95
    }
760
1.05k
    disp_off = strstr(OS->buffer, "{za0.s,za3.s}");
761
1.05k
    if (disp_off) {
762
88
      ADD_ZA0_S;
763
88
      ADD_ZA3_S;
764
88
      return;
765
88
    }
766
964
    disp_off = strstr(OS->buffer, "{za1.s,za2.s}");
767
964
    if (disp_off) {
768
184
      ADD_ZA1_S;
769
184
      ADD_ZA2_S;
770
184
      return;
771
184
    }
772
780
    disp_off = strstr(OS->buffer, "{za2.s,za3.s}");
773
780
    if (disp_off) {
774
18
      ADD_ZA2_S;
775
18
      ADD_ZA3_S;
776
18
      return;
777
18
    }
778
762
    disp_off = strstr(OS->buffer, "{za0.s,za1.s,za2.s}");
779
762
    if (disp_off) {
780
159
      ADD_ZA0_S;
781
159
      ADD_ZA1_S;
782
159
      ADD_ZA2_S;
783
159
      return;
784
159
    }
785
603
    disp_off = strstr(OS->buffer, "{za0.s,za1.s,za3.s}");
786
603
    if (disp_off) {
787
11
      ADD_ZA0_S;
788
11
      ADD_ZA1_S;
789
11
      ADD_ZA3_S;
790
11
      return;
791
11
    }
792
592
    disp_off = strstr(OS->buffer, "{za0.s,za2.s,za3.s}");
793
592
    if (disp_off) {
794
20
      ADD_ZA0_S;
795
20
      ADD_ZA2_S;
796
20
      ADD_ZA3_S;
797
20
      return;
798
20
    }
799
572
    disp_off = strstr(OS->buffer, "{za1.s,za2.s,za3.s}");
800
572
    if (disp_off) {
801
572
      ADD_ZA1_S;
802
572
      ADD_ZA2_S;
803
572
      ADD_ZA3_S;
804
572
      return;
805
572
    }
806
0
    break;
807
572
  }
808
39.1k
  }
809
39.1k
}
810
811
void AArch64_set_instr_map_data(MCInst *MI)
812
288k
{
813
288k
  map_cs_id(MI, aarch64_insns, ARR_SIZE(aarch64_insns));
814
288k
  map_implicit_reads(MI, aarch64_insns);
815
288k
  map_implicit_writes(MI, aarch64_insns);
816
288k
  map_groups(MI, aarch64_insns);
817
288k
}
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
288k
{
823
288k
  AArch64_init_cs_detail(MI);
824
288k
  DecodeStatus Result = AArch64_LLVM_getInstruction(handle, code, code_len, MI,
825
288k
              size, address,
826
288k
              info);
827
288k
  AArch64_set_instr_map_data(MI);
828
288k
  if (Result == MCDisassembler_SoftFail) {
829
6.25k
    MCInst_setSoftFail(MI);
830
6.25k
  }
831
288k
  return Result != MCDisassembler_Fail;
832
288k
}
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
282k
{
870
282k
  unsigned Opcode = MI->flat_insn->id;
871
282k
  switch (Opcode) {
872
276k
  default:
873
276k
    return;
874
276k
  case AARCH64_INS_SVC:
875
19
    add_group(MI, AARCH64_GRP_INT);
876
19
    break;
877
34
  case AARCH64_INS_SMC:
878
5.21k
  case AARCH64_INS_MSR:
879
6.20k
  case AARCH64_INS_MRS:
880
6.20k
    add_group(MI, AARCH64_GRP_PRIVILEGE);
881
6.20k
    break;
882
38
  case AARCH64_INS_RET:
883
91
  case AARCH64_INS_RETAA:
884
262
  case AARCH64_INS_RETAB:
885
262
    add_group(MI, AARCH64_GRP_RET);
886
262
    break;
887
282k
  }
888
282k
}
889
890
282k
static void AArch64_correct_mem_access(MCInst *MI) {
891
282k
  if (!detail_is_set(MI))
892
0
    return;
893
282k
  cs_ac_type access = aarch64_insns[MI->Opcode].suppl_info.aarch64.mem_acc;
894
282k
  if (access == CS_AC_INVALID) {
895
188k
    return;
896
188k
  }
897
201k
  for (int i = 0; i < AArch64_get_detail(MI)->op_count; ++i) {
898
199k
    if (AArch64_get_detail_op(MI, -i)->type == AARCH64_OP_MEM) {
899
92.2k
      AArch64_get_detail_op(MI, -i)->access = access;
900
92.2k
      return;
901
92.2k
    }
902
199k
  }
903
94.2k
}
904
905
void AArch64_printer(MCInst *MI, SStream *O, void * /* MCRegisterInfo* */ info)
906
282k
{
907
282k
  MCRegisterInfo *MRI = (MCRegisterInfo *)info;
908
282k
  MI->MRI = MRI;
909
282k
  MI->fillDetailOps = detail_is_set(MI);
910
282k
  MI->flat_insn->usesAliasDetails = map_use_alias_details(MI);
911
282k
  AArch64_LLVM_printInstruction(MI, O, info);
912
282k
  if (detail_is_set(MI)) {
913
282k
    if (AArch64_get_detail(MI)->is_doing_sme) {
914
      // Last operand still needs to be closed.
915
4.46k
      AArch64_get_detail(MI)->is_doing_sme = false;
916
4.46k
      AArch64_inc_op_count(MI);
917
4.46k
    }
918
282k
    AArch64_get_detail(MI)->post_index =
919
282k
      AArch64_check_post_index_am(MI, O);
920
282k
  }
921
282k
  AArch64_check_updates_flags(MI);
922
282k
  map_set_alias_id(MI, O, insn_alias_mnem_map,
923
282k
       ARR_SIZE(insn_alias_mnem_map) - 1);
924
282k
  int syntax_opt = MI->csh->syntax;
925
282k
  if (syntax_opt & CS_OPT_SYNTAX_CS_REG_ALIAS)
926
0
    patch_cs_reg_alias(O->buffer);
927
282k
  AArch64_add_not_defined_ops(MI, O);
928
282k
  AArch64_add_cs_groups(MI);
929
282k
  AArch64_add_vas(MI, O);
930
282k
  AArch64_correct_mem_access(MI);
931
282k
}
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
282k
{
936
  // Done after disassembly
937
282k
  return;
938
282k
}
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
282k
{
946
282k
#ifndef CAPSTONE_DIET
947
282k
  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
282k
  if (id >= AARCH64_INS_ENDING)
956
0
    return NULL;
957
958
282k
  if (id < ARR_SIZE(insn_name_maps))
959
282k
    return insn_name_maps[id];
960
961
  // not found
962
0
  return NULL;
963
#else
964
  return NULL;
965
#endif
966
282k
}
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
390k
{
986
390k
#ifndef CAPSTONE_DIET
987
390k
  return id2name(group_name_maps, ARR_SIZE(group_name_maps), id);
988
#else
989
  return NULL;
990
#endif
991
390k
}
992
993
// map instruction name to public instruction ID
994
aarch64_insn AArch64_map_insn(const char *name)
995
60.0k
{
996
60.0k
  unsigned int i;
997
998
39.6M
  for (i = 1; i < ARR_SIZE(insn_name_maps); i++) {
999
39.6M
    if (!strcmp(name, insn_name_maps[i]))
1000
59.6k
      return i;
1001
39.6M
  }
1002
1003
  // not found
1004
378
  return AARCH64_INS_INVALID;
1005
60.0k
}
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
173k
{
1131
173k
  switch (suffix) {
1132
44.9k
  default:
1133
44.9k
    return AARCH64LAYOUT_INVALID;
1134
26.4k
  case 'b':
1135
26.4k
  case 'B':
1136
26.4k
    return AARCH64LAYOUT_VL_B;
1137
33.1k
  case 'h':
1138
33.1k
  case 'H':
1139
33.1k
    return AARCH64LAYOUT_VL_H;
1140
32.4k
  case 's':
1141
32.4k
  case 'S':
1142
32.4k
    return AARCH64LAYOUT_VL_S;
1143
34.7k
  case 'd':
1144
34.7k
  case 'D':
1145
34.7k
    return AARCH64LAYOUT_VL_D;
1146
2.10k
  case 'q':
1147
2.10k
  case 'Q':
1148
2.10k
    return AARCH64LAYOUT_VL_Q;
1149
173k
  }
1150
173k
}
1151
1152
static unsigned get_vec_list_num_regs(MCInst *MI, unsigned Reg)
1153
59.1k
{
1154
  // Work out how many registers there are in the list (if there is an actual
1155
  // list).
1156
59.1k
  unsigned NumRegs = 1;
1157
59.1k
  if (MCRegisterClass_contains(
1158
59.1k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_DDRegClassID),
1159
59.1k
        Reg) ||
1160
59.1k
      MCRegisterClass_contains(
1161
58.4k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_ZPR2RegClassID),
1162
58.4k
        Reg) ||
1163
59.1k
      MCRegisterClass_contains(
1164
47.5k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_QQRegClassID),
1165
47.5k
        Reg) ||
1166
59.1k
      MCRegisterClass_contains(
1167
41.2k
        MCRegisterInfo_getRegClass(MI->MRI, AArch64_PPR2RegClassID),
1168
41.2k
        Reg) ||
1169
59.1k
      MCRegisterClass_contains(
1170
40.6k
        MCRegisterInfo_getRegClass(MI->MRI,
1171
40.6k
                 AArch64_ZPR2StridedRegClassID),
1172
40.6k
        Reg))
1173
20.4k
    NumRegs = 2;
1174
38.7k
  else if (MCRegisterClass_contains(
1175
38.7k
       MCRegisterInfo_getRegClass(MI->MRI,
1176
38.7k
                AArch64_DDDRegClassID),
1177
38.7k
       Reg) ||
1178
38.7k
     MCRegisterClass_contains(
1179
36.6k
       MCRegisterInfo_getRegClass(MI->MRI,
1180
36.6k
                AArch64_ZPR3RegClassID),
1181
36.6k
       Reg) ||
1182
38.7k
     MCRegisterClass_contains(
1183
36.4k
       MCRegisterInfo_getRegClass(MI->MRI,
1184
36.4k
                AArch64_QQQRegClassID),
1185
36.4k
       Reg))
1186
7.60k
    NumRegs = 3;
1187
31.1k
  else if (MCRegisterClass_contains(
1188
31.1k
       MCRegisterInfo_getRegClass(MI->MRI,
1189
31.1k
                AArch64_DDDDRegClassID),
1190
31.1k
       Reg) ||
1191
31.1k
     MCRegisterClass_contains(
1192
30.4k
       MCRegisterInfo_getRegClass(MI->MRI,
1193
30.4k
                AArch64_ZPR4RegClassID),
1194
30.4k
       Reg) ||
1195
31.1k
     MCRegisterClass_contains(
1196
21.7k
       MCRegisterInfo_getRegClass(MI->MRI,
1197
21.7k
                AArch64_QQQQRegClassID),
1198
21.7k
       Reg) ||
1199
31.1k
     MCRegisterClass_contains(
1200
15.4k
       MCRegisterInfo_getRegClass(
1201
15.4k
         MI->MRI, AArch64_ZPR4StridedRegClassID),
1202
15.4k
       Reg))
1203
16.8k
    NumRegs = 4;
1204
59.1k
  return NumRegs;
1205
59.1k
}
1206
1207
static unsigned get_vec_list_stride(MCInst *MI, unsigned Reg)
1208
59.1k
{
1209
59.1k
  unsigned Stride = 1;
1210
59.1k
  if (MCRegisterClass_contains(
1211
59.1k
        MCRegisterInfo_getRegClass(MI->MRI,
1212
59.1k
                 AArch64_ZPR2StridedRegClassID),
1213
59.1k
        Reg))
1214
1.87k
    Stride = 8;
1215
57.3k
  else if (MCRegisterClass_contains(
1216
57.3k
       MCRegisterInfo_getRegClass(
1217
57.3k
         MI->MRI, AArch64_ZPR4StridedRegClassID),
1218
57.3k
       Reg))
1219
1.20k
    Stride = 4;
1220
59.1k
  return Stride;
1221
59.1k
}
1222
1223
static unsigned get_vec_list_first_reg(MCInst *MI, unsigned RegL)
1224
59.1k
{
1225
59.1k
  unsigned Reg = RegL;
1226
  // Now forget about the list and find out what the first register is.
1227
59.1k
  if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_dsub0))
1228
3.64k
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_dsub0);
1229
55.5k
  else if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_qsub0))
1230
17.8k
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_qsub0);
1231
37.6k
  else if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_zsub0))
1232
22.8k
    Reg = MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_zsub0);
1233
14.8k
  else if (MCRegisterInfo_getSubReg(MI->MRI, RegL, AArch64_psub0))
1234
550
    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.1k
  if (MCRegisterClass_contains(MCRegisterInfo_getRegClass(
1239
59.1k
               MI->MRI, AArch64_FPR64RegClassID),
1240
59.1k
             Reg)) {
1241
3.71k
    const MCRegisterClass *FPR128RC = MCRegisterInfo_getRegClass(
1242
3.71k
      MI->MRI, AArch64_FPR128RegClassID);
1243
3.71k
    Reg = MCRegisterInfo_getMatchingSuperReg(
1244
3.71k
      MI->MRI, Reg, AArch64_dsub, FPR128RC);
1245
3.71k
  }
1246
59.1k
  return Reg;
1247
59.1k
}
1248
1249
static bool is_vector_reg(unsigned Reg)
1250
204k
{
1251
204k
  if ((Reg >= AArch64_Q0) && (Reg <= AArch64_Q31))
1252
70.8k
    return true;
1253
133k
  else if ((Reg >= AArch64_Z0) && (Reg <= AArch64_Z31))
1254
132k
    return true;
1255
1.11k
  else if ((Reg >= AArch64_P0) && (Reg <= AArch64_P15))
1256
1.11k
    return true;
1257
0
  return false;
1258
204k
}
1259
1260
static unsigned getNextVectorRegister(unsigned Reg, unsigned Stride /* = 1 */)
1261
146k
{
1262
351k
  while (Stride--) {
1263
204k
    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
204k
    else if (Reg == AArch64_Q31)
1269
3.75k
      Reg = AArch64_Q0;
1270
    // Vector lists can wrap around.
1271
200k
    else if (Reg == AArch64_Z31)
1272
2.66k
      Reg = AArch64_Z0;
1273
    // Vector lists can wrap around.
1274
198k
    else if (Reg == AArch64_P15)
1275
36
      Reg = AArch64_P0;
1276
198k
    else
1277
      // Assume ordered registers
1278
198k
      ++Reg;
1279
204k
  }
1280
146k
  return Reg;
1281
146k
}
1282
1283
static aarch64_extender llvm_to_cs_ext(AArch64_AM_ShiftExtendType ExtType)
1284
13.2k
{
1285
13.2k
  switch (ExtType) {
1286
10.5k
  default:
1287
10.5k
    return AARCH64_EXT_INVALID;
1288
638
  case AArch64_AM_UXTB:
1289
638
    return AARCH64_EXT_UXTB;
1290
418
  case AArch64_AM_UXTH:
1291
418
    return AARCH64_EXT_UXTH;
1292
250
  case AArch64_AM_UXTW:
1293
250
    return AARCH64_EXT_UXTW;
1294
406
  case AArch64_AM_UXTX:
1295
406
    return AARCH64_EXT_UXTX;
1296
406
  case AArch64_AM_SXTB:
1297
406
    return AARCH64_EXT_SXTB;
1298
69
  case AArch64_AM_SXTH:
1299
69
    return AARCH64_EXT_SXTH;
1300
25
  case AArch64_AM_SXTW:
1301
25
    return AARCH64_EXT_SXTW;
1302
445
  case AArch64_AM_SXTX:
1303
445
    return AARCH64_EXT_SXTX;
1304
13.2k
  }
1305
13.2k
}
1306
1307
static aarch64_shifter llvm_to_cs_shift(AArch64_AM_ShiftExtendType ShiftExtType)
1308
10.5k
{
1309
10.5k
  switch (ShiftExtType) {
1310
0
  default:
1311
0
    return AARCH64_SFT_INVALID;
1312
5.76k
  case AArch64_AM_LSL:
1313
5.76k
    return AARCH64_SFT_LSL;
1314
1.22k
  case AArch64_AM_LSR:
1315
1.22k
    return AARCH64_SFT_LSR;
1316
1.45k
  case AArch64_AM_ASR:
1317
1.45k
    return AARCH64_SFT_ASR;
1318
1.56k
  case AArch64_AM_ROR:
1319
1.56k
    return AARCH64_SFT_ROR;
1320
534
  case AArch64_AM_MSL:
1321
534
    return AARCH64_SFT_MSL;
1322
10.5k
  }
1323
10.5k
}
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
324k
{
1330
324k
  if (!detail_is_set(MI))
1331
0
    return;
1332
324k
  set_doing_mem(MI, status);
1333
324k
  if (status) {
1334
162k
    if (AArch64_get_detail(MI)->op_count > 0 &&
1335
162k
        AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_MEM &&
1336
162k
        AArch64_get_detail_op(MI, -1)->mem.index ==
1337
61.0k
          AARCH64_REG_INVALID &&
1338
162k
        AArch64_get_detail_op(MI, -1)->mem.disp == 0) {
1339
      // Previous memory operand not done yet. Select it.
1340
60.1k
      AArch64_dec_op_count(MI);
1341
60.1k
      return;
1342
60.1k
    }
1343
1344
    // Init a new one.
1345
101k
    AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_MEM;
1346
101k
    AArch64_get_detail_op(MI, 0)->mem.base = AARCH64_REG_INVALID;
1347
101k
    AArch64_get_detail_op(MI, 0)->mem.index = AARCH64_REG_INVALID;
1348
101k
    AArch64_get_detail_op(MI, 0)->mem.disp = 0;
1349
1350
101k
#ifndef CAPSTONE_DIET
1351
101k
    uint8_t access =
1352
101k
      map_get_op_access(MI, AArch64_get_detail(MI)->op_count);
1353
101k
    AArch64_get_detail_op(MI, 0)->access = access;
1354
101k
#endif
1355
162k
  } else {
1356
    // done, select the next operand slot
1357
162k
    AArch64_inc_op_count(MI);
1358
162k
  }
1359
324k
}
1360
1361
/// Common prefix for all AArch64_add_cs_detail_* functions
1362
static bool add_cs_detail_begin(MCInst *MI, unsigned op_num)
1363
880k
{
1364
880k
  if (!detail_is_set(MI) || !map_fill_detail_ops(MI))
1365
0
    return false;
1366
1367
880k
  if (AArch64_get_detail(MI)->is_doing_sme) {
1368
    // Unset the flag if there is no bound operand anymore.
1369
100k
    if (!(map_get_op_type(MI, op_num) & CS_OP_BOUND)) {
1370
66.9k
      AArch64_get_detail(MI)->is_doing_sme = false;
1371
66.9k
      AArch64_inc_op_count(MI);
1372
66.9k
    }
1373
100k
  }
1374
880k
  return true;
1375
880k
}
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
506k
{
1383
506k
  if (!add_cs_detail_begin(MI, OpNum))
1384
0
    return;
1385
1386
  // Fill cs_detail
1387
506k
  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
366k
  case AArch64_OP_GROUP_Operand: {
1392
366k
    cs_op_type primary_op_type = map_get_op_type(MI, OpNum) &
1393
366k
               ~(CS_OP_MEM | CS_OP_BOUND);
1394
366k
    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
308k
    case AARCH64_OP_REG:
1400
308k
      AArch64_set_detail_op_reg(MI, OpNum,
1401
308k
              MCInst_getOpVal(MI, OpNum));
1402
308k
      break;
1403
57.5k
    case AARCH64_OP_IMM:
1404
57.5k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1405
57.5k
              MCInst_getOpVal(MI, OpNum));
1406
57.5k
      break;
1407
497
    case AARCH64_OP_FP: {
1408
      // printOperand does not handle FP operands. But sometimes
1409
      // is used to print FP operands as normal immediate.
1410
497
      AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_IMM;
1411
497
      AArch64_get_detail_op(MI, 0)->imm =
1412
497
        MCInst_getOpVal(MI, OpNum);
1413
497
      AArch64_get_detail_op(MI, 0)->access =
1414
497
        map_get_op_access(MI, OpNum);
1415
497
      AArch64_inc_op_count(MI);
1416
497
      break;
1417
0
    }
1418
366k
    }
1419
366k
    break;
1420
366k
  }
1421
366k
  case AArch64_OP_GROUP_AddSubImm: {
1422
3.60k
    unsigned Val = (MCInst_getOpVal(MI, OpNum) & 0xfff);
1423
3.60k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1424
    // Shift is added in printShifter()
1425
3.60k
    break;
1426
366k
  }
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
366k
  }
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
366k
  }
1451
5.04k
  case AArch64_OP_GROUP_AdrAdrpLabel: {
1452
5.04k
    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.04k
    int64_t Offset = MCInst_getOpVal(MI, OpNum);
1459
5.04k
    uint64_t Address = MI->address;
1460
5.04k
    if (MCInst_getOpcode(MI) == AArch64_ADRP) {
1461
2.63k
      Offset = Offset * 4096;
1462
2.63k
      Address = Address & -4096;
1463
2.63k
    }
1464
5.04k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1465
5.04k
            Address + Offset);
1466
5.04k
    break;
1467
5.04k
  }
1468
12.0k
  case AArch64_OP_GROUP_AlignedLabel: {
1469
12.0k
    if (MCOperand_isImm(MCInst_getOperand(MI, OpNum))) {
1470
11.8k
      int64_t Offset = MCInst_getOpVal(MI, OpNum) * 4;
1471
11.8k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1472
11.8k
              MI->address + Offset);
1473
11.8k
    } else {
1474
      // Expression
1475
192
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1476
192
              MCOperand_isImm(MCInst_getOperand(MI, OpNum)));
1477
192
    }
1478
12.0k
    break;
1479
5.04k
  }
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.04k
  }
1485
2.65k
  case AArch64_OP_GROUP_ArithExtend: {
1486
2.65k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1487
2.65k
    AArch64_AM_ShiftExtendType ExtType =
1488
2.65k
      AArch64_AM_getArithExtendType(Val);
1489
2.65k
    unsigned ShiftVal = AArch64_AM_getArithShiftValue(Val);
1490
1491
2.65k
    AArch64_get_detail_op(MI, -1)->ext = llvm_to_cs_ext(ExtType);
1492
2.65k
    AArch64_get_detail_op(MI, -1)->shift.value = ShiftVal;
1493
2.65k
    AArch64_get_detail_op(MI, -1)->shift.type = AARCH64_SFT_LSL;
1494
2.65k
    break;
1495
5.04k
  }
1496
77
  case AArch64_OP_GROUP_BarriernXSOption: {
1497
77
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1498
77
    aarch64_sysop sysop = { 0 };
1499
77
    const AArch64DBnXS_DBnXS *DB =
1500
77
      AArch64DBnXS_lookupDBnXSByEncoding(Val);
1501
77
    if (DB)
1502
77
      sysop.imm.dbnxs = (aarch64_dbnxs) DB->SysImm.dbnxs;
1503
0
    else
1504
0
      sysop.imm.raw_val = Val;
1505
77
    sysop.sub_type = AARCH64_OP_DBNXS;
1506
77
    AArch64_set_detail_op_sys(MI, OpNum, sysop, AARCH64_OP_SYSIMM);
1507
77
    break;
1508
5.04k
  }
1509
41
  case AArch64_OP_GROUP_AppleSysBarrierOption: {
1510
    // Proprietary stuff. We just add the
1511
    // immediate here.
1512
41
    unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1513
41
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1514
41
    break;
1515
5.04k
  }
1516
376
  case AArch64_OP_GROUP_BarrierOption: {
1517
376
    unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1518
376
    unsigned Opcode = MCInst_getOpcode(MI);
1519
376
    aarch64_sysop sysop = { 0 };
1520
1521
376
    if (Opcode == AArch64_ISB) {
1522
80
      const AArch64ISB_ISB *ISB =
1523
80
        AArch64ISB_lookupISBByEncoding(Val);
1524
80
      if (ISB)
1525
0
        sysop.alias.isb = (aarch64_isb) ISB->SysAlias.isb;
1526
80
      else
1527
80
        sysop.alias.raw_val = Val;
1528
80
      sysop.sub_type = AARCH64_OP_ISB;
1529
80
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1530
80
              AARCH64_OP_SYSALIAS);
1531
296
    } else if (Opcode == AArch64_TSB) {
1532
20
      const AArch64TSB_TSB *TSB =
1533
20
        AArch64TSB_lookupTSBByEncoding(Val);
1534
20
      if (TSB)
1535
20
        sysop.alias.tsb = (aarch64_tsb) TSB->SysAlias.tsb;
1536
0
      else
1537
0
        sysop.alias.raw_val = Val;
1538
20
      sysop.sub_type = AARCH64_OP_TSB;
1539
20
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1540
20
              AARCH64_OP_SYSALIAS);
1541
276
    } else {
1542
276
      const AArch64DB_DB *DB =
1543
276
        AArch64DB_lookupDBByEncoding(Val);
1544
276
      if (DB)
1545
257
        sysop.alias.db = (aarch64_db) DB->SysAlias.db;
1546
19
      else
1547
19
        sysop.alias.raw_val = Val;
1548
276
      sysop.sub_type = AARCH64_OP_DB;
1549
276
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1550
276
              AARCH64_OP_SYSALIAS);
1551
276
    }
1552
376
    break;
1553
5.04k
  }
1554
354
  case AArch64_OP_GROUP_BTIHintOp: {
1555
354
    aarch64_sysop sysop = { 0 };
1556
354
    unsigned btihintop = MCInst_getOpVal(MI, OpNum) ^ 32;
1557
354
    const AArch64BTIHint_BTI *BTI =
1558
354
      AArch64BTIHint_lookupBTIByEncoding(btihintop);
1559
354
    if (BTI)
1560
354
      sysop.alias.bti = (aarch64_bti) BTI->SysAlias.bti;
1561
0
    else
1562
0
      sysop.alias.raw_val = btihintop;
1563
354
    sysop.sub_type = AARCH64_OP_BTI;
1564
354
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1565
354
            AARCH64_OP_SYSALIAS);
1566
354
    break;
1567
5.04k
  }
1568
2.34k
  case AArch64_OP_GROUP_CondCode: {
1569
2.34k
    AArch64_get_detail(MI)->cc = MCInst_getOpVal(MI, OpNum);
1570
2.34k
    break;
1571
5.04k
  }
1572
1.09k
  case AArch64_OP_GROUP_ExtendedRegister: {
1573
1.09k
    AArch64_set_detail_op_reg(MI, OpNum,
1574
1.09k
            MCInst_getOpVal(MI, OpNum));
1575
1.09k
    break;
1576
5.04k
  }
1577
696
  case AArch64_OP_GROUP_FPImmOperand: {
1578
696
    MCOperand *MO = MCInst_getOperand(MI, (OpNum));
1579
696
    float FPImm =
1580
696
      MCOperand_isDFPImm(MO) ?
1581
0
        BitsToDouble(MCOperand_getImm(MO)) :
1582
696
        AArch64_AM_getFPImmFloat(MCOperand_getImm(MO));
1583
696
    AArch64_set_detail_op_float(MI, OpNum, FPImm);
1584
696
    break;
1585
5.04k
  }
1586
5.91k
  case AArch64_OP_GROUP_GPR64as32: {
1587
5.91k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
1588
5.91k
    AArch64_set_detail_op_reg(MI, OpNum, getWRegFromXReg(Reg));
1589
5.91k
    break;
1590
5.04k
  }
1591
59
  case AArch64_OP_GROUP_GPR64x8: {
1592
59
    unsigned Reg = MCInst_getOpVal(MI, (OpNum));
1593
59
    Reg = MCRegisterInfo_getSubReg(MI->MRI, Reg, AArch64_x8sub_0);
1594
59
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1595
59
    break;
1596
5.04k
  }
1597
4.05k
  case AArch64_OP_GROUP_Imm:
1598
4.11k
  case AArch64_OP_GROUP_ImmHex:
1599
4.11k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1600
4.11k
            MCInst_getOpVal(MI, OpNum));
1601
4.11k
    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
52
  case AArch64_OP_GROUP_InverseCondCode: {
1608
52
    AArch64CC_CondCode CC = (AArch64CC_CondCode)MCOperand_getImm(
1609
52
      MCInst_getOperand(MI, (OpNum)));
1610
52
    AArch64_get_detail(MI)->cc = AArch64CC_getInvertedCondCode(CC);
1611
52
    break;
1612
4.05k
  }
1613
1.85k
  case AArch64_OP_GROUP_MatrixTile: {
1614
1.85k
    const char *RegName = AArch64_LLVM_getRegisterName(
1615
1.85k
      MCInst_getOpVal(MI, OpNum), AArch64_NoRegAltName);
1616
1.85k
    const char *Dot = strstr(RegName, ".");
1617
1.85k
    AArch64Layout_VectorLayout vas = AARCH64LAYOUT_INVALID;
1618
1.85k
    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
1.85k
      vas = get_vl_by_suffix(Dot[1]);
1625
1.85k
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
1626
1.85k
            vas);
1627
1.85k
    break;
1628
4.05k
  }
1629
678
  case AArch64_OP_GROUP_MatrixTileList: {
1630
678
    unsigned MaxRegs = 8;
1631
678
    unsigned RegMask = MCInst_getOpVal(MI, (OpNum));
1632
1633
6.10k
    for (unsigned I = 0; I < MaxRegs; ++I) {
1634
5.42k
      unsigned Reg = RegMask & (1 << I);
1635
5.42k
      if (Reg == 0)
1636
2.21k
        continue;
1637
3.21k
      AArch64_get_detail_op(MI, 0)->is_list_member = true;
1638
3.21k
      AArch64_set_detail_op_sme(MI, OpNum,
1639
3.21k
              AARCH64_SME_MATRIX_TILE_LIST,
1640
3.21k
              AARCH64LAYOUT_VL_D,
1641
3.21k
              (int) (AARCH64_REG_ZAD0 + I));
1642
3.21k
      AArch64_inc_op_count(MI);
1643
3.21k
    }
1644
678
    AArch64_get_detail(MI)->is_doing_sme = false;
1645
678
    break;
1646
4.05k
  }
1647
1.45k
  case AArch64_OP_GROUP_MRSSystemRegister:
1648
5.43k
  case AArch64_OP_GROUP_MSRSystemRegister: {
1649
5.43k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1650
5.43k
    const AArch64SysReg_SysReg *Reg =
1651
5.43k
      AArch64SysReg_lookupSysRegByEncoding(Val);
1652
5.43k
    bool Read = (op_group == AArch64_OP_GROUP_MRSSystemRegister) ?
1653
1.45k
            true :
1654
5.43k
            false;
1655
1656
5.43k
    bool isValidSysReg =
1657
5.43k
      (Reg && (Read ? Reg->Readable : Reg->Writeable) &&
1658
5.43k
       AArch64_testFeatureList(MI->csh->mode,
1659
375
             Reg->FeaturesRequired));
1660
1661
5.43k
    if (Reg && !isValidSysReg)
1662
1.02k
      Reg = AArch64SysReg_lookupSysRegByName(Reg->AltName);
1663
5.43k
    aarch64_sysop sysop = { 0 };
1664
    // If Reg is NULL it is a generic system register.
1665
5.43k
    if (Reg)
1666
1.39k
      sysop.reg.sysreg = (aarch64_sysreg) Reg->SysReg.sysreg;
1667
4.04k
    else {
1668
4.04k
      sysop.reg.raw_val = Val;
1669
4.04k
    }
1670
5.43k
    aarch64_op_type type =
1671
5.43k
      (op_group == AArch64_OP_GROUP_MRSSystemRegister) ?
1672
1.45k
        AARCH64_OP_REG_MRS :
1673
5.43k
        AARCH64_OP_REG_MSR;
1674
5.43k
    sysop.sub_type = type;
1675
5.43k
    AArch64_set_detail_op_sys(MI, OpNum, sysop, AARCH64_OP_SYSREG);
1676
5.43k
    break;
1677
1.45k
  }
1678
79
  case AArch64_OP_GROUP_PSBHintOp: {
1679
79
    unsigned psbhintop = MCInst_getOpVal(MI, OpNum);
1680
79
    const AArch64PSBHint_PSB *PSB =
1681
79
      AArch64PSBHint_lookupPSBByEncoding(psbhintop);
1682
79
    aarch64_sysop sysop = { 0 };
1683
79
    if (PSB)
1684
79
      sysop.alias.psb = (aarch64_psb) PSB->SysAlias.psb;
1685
0
    else
1686
0
      sysop.alias.raw_val = psbhintop;
1687
79
    sysop.sub_type = AARCH64_OP_PSB;
1688
79
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1689
79
            AARCH64_OP_SYSALIAS);
1690
79
    break;
1691
1.45k
  }
1692
772
  case AArch64_OP_GROUP_RPRFMOperand: {
1693
772
    unsigned prfop = MCInst_getOpVal(MI, OpNum);
1694
772
    const AArch64PRFM_PRFM *PRFM =
1695
772
      AArch64PRFM_lookupPRFMByEncoding(prfop);
1696
772
    aarch64_sysop sysop = { 0 };
1697
772
    if (PRFM)
1698
689
      sysop.alias.prfm = (aarch64_prfm) PRFM->SysAlias.prfm;
1699
83
    else
1700
83
      sysop.alias.raw_val = prfop;
1701
772
    sysop.sub_type = AARCH64_OP_PRFM;
1702
772
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1703
772
            AARCH64_OP_SYSALIAS);
1704
772
    break;
1705
1.45k
  }
1706
5.80k
  case AArch64_OP_GROUP_ShiftedRegister: {
1707
5.80k
    AArch64_set_detail_op_reg(MI, OpNum,
1708
5.80k
            MCInst_getOpVal(MI, OpNum));
1709
    // Shift part is handled in printShifter()
1710
5.80k
    break;
1711
1.45k
  }
1712
10.5k
  case AArch64_OP_GROUP_Shifter: {
1713
10.5k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1714
10.5k
    AArch64_AM_ShiftExtendType ShExtType =
1715
10.5k
      AArch64_AM_getShiftType(Val);
1716
10.5k
    AArch64_get_detail_op(MI, -1)->ext = llvm_to_cs_ext(ShExtType);
1717
10.5k
    AArch64_get_detail_op(MI, -1)->shift.type =
1718
10.5k
      llvm_to_cs_shift(ShExtType);
1719
10.5k
    AArch64_get_detail_op(MI, -1)->shift.value =
1720
10.5k
      AArch64_AM_getShiftValue(Val);
1721
10.5k
    break;
1722
1.45k
  }
1723
1.33k
  case AArch64_OP_GROUP_SIMDType10Operand: {
1724
1.33k
    unsigned RawVal = MCInst_getOpVal(MI, OpNum);
1725
1.33k
    uint64_t Val = AArch64_AM_decodeAdvSIMDModImmType10(RawVal);
1726
1.33k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1727
1.33k
    break;
1728
1.45k
  }
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
1.45k
  }
1743
7.00k
  case AArch64_OP_GROUP_SVEPattern: {
1744
7.00k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1745
7.00k
    const AArch64SVEPredPattern_SVEPREDPAT *Pat =
1746
7.00k
      AArch64SVEPredPattern_lookupSVEPREDPATByEncoding(Val);
1747
7.00k
    if (!Pat) {
1748
2.69k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1749
2.69k
      break;
1750
2.69k
    }
1751
4.30k
    aarch64_sysop sysop = { 0 };
1752
4.30k
    sysop.alias = Pat->SysAlias;
1753
4.30k
    sysop.sub_type = AARCH64_OP_SVEPREDPAT;
1754
4.30k
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1755
4.30k
            AARCH64_OP_SYSALIAS);
1756
4.30k
    break;
1757
7.00k
  }
1758
674
  case AArch64_OP_GROUP_SVEVecLenSpecifier: {
1759
674
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1760
    // Pattern has only 1 bit
1761
674
    if (Val > 1)
1762
0
      CS_ASSERT_RET(0 && "Invalid vector length specifier");
1763
674
    const AArch64SVEVecLenSpecifier_SVEVECLENSPECIFIER *Pat =
1764
674
      AArch64SVEVecLenSpecifier_lookupSVEVECLENSPECIFIERByEncoding(
1765
674
        Val);
1766
674
    if (!Pat)
1767
0
      break;
1768
674
    aarch64_sysop sysop = { 0 };
1769
674
    sysop.alias = Pat->SysAlias;
1770
674
    sysop.sub_type = AARCH64_OP_SVEVECLENSPECIFIER;
1771
674
    AArch64_set_detail_op_sys(MI, OpNum, sysop,
1772
674
            AARCH64_OP_SYSALIAS);
1773
674
    break;
1774
674
  }
1775
10.6k
  case AArch64_OP_GROUP_SysCROperand: {
1776
10.6k
    uint64_t cimm = MCInst_getOpVal(MI, OpNum);
1777
10.6k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_CIMM, cimm);
1778
10.6k
    break;
1779
674
  }
1780
713
  case AArch64_OP_GROUP_SyspXzrPair: {
1781
713
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
1782
713
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1783
713
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1784
713
    break;
1785
674
  }
1786
1.16k
  case AArch64_OP_GROUP_SystemPStateField: {
1787
1.16k
    unsigned Val = MCInst_getOpVal(MI, OpNum);
1788
1789
1.16k
    aarch64_sysop sysop = { 0 };
1790
1.16k
    const AArch64PState_PStateImm0_15 *PStateImm15 =
1791
1.16k
      AArch64PState_lookupPStateImm0_15ByEncoding(Val);
1792
1.16k
    const AArch64PState_PStateImm0_1 *PStateImm1 =
1793
1.16k
      AArch64PState_lookupPStateImm0_1ByEncoding(Val);
1794
1.16k
    if (PStateImm15 &&
1795
1.16k
        AArch64_testFeatureList(MI->csh->mode,
1796
1.08k
              PStateImm15->FeaturesRequired)) {
1797
1.08k
      sysop.alias = PStateImm15->SysAlias;
1798
1.08k
      sysop.sub_type = AARCH64_OP_PSTATEIMM0_15;
1799
1.08k
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1800
1.08k
              AARCH64_OP_SYSALIAS);
1801
1.08k
    } else if (PStateImm1 &&
1802
80
         AArch64_testFeatureList(
1803
80
           MI->csh->mode,
1804
80
           PStateImm1->FeaturesRequired)) {
1805
80
      sysop.alias = PStateImm1->SysAlias;
1806
80
      sysop.sub_type = AARCH64_OP_PSTATEIMM0_1;
1807
80
      AArch64_set_detail_op_sys(MI, OpNum, sysop,
1808
80
              AARCH64_OP_SYSALIAS);
1809
80
    } else {
1810
0
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1811
0
              Val);
1812
0
    }
1813
1.16k
    break;
1814
674
  }
1815
53.8k
  case AArch64_OP_GROUP_VRegOperand: {
1816
53.8k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
1817
53.8k
    AArch64_get_detail_op(MI, 0)->is_vreg = true;
1818
53.8k
    AArch64_set_detail_op_reg(MI, OpNum, Reg);
1819
53.8k
    break;
1820
674
  }
1821
506k
  }
1822
506k
}
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
287k
{
1830
287k
  if (!add_cs_detail_begin(MI, OpNum))
1831
0
    return;
1832
287k
  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
68
  case AArch64_OP_GROUP_GPRSeqPairsClassOperand_32:
1837
1.57k
  case AArch64_OP_GROUP_GPRSeqPairsClassOperand_64: {
1838
1.57k
    unsigned size = temp_arg_0;
1839
1.57k
    unsigned Reg = MCInst_getOpVal(MI, (OpNum));
1840
1841
1.57k
    unsigned Sube = (size == 32) ? AArch64_sube32 : AArch64_sube64;
1842
1.57k
    unsigned Subo = (size == 32) ? AArch64_subo32 : AArch64_subo64;
1843
1844
1.57k
    unsigned Even = MCRegisterInfo_getSubReg(MI->MRI, Reg, Sube);
1845
1.57k
    unsigned Odd = MCRegisterInfo_getSubReg(MI->MRI, Reg, Subo);
1846
1.57k
    AArch64_set_detail_op_reg(MI, OpNum, Even);
1847
1.57k
    AArch64_set_detail_op_reg(MI, OpNum, Odd);
1848
1.57k
    break;
1849
68
  }
1850
713
  case AArch64_OP_GROUP_Imm8OptLsl_int16_t:
1851
925
  case AArch64_OP_GROUP_Imm8OptLsl_int32_t:
1852
1.27k
  case AArch64_OP_GROUP_Imm8OptLsl_int64_t:
1853
1.44k
  case AArch64_OP_GROUP_Imm8OptLsl_int8_t:
1854
1.91k
  case AArch64_OP_GROUP_Imm8OptLsl_uint16_t:
1855
3.33k
  case AArch64_OP_GROUP_Imm8OptLsl_uint32_t:
1856
4.00k
  case AArch64_OP_GROUP_Imm8OptLsl_uint64_t:
1857
4.06k
  case AArch64_OP_GROUP_Imm8OptLsl_uint8_t: {
1858
4.06k
    unsigned UnscaledVal = MCInst_getOpVal(MI, (OpNum));
1859
4.06k
    unsigned Shift = MCInst_getOpVal(MI, (OpNum + 1));
1860
1861
4.06k
    if ((UnscaledVal == 0) &&
1862
4.06k
        (AArch64_AM_getShiftValue(Shift) != 0)) {
1863
294
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1864
294
              UnscaledVal);
1865
      // Shift is handled in printShifter()
1866
294
      break;
1867
294
    }
1868
1869
3.77k
  #define SCALE_SET(T) \
1870
3.77k
    do { \
1871
3.77k
      T Val; \
1872
3.77k
      if (CHAR(T) == 'i') /* Signed */ \
1873
3.77k
        Val = (int8_t)UnscaledVal * \
1874
1.30k
              (1 << AArch64_AM_getShiftValue(Shift)); \
1875
3.77k
      else \
1876
3.77k
        Val = (uint8_t)UnscaledVal * \
1877
2.46k
              (1 << AArch64_AM_getShiftValue(Shift)); \
1878
3.77k
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val); \
1879
3.77k
    } while (0)
1880
1881
3.77k
    switch (op_group) {
1882
0
    default:
1883
0
      CS_ASSERT_RET(0 &&
1884
0
             "Operand group for Imm8OptLsl not handled.");
1885
663
    case AArch64_OP_GROUP_Imm8OptLsl_int16_t: {
1886
663
      SCALE_SET(int16_t);
1887
663
      break;
1888
0
    }
1889
135
    case AArch64_OP_GROUP_Imm8OptLsl_int32_t: {
1890
135
      SCALE_SET(int32_t);
1891
135
      break;
1892
0
    }
1893
335
    case AArch64_OP_GROUP_Imm8OptLsl_int64_t: {
1894
335
      SCALE_SET(int64_t);
1895
335
      break;
1896
0
    }
1897
169
    case AArch64_OP_GROUP_Imm8OptLsl_int8_t: {
1898
169
      SCALE_SET(int8_t);
1899
169
      break;
1900
0
    }
1901
445
    case AArch64_OP_GROUP_Imm8OptLsl_uint16_t: {
1902
445
      SCALE_SET(uint16_t);
1903
445
      break;
1904
0
    }
1905
1.34k
    case AArch64_OP_GROUP_Imm8OptLsl_uint32_t: {
1906
1.34k
      SCALE_SET(uint32_t);
1907
1.34k
      break;
1908
0
    }
1909
611
    case AArch64_OP_GROUP_Imm8OptLsl_uint64_t: {
1910
611
      SCALE_SET(uint64_t);
1911
611
      break;
1912
0
    }
1913
63
    case AArch64_OP_GROUP_Imm8OptLsl_uint8_t: {
1914
63
      SCALE_SET(uint8_t);
1915
63
      break;
1916
0
    }
1917
3.77k
    }
1918
3.77k
    break;
1919
3.77k
  }
1920
3.77k
  case AArch64_OP_GROUP_ImmScale_16:
1921
5.56k
  case AArch64_OP_GROUP_ImmScale_2:
1922
5.68k
  case AArch64_OP_GROUP_ImmScale_3:
1923
5.73k
  case AArch64_OP_GROUP_ImmScale_32:
1924
14.7k
  case AArch64_OP_GROUP_ImmScale_4:
1925
20.6k
  case AArch64_OP_GROUP_ImmScale_8: {
1926
20.6k
    unsigned Scale = temp_arg_0;
1927
20.6k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1928
20.6k
            Scale * MCInst_getOpVal(MI, OpNum));
1929
20.6k
    break;
1930
14.7k
  }
1931
1.04k
  case AArch64_OP_GROUP_LogicalImm_int16_t:
1932
4.23k
  case AArch64_OP_GROUP_LogicalImm_int32_t:
1933
6.46k
  case AArch64_OP_GROUP_LogicalImm_int64_t:
1934
9.12k
  case AArch64_OP_GROUP_LogicalImm_int8_t: {
1935
9.12k
    unsigned TypeSize = temp_arg_0;
1936
9.12k
    uint64_t Val = AArch64_AM_decodeLogicalImmediate(
1937
9.12k
      MCInst_getOpVal(MI, OpNum), 8 * TypeSize);
1938
9.12k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Val);
1939
9.12k
    break;
1940
6.46k
  }
1941
177
  case AArch64_OP_GROUP_Matrix_0:
1942
3.23k
  case AArch64_OP_GROUP_Matrix_16:
1943
8.76k
  case AArch64_OP_GROUP_Matrix_32:
1944
10.8k
  case AArch64_OP_GROUP_Matrix_64: {
1945
10.8k
    unsigned EltSize = temp_arg_0;
1946
10.8k
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
1947
10.8k
            (AArch64Layout_VectorLayout)EltSize);
1948
10.8k
    break;
1949
8.76k
  }
1950
0
  case AArch64_OP_GROUP_MatrixIndex_0:
1951
10.5k
  case AArch64_OP_GROUP_MatrixIndex_1:
1952
10.8k
  case AArch64_OP_GROUP_MatrixIndex_8: {
1953
10.8k
    unsigned scale = temp_arg_0;
1954
10.8k
    if (AArch64_get_detail_op(MI, 0)->type ==
1955
10.8k
        AARCH64_OP_SME) {
1956
        // The index is part of an SME matrix
1957
10.0k
        AArch64_set_detail_op_sme(MI, OpNum,
1958
10.0k
                AARCH64_SME_MATRIX_SLICE_OFF,
1959
10.0k
                AARCH64LAYOUT_INVALID,
1960
10.0k
                (uint32_t) (MCInst_getOpVal(MI, OpNum) * scale));
1961
10.0k
    } else if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_PRED) {
1962
      // The index is part of a predicate
1963
252
      AArch64_set_detail_op_pred(MI, OpNum);
1964
538
    } else {
1965
      // The index is used for an SVE2 instruction.
1966
538
      AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
1967
538
              scale * MCInst_getOpVal(MI, OpNum));
1968
538
    }
1969
10.8k
    break;
1970
10.5k
  }
1971
2.22k
  case AArch64_OP_GROUP_MatrixTileVector_0:
1972
5.70k
  case AArch64_OP_GROUP_MatrixTileVector_1: {
1973
5.70k
    bool isVertical = temp_arg_0;
1974
5.70k
    const char *RegName = AArch64_LLVM_getRegisterName(
1975
5.70k
      MCInst_getOpVal(MI, OpNum), AArch64_NoRegAltName);
1976
5.70k
    const char *Dot = strstr(RegName, ".");
1977
5.70k
    AArch64Layout_VectorLayout vas = AARCH64LAYOUT_INVALID;
1978
5.70k
    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
5.70k
      vas = get_vl_by_suffix(Dot[1]);
1985
5.70k
    setup_sme_operand(MI);
1986
5.70k
    AArch64_set_detail_op_sme(MI, OpNum, AARCH64_SME_MATRIX_TILE,
1987
5.70k
            vas);
1988
5.70k
    AArch64_get_detail_op(MI, 0)->sme.is_vertical = isVertical;
1989
5.70k
    break;
1990
2.22k
  }
1991
1.29k
  case AArch64_OP_GROUP_PostIncOperand_1:
1992
1.41k
  case AArch64_OP_GROUP_PostIncOperand_12:
1993
3.04k
  case AArch64_OP_GROUP_PostIncOperand_16:
1994
3.28k
  case AArch64_OP_GROUP_PostIncOperand_2:
1995
6.06k
  case AArch64_OP_GROUP_PostIncOperand_24:
1996
6.62k
  case AArch64_OP_GROUP_PostIncOperand_3:
1997
7.28k
  case AArch64_OP_GROUP_PostIncOperand_32:
1998
8.13k
  case AArch64_OP_GROUP_PostIncOperand_4:
1999
8.75k
  case AArch64_OP_GROUP_PostIncOperand_48:
2000
9.55k
  case AArch64_OP_GROUP_PostIncOperand_6:
2001
9.59k
  case AArch64_OP_GROUP_PostIncOperand_64:
2002
11.1k
  case AArch64_OP_GROUP_PostIncOperand_8: {
2003
11.1k
    uint64_t Imm = temp_arg_0;
2004
11.1k
    unsigned Reg = MCInst_getOpVal(MI, OpNum);
2005
11.1k
    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
11.1k
      AArch64_set_detail_op_reg(MI, OpNum, Reg);
2011
11.1k
    break;
2012
9.59k
  }
2013
5.75k
  case AArch64_OP_GROUP_PredicateAsCounter_0:
2014
5.89k
  case AArch64_OP_GROUP_PredicateAsCounter_16:
2015
5.96k
  case AArch64_OP_GROUP_PredicateAsCounter_32:
2016
6.24k
  case AArch64_OP_GROUP_PredicateAsCounter_64:
2017
6.43k
  case AArch64_OP_GROUP_PredicateAsCounter_8: {
2018
6.43k
    unsigned EltSize = temp_arg_0;
2019
6.43k
    AArch64_get_detail_op(MI, 0)->vas = EltSize;
2020
6.43k
    AArch64_set_detail_op_reg(
2021
6.43k
      MI, OpNum, MCInst_getOpVal(MI, OpNum));
2022
6.43k
    break;
2023
6.24k
  }
2024
1.79k
  case AArch64_OP_GROUP_PrefetchOp_0:
2025
5.10k
  case AArch64_OP_GROUP_PrefetchOp_1: {
2026
5.10k
    bool IsSVEPrefetch = (bool)temp_arg_0;
2027
5.10k
    unsigned prfop = MCInst_getOpVal(MI, (OpNum));
2028
5.10k
    aarch64_sysop sysop = { 0 };
2029
5.10k
    if (IsSVEPrefetch) {
2030
3.31k
      const AArch64SVEPRFM_SVEPRFM *PRFM =
2031
3.31k
        AArch64SVEPRFM_lookupSVEPRFMByEncoding(prfop);
2032
3.31k
      if (PRFM) {
2033
2.69k
        sysop.alias = PRFM->SysAlias;
2034
2.69k
        sysop.sub_type = AARCH64_OP_SVEPRFM;
2035
2.69k
        AArch64_set_detail_op_sys(MI, OpNum, sysop,
2036
2.69k
                AARCH64_OP_SYSALIAS);
2037
2.69k
        break;
2038
2.69k
      }
2039
3.31k
    } else {
2040
1.79k
      const AArch64PRFM_PRFM *PRFM =
2041
1.79k
        AArch64PRFM_lookupPRFMByEncoding(prfop);
2042
1.79k
      if (PRFM &&
2043
1.79k
          AArch64_testFeatureList(MI->csh->mode,
2044
1.26k
                PRFM->FeaturesRequired)) {
2045
1.26k
        sysop.alias = PRFM->SysAlias;
2046
1.26k
        sysop.sub_type = AARCH64_OP_PRFM;
2047
1.26k
        AArch64_set_detail_op_sys(MI, OpNum, sysop,
2048
1.26k
                AARCH64_OP_SYSALIAS);
2049
1.26k
        break;
2050
1.26k
      }
2051
1.79k
    }
2052
1.14k
    AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_IMM;
2053
1.14k
    AArch64_get_detail_op(MI, 0)->imm = prfop;
2054
1.14k
    AArch64_get_detail_op(MI, 0)->access =
2055
1.14k
      map_get_op_access(MI, OpNum);
2056
1.14k
    AArch64_inc_op_count(MI);
2057
1.14k
    break;
2058
5.10k
  }
2059
424
  case AArch64_OP_GROUP_SImm_16:
2060
1.03k
  case AArch64_OP_GROUP_SImm_8: {
2061
1.03k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
2062
1.03k
            MCInst_getOpVal(MI, OpNum));
2063
1.03k
    break;
2064
424
  }
2065
681
  case AArch64_OP_GROUP_SVELogicalImm_int16_t:
2066
2.32k
  case AArch64_OP_GROUP_SVELogicalImm_int32_t:
2067
3.51k
  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.51k
    uint64_t Val = MCInst_getOpVal(MI, OpNum);
2071
3.51k
    uint64_t DecodedVal =
2072
3.51k
      AArch64_AM_decodeLogicalImmediate(Val, 64);
2073
3.51k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
2074
3.51k
            DecodedVal);
2075
3.51k
    break;
2076
2.32k
  }
2077
39.8k
  case AArch64_OP_GROUP_SVERegOp_0:
2078
64.9k
  case AArch64_OP_GROUP_SVERegOp_b:
2079
91.1k
  case AArch64_OP_GROUP_SVERegOp_d:
2080
123k
  case AArch64_OP_GROUP_SVERegOp_h:
2081
124k
  case AArch64_OP_GROUP_SVERegOp_q:
2082
153k
  case AArch64_OP_GROUP_SVERegOp_s: {
2083
153k
    char Suffix = (char)temp_arg_0;
2084
153k
    AArch64_get_detail_op(MI, 0)->vas = get_vl_by_suffix(Suffix);
2085
153k
    AArch64_set_detail_op_reg(MI, OpNum,
2086
153k
            MCInst_getOpVal(MI, OpNum));
2087
153k
    break;
2088
124k
  }
2089
1.92k
  case AArch64_OP_GROUP_UImm12Offset_1:
2090
2.24k
  case AArch64_OP_GROUP_UImm12Offset_16:
2091
3.73k
  case AArch64_OP_GROUP_UImm12Offset_2:
2092
4.41k
  case AArch64_OP_GROUP_UImm12Offset_4:
2093
5.28k
  case AArch64_OP_GROUP_UImm12Offset_8: {
2094
    // Otherwise it is an expression. For which we only add the immediate
2095
5.28k
    unsigned Scale = MCOperand_isImm(MCInst_getOperand(MI, OpNum)) ? temp_arg_0 : 1;
2096
5.28k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM,
2097
5.28k
            Scale * MCInst_getOpVal(MI, OpNum));
2098
5.28k
    break;
2099
4.41k
  }
2100
37.1k
  case AArch64_OP_GROUP_VectorIndex_1:
2101
37.1k
  case AArch64_OP_GROUP_VectorIndex_8: {
2102
37.1k
    CS_ASSERT_RET(AArch64_get_detail(MI)->op_count > 0);
2103
37.1k
    unsigned Scale = temp_arg_0;
2104
37.1k
    unsigned VIndex = Scale * MCInst_getOpVal(MI, OpNum);
2105
    // The index can either be for one operand, or for each operand of a list.
2106
37.1k
    if (!AArch64_get_detail_op(MI, -1)->is_list_member) {
2107
17.2k
      AArch64_get_detail_op(MI, -1)->vector_index = VIndex;
2108
17.2k
      break;
2109
17.2k
    }
2110
68.9k
    for (int i = AArch64_get_detail(MI)->op_count - 1; i >= 0;
2111
49.0k
         --i) {
2112
49.0k
      if (!AArch64_get_detail(MI)->operands[i].is_list_member)
2113
0
        break;
2114
49.0k
      AArch64_get_detail(MI)->operands[i].vector_index =
2115
49.0k
        VIndex;
2116
49.0k
    }
2117
19.9k
    break;
2118
37.1k
  }
2119
19
  case AArch64_OP_GROUP_ZPRasFPR_128:
2120
473
  case AArch64_OP_GROUP_ZPRasFPR_16:
2121
695
  case AArch64_OP_GROUP_ZPRasFPR_32:
2122
1.39k
  case AArch64_OP_GROUP_ZPRasFPR_64:
2123
1.42k
  case AArch64_OP_GROUP_ZPRasFPR_8: {
2124
1.42k
    unsigned Base = AArch64_NoRegister;
2125
1.42k
    unsigned Width = temp_arg_0;
2126
1.42k
    switch (Width) {
2127
36
    case 8:
2128
36
      Base = AArch64_B0;
2129
36
      break;
2130
454
    case 16:
2131
454
      Base = AArch64_H0;
2132
454
      break;
2133
222
    case 32:
2134
222
      Base = AArch64_S0;
2135
222
      break;
2136
698
    case 64:
2137
698
      Base = AArch64_D0;
2138
698
      break;
2139
19
    case 128:
2140
19
      Base = AArch64_Q0;
2141
19
      break;
2142
0
    default:
2143
0
      CS_ASSERT_RET(0 && "Unsupported width");
2144
1.42k
    }
2145
1.42k
    unsigned Reg = MCInst_getOpVal(MI, (OpNum));
2146
1.42k
    AArch64_set_detail_op_reg(MI, OpNum, Reg - AArch64_Z0 + Base);
2147
1.42k
    break;
2148
1.42k
  }
2149
287k
  }
2150
287k
}
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
74.3k
{
2159
74.3k
  if (!add_cs_detail_begin(MI, OpNum))
2160
0
    return;
2161
74.3k
  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
651
  case AArch64_OP_GROUP_ComplexRotationOp_180_90:
2166
2.21k
  case AArch64_OP_GROUP_ComplexRotationOp_90_0: {
2167
2.21k
    unsigned Angle = temp_arg_0;
2168
2.21k
    unsigned Remainder = temp_arg_1;
2169
2.21k
    unsigned Imm = (MCInst_getOpVal(MI, OpNum) * Angle) + Remainder;
2170
2.21k
    AArch64_set_detail_op_imm(MI, OpNum, AARCH64_OP_IMM, Imm);
2171
2.21k
    break;
2172
651
  }
2173
96
  case AArch64_OP_GROUP_ExactFPImm_AArch64ExactFPImm_half_AArch64ExactFPImm_one:
2174
1.53k
  case AArch64_OP_GROUP_ExactFPImm_AArch64ExactFPImm_half_AArch64ExactFPImm_two:
2175
2.69k
  case AArch64_OP_GROUP_ExactFPImm_AArch64ExactFPImm_zero_AArch64ExactFPImm_one: {
2176
2.69k
    aarch64_exactfpimm ImmIs0 = temp_arg_0;
2177
2.69k
    aarch64_exactfpimm ImmIs1 = temp_arg_1;
2178
2.69k
    const AArch64ExactFPImm_ExactFPImm *Imm0Desc =
2179
2.69k
      AArch64ExactFPImm_lookupExactFPImmByEnum(ImmIs0);
2180
2.69k
    const AArch64ExactFPImm_ExactFPImm *Imm1Desc =
2181
2.69k
      AArch64ExactFPImm_lookupExactFPImmByEnum(ImmIs1);
2182
2.69k
    unsigned Val = MCInst_getOpVal(MI, (OpNum));
2183
2.69k
    aarch64_sysop sysop = { 0 };
2184
2.69k
    sysop.imm = Val ? Imm1Desc->SysImm : Imm0Desc->SysImm;
2185
2.69k
    sysop.sub_type = AARCH64_OP_EXACTFPIMM;
2186
2.69k
    AArch64_set_detail_op_sys(MI, OpNum, sysop, AARCH64_OP_SYSIMM);
2187
2.69k
    break;
2188
1.53k
  }
2189
3.06k
  case AArch64_OP_GROUP_ImmRangeScale_2_1:
2190
6.45k
  case AArch64_OP_GROUP_ImmRangeScale_4_3: {
2191
6.45k
    uint64_t Scale = temp_arg_0;
2192
6.45k
    uint64_t Offset = temp_arg_1;
2193
6.45k
    unsigned FirstImm = Scale * MCInst_getOpVal(MI, (OpNum));
2194
6.45k
    AArch64_set_detail_op_imm_range(MI, OpNum, FirstImm, FirstImm + Offset);
2195
6.45k
    break;
2196
3.06k
  }
2197
12
  case AArch64_OP_GROUP_MemExtend_w_128:
2198
238
  case AArch64_OP_GROUP_MemExtend_w_16:
2199
433
  case AArch64_OP_GROUP_MemExtend_w_32:
2200
586
  case AArch64_OP_GROUP_MemExtend_w_64:
2201
1.53k
  case AArch64_OP_GROUP_MemExtend_w_8:
2202
1.56k
  case AArch64_OP_GROUP_MemExtend_x_128:
2203
2.07k
  case AArch64_OP_GROUP_MemExtend_x_16:
2204
2.29k
  case AArch64_OP_GROUP_MemExtend_x_32:
2205
2.98k
  case AArch64_OP_GROUP_MemExtend_x_64:
2206
3.74k
  case AArch64_OP_GROUP_MemExtend_x_8: {
2207
3.74k
    char SrcRegKind = (char)temp_arg_0;
2208
3.74k
    unsigned ExtWidth = temp_arg_1;
2209
3.74k
    bool SignExtend = MCInst_getOpVal(MI, OpNum);
2210
3.74k
    bool DoShift = MCInst_getOpVal(MI, OpNum + 1);
2211
3.74k
    AArch64_set_detail_shift_ext(MI, OpNum, SignExtend, DoShift,
2212
3.74k
               ExtWidth, SrcRegKind);
2213
3.74k
    break;
2214
2.98k
  }
2215
12.7k
  case AArch64_OP_GROUP_TypedVectorList_0_b:
2216
28.8k
  case AArch64_OP_GROUP_TypedVectorList_0_d:
2217
41.0k
  case AArch64_OP_GROUP_TypedVectorList_0_h:
2218
42.1k
  case AArch64_OP_GROUP_TypedVectorList_0_q:
2219
51.1k
  case AArch64_OP_GROUP_TypedVectorList_0_s:
2220
51.2k
  case AArch64_OP_GROUP_TypedVectorList_0_0:
2221
53.1k
  case AArch64_OP_GROUP_TypedVectorList_16_b:
2222
54.4k
  case AArch64_OP_GROUP_TypedVectorList_1_d:
2223
55.7k
  case AArch64_OP_GROUP_TypedVectorList_2_d:
2224
56.4k
  case AArch64_OP_GROUP_TypedVectorList_2_s:
2225
57.5k
  case AArch64_OP_GROUP_TypedVectorList_4_h:
2226
58.0k
  case AArch64_OP_GROUP_TypedVectorList_4_s:
2227
58.7k
  case AArch64_OP_GROUP_TypedVectorList_8_b:
2228
59.1k
  case AArch64_OP_GROUP_TypedVectorList_8_h: {
2229
59.1k
    uint8_t NumLanes = (uint8_t)temp_arg_0;
2230
59.1k
    char LaneKind = (char)temp_arg_1;
2231
59.1k
    uint16_t Pair = ((NumLanes << 8) | LaneKind);
2232
2233
59.1k
    AArch64Layout_VectorLayout vas = AARCH64LAYOUT_INVALID;
2234
59.1k
    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
619
    case ((8 << 8) | 'b'):
2240
619
      vas = AARCH64LAYOUT_VL_8B;
2241
619
      break;
2242
1.12k
    case ((4 << 8) | 'h'):
2243
1.12k
      vas = AARCH64LAYOUT_VL_4H;
2244
1.12k
      break;
2245
699
    case ((2 << 8) | 's'):
2246
699
      vas = AARCH64LAYOUT_VL_2S;
2247
699
      break;
2248
1.27k
    case ((1 << 8) | 'd'):
2249
1.27k
      vas = AARCH64LAYOUT_VL_1D;
2250
1.27k
      break;
2251
1.94k
    case ((16 << 8) | 'b'):
2252
1.94k
      vas = AARCH64LAYOUT_VL_16B;
2253
1.94k
      break;
2254
491
    case ((8 << 8) | 'h'):
2255
491
      vas = AARCH64LAYOUT_VL_8H;
2256
491
      break;
2257
558
    case ((4 << 8) | 's'):
2258
558
      vas = AARCH64LAYOUT_VL_4S;
2259
558
      break;
2260
1.24k
    case ((2 << 8) | 'd'):
2261
1.24k
      vas = AARCH64LAYOUT_VL_2D;
2262
1.24k
      break;
2263
12.7k
    case 'b':
2264
12.7k
      vas = AARCH64LAYOUT_VL_B;
2265
12.7k
      break;
2266
12.2k
    case 'h':
2267
12.2k
      vas = AARCH64LAYOUT_VL_H;
2268
12.2k
      break;
2269
8.99k
    case 's':
2270
8.99k
      vas = AARCH64LAYOUT_VL_S;
2271
8.99k
      break;
2272
16.0k
    case 'd':
2273
16.0k
      vas = AARCH64LAYOUT_VL_D;
2274
16.0k
      break;
2275
1.13k
    case 'q':
2276
1.13k
      vas = AARCH64LAYOUT_VL_Q;
2277
1.13k
      break;
2278
70
    case '0':
2279
      // Implicitly Typed register
2280
70
      break;
2281
59.1k
    }
2282
2283
59.1k
    unsigned Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
2284
59.1k
    unsigned NumRegs = get_vec_list_num_regs(MI, Reg);
2285
59.1k
    unsigned Stride = get_vec_list_stride(MI, Reg);
2286
59.1k
    Reg = get_vec_list_first_reg(MI, Reg);
2287
2288
59.1k
    if ((MCRegisterClass_contains(
2289
59.1k
           MCRegisterInfo_getRegClass(MI->MRI,
2290
59.1k
              AArch64_ZPRRegClassID),
2291
59.1k
           Reg) ||
2292
59.1k
         MCRegisterClass_contains(
2293
28.4k
           MCRegisterInfo_getRegClass(MI->MRI,
2294
28.4k
              AArch64_PPRRegClassID),
2295
28.4k
           Reg)) &&
2296
59.1k
        NumRegs > 1 && Stride == 1 &&
2297
59.1k
        Reg < getNextVectorRegister(Reg, NumRegs - 1)) {
2298
19.3k
      AArch64_get_detail_op(MI, 0)->is_list_member = true;
2299
19.3k
      AArch64_get_detail_op(MI, 0)->vas = vas;
2300
19.3k
      AArch64_set_detail_op_reg(MI, OpNum, Reg);
2301
19.3k
      if (NumRegs > 1) {
2302
        // Add all registers of the list to the details.
2303
54.5k
        for (size_t i = 0; i < NumRegs - 1; ++i) {
2304
35.1k
          AArch64_get_detail_op(MI, 0)->is_list_member =
2305
35.1k
            true;
2306
35.1k
          AArch64_get_detail_op(MI, 0)->vas = vas;
2307
35.1k
          AArch64_set_detail_op_reg(
2308
35.1k
            MI, OpNum,
2309
35.1k
            getNextVectorRegister(Reg + i, 1));
2310
35.1k
        }
2311
19.3k
      }
2312
39.8k
    } else {
2313
130k
      for (unsigned i = 0; i < NumRegs;
2314
90.9k
           ++i, Reg = getNextVectorRegister(Reg, Stride)) {
2315
90.9k
        if (!(MCRegisterClass_contains(
2316
90.9k
            MCRegisterInfo_getRegClass(
2317
90.9k
              MI->MRI, AArch64_ZPRRegClassID),
2318
90.9k
            Reg) ||
2319
90.9k
          MCRegisterClass_contains(
2320
70.9k
            MCRegisterInfo_getRegClass(
2321
70.9k
              MI->MRI, AArch64_PPRRegClassID),
2322
70.9k
            Reg))) {
2323
70.8k
          AArch64_get_detail_op(MI, 0)->is_vreg = true;
2324
70.8k
        }
2325
90.9k
        AArch64_get_detail_op(MI, 0)->is_list_member =
2326
90.9k
          true;
2327
90.9k
        AArch64_get_detail_op(MI, 0)->vas = vas;
2328
90.9k
        AArch64_set_detail_op_reg(MI, OpNum, Reg);
2329
90.9k
      }
2330
39.8k
    }
2331
59.1k
  }
2332
74.3k
  }
2333
74.3k
}
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
13.0k
{
2343
13.0k
  if (!add_cs_detail_begin(MI, OpNum))
2344
0
    return;
2345
13.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
643
  case AArch64_OP_GROUP_RegWithShiftExtend_0_128_x_0:
2350
793
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_w_d:
2351
967
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_w_s:
2352
1.89k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_x_0:
2353
2.04k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_x_d:
2354
2.06k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_16_x_s:
2355
2.47k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_w_d:
2356
2.76k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_w_s:
2357
3.26k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_x_0:
2358
3.63k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_x_d:
2359
3.71k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_32_x_s:
2360
4.20k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_w_d:
2361
4.22k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_w_s:
2362
5.02k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_x_0:
2363
5.16k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_x_d:
2364
5.17k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_64_x_s:
2365
6.88k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_w_d:
2366
7.17k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_w_s:
2367
9.38k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_x_0:
2368
9.93k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_x_d:
2369
9.96k
  case AArch64_OP_GROUP_RegWithShiftExtend_0_8_x_s:
2370
10.1k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_16_w_d:
2371
10.2k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_16_w_s:
2372
10.6k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_32_w_d:
2373
10.8k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_32_w_s:
2374
11.0k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_64_w_d:
2375
11.3k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_64_w_s:
2376
12.6k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_8_w_d:
2377
13.0k
  case AArch64_OP_GROUP_RegWithShiftExtend_1_8_w_s: {
2378
    // signed (s) and unsigned (u) extend
2379
13.0k
    bool SignExtend = (bool)temp_arg_0;
2380
    // Extend width
2381
13.0k
    int ExtWidth = (int)temp_arg_1;
2382
    // w = word, x = doubleword
2383
13.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
13.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
13.0k
    AArch64_get_detail_op(MI, -1)->vas = get_vl_by_suffix(Suffix);
2393
2394
13.0k
    bool DoShift = ExtWidth != 8;
2395
13.0k
    if (!(SignExtend || DoShift || SrcRegKind == 'w'))
2396
2.78k
      return;
2397
2398
10.2k
    AArch64_set_detail_shift_ext(MI, OpNum, SignExtend, DoShift,
2399
10.2k
               ExtWidth, SrcRegKind);
2400
10.2k
    break;
2401
13.0k
  }
2402
13.0k
  }
2403
13.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
708k
{
2409
708k
  if (!detail_is_set(MI))
2410
0
    return;
2411
708k
  AArch64_check_safe_inc(MI);
2412
2413
708k
  if (Reg == AARCH64_REG_ZA ||
2414
708k
      (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
708k
  } else if (((Reg >= AARCH64_REG_P0) && (Reg <= AARCH64_REG_P15)) ||
2420
708k
             ((Reg >= AARCH64_REG_PN0) && (Reg <= AARCH64_REG_PN15))) {
2421
    // SME/SVE predicate register.
2422
53.5k
    AArch64_set_detail_op_pred(MI, OpNum);
2423
53.5k
    return;
2424
654k
  } else if (AArch64_get_detail(MI)->is_doing_sme) {
2425
16.8k
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) & CS_OP_BOUND);
2426
16.8k
    if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_SME) {
2427
16.5k
      AArch64_set_detail_op_sme(MI, OpNum,
2428
16.5k
              AARCH64_SME_MATRIX_SLICE_REG,
2429
16.5k
              AARCH64LAYOUT_INVALID);
2430
16.5k
    } else if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_PRED) {
2431
252
      AArch64_set_detail_op_pred(MI, OpNum);
2432
252
    } else {
2433
0
      CS_ASSERT_RET(0 && "Unkown SME/SVE operand type");
2434
0
    }
2435
16.8k
    return;
2436
16.8k
  }
2437
638k
  if (map_get_op_type(MI, OpNum) & CS_OP_MEM) {
2438
115k
    AArch64_set_detail_op_mem(MI, OpNum, Reg);
2439
115k
    return;
2440
115k
  }
2441
2442
522k
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_BOUND));
2443
522k
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_MEM));
2444
522k
  CS_ASSERT_RET(map_get_op_type(MI, OpNum) == CS_OP_REG);
2445
2446
522k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_REG;
2447
522k
  AArch64_get_detail_op(MI, 0)->reg = Reg;
2448
522k
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2449
522k
  AArch64_inc_op_count(MI);
2450
522k
}
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
111k
static bool prev_is_membase_wb(MCInst *MI) {
2458
111k
  return AArch64_get_detail(MI)->op_count > 0 &&
2459
111k
         AArch64_get_detail_op(MI, -1)->type == AARCH64_OP_MEM &&
2460
111k
         AArch64_get_detail_op(MI, -1)->mem.disp == 0 &&
2461
111k
         get_detail(MI)->writeback;
2462
111k
}
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
150k
{
2469
150k
  if (!detail_is_set(MI))
2470
0
    return;
2471
150k
  AArch64_check_safe_inc(MI);
2472
2473
150k
  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
150k
  if (map_get_op_type(MI, OpNum) & CS_OP_MEM || prev_is_membase_wb(MI)) {
2487
46.4k
    AArch64_set_detail_op_mem(MI, OpNum, Imm);
2488
46.4k
    return;
2489
46.4k
  }
2490
2491
103k
  CS_ASSERT_RET(!(map_get_op_type(MI, OpNum) & CS_OP_MEM));
2492
103k
  CS_ASSERT_RET((map_get_op_type(MI, OpNum) & ~CS_OP_BOUND) == CS_OP_IMM);
2493
103k
  CS_ASSERT_RET(ImmType == AARCH64_OP_IMM || ImmType == AARCH64_OP_CIMM);
2494
2495
103k
  AArch64_get_detail_op(MI, 0)->type = ImmType;
2496
103k
  AArch64_get_detail_op(MI, 0)->imm = Imm;
2497
103k
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2498
103k
  AArch64_inc_op_count(MI);
2499
103k
}
2500
2501
void AArch64_set_detail_op_imm_range(MCInst *MI, unsigned OpNum,
2502
             uint32_t FirstImm, uint32_t Offset)
2503
6.45k
{
2504
6.45k
  if (!detail_is_set(MI))
2505
0
    return;
2506
6.45k
  AArch64_check_safe_inc(MI);
2507
2508
6.45k
  if (AArch64_get_detail(MI)->is_doing_sme) {
2509
6.45k
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) & CS_OP_BOUND);
2510
6.45k
    if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_SME) {
2511
6.45k
      AArch64_set_detail_op_sme(MI, OpNum,
2512
6.45k
              AARCH64_SME_MATRIX_SLICE_OFF_RANGE,
2513
6.45k
              AARCH64LAYOUT_INVALID, (uint32_t) FirstImm,
2514
6.45k
              (uint32_t) Offset);
2515
6.45k
    } 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
6.45k
    return;
2521
6.45k
  }
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
162k
{
2537
162k
  if (!detail_is_set(MI))
2538
0
    return;
2539
162k
  AArch64_check_safe_inc(MI);
2540
2541
162k
  AArch64_set_mem_access(MI, true);
2542
2543
162k
  cs_op_type secondary_type = map_get_op_type(MI, OpNum) & ~CS_OP_MEM;
2544
162k
  switch (secondary_type) {
2545
0
  default:
2546
0
    CS_ASSERT_RET(0 && "Secondary type not supported yet.");
2547
115k
  case CS_OP_REG: {
2548
115k
    bool is_index_reg = AArch64_get_detail_op(MI, 0)->mem.base !=
2549
115k
            AARCH64_REG_INVALID;
2550
115k
    if (is_index_reg)
2551
19.7k
      AArch64_get_detail_op(MI, 0)->mem.index = Val;
2552
95.8k
    else {
2553
95.8k
      AArch64_get_detail_op(MI, 0)->mem.base = Val;
2554
95.8k
    }
2555
2556
115k
    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
28.3k
      map_add_implicit_write(MI, MCInst_getOpVal(MI, OpNum));
2565
28.3k
    }
2566
115k
    break;
2567
0
  }
2568
46.4k
  case CS_OP_IMM: {
2569
46.4k
    AArch64_get_detail_op(MI, 0)->mem.disp = Val;
2570
46.4k
    break;
2571
0
  }
2572
162k
  }
2573
2574
162k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_MEM;
2575
162k
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2576
162k
  AArch64_set_mem_access(MI, false);
2577
162k
}
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
14.0k
{
2585
14.0k
  bool IsLSL = !SignExtend && SrcRegKind == 'x';
2586
14.0k
  if (IsLSL)
2587
4.86k
    AArch64_get_detail_op(MI, -1)->shift.type = AARCH64_SFT_LSL;
2588
9.14k
  else {
2589
9.14k
    aarch64_extender ext = SignExtend ? AARCH64_EXT_SXTB :
2590
9.14k
                AARCH64_EXT_UXTB;
2591
9.14k
    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
8.14k
    case 'w':
2601
8.14k
      ext += 2;
2602
8.14k
      break;
2603
996
    case 'x':
2604
996
      ext += 3;
2605
996
      break;
2606
9.14k
    }
2607
9.14k
    AArch64_get_detail_op(MI, -1)->ext = ext;
2608
9.14k
  }
2609
14.0k
  if (DoShift || IsLSL) {
2610
9.00k
    unsigned ShiftAmount = DoShift ? Log2_32(ExtWidth / 8) : 0;
2611
9.00k
    AArch64_get_detail_op(MI, -1)->shift.type = AARCH64_SFT_LSL;
2612
9.00k
    AArch64_get_detail_op(MI, -1)->shift.value = ShiftAmount;
2613
9.00k
  }
2614
14.0k
}
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
696
{
2620
696
  if (!detail_is_set(MI))
2621
0
    return;
2622
696
  AArch64_check_safe_inc(MI);
2623
2624
696
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_FP;
2625
696
  AArch64_get_detail_op(MI, 0)->fp = Val;
2626
696
  AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2627
696
  AArch64_inc_op_count(MI);
2628
696
}
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
19.8k
{
2635
19.8k
  if (!detail_is_set(MI))
2636
0
    return;
2637
19.8k
  AArch64_check_safe_inc(MI);
2638
2639
19.8k
  AArch64_get_detail_op(MI, 0)->type = type;
2640
19.8k
  AArch64_get_detail_op(MI, 0)->sysop = sys_op;
2641
19.8k
  if (sys_op.sub_type == AARCH64_OP_EXACTFPIMM) {
2642
2.69k
    AArch64_get_detail_op(MI, 0)->fp = aarch64_exact_fp_to_fp(sys_op.imm.exactfpimm);
2643
2.69k
  }
2644
19.8k
  AArch64_inc_op_count(MI);
2645
19.8k
}
2646
2647
54.0k
void AArch64_set_detail_op_pred(MCInst *MI, unsigned OpNum) {
2648
54.0k
  if (!detail_is_set(MI))
2649
0
    return;
2650
54.0k
  AArch64_check_safe_inc(MI);
2651
2652
54.0k
  if (AArch64_get_detail_op(MI, 0)->type == AARCH64_OP_INVALID) {
2653
53.0k
    setup_pred_operand(MI);
2654
53.0k
  }
2655
54.0k
  aarch64_op_pred *p = &AArch64_get_detail_op(MI, 0)->pred;
2656
54.0k
  if (p->reg == AARCH64_REG_INVALID) {
2657
53.0k
    p->reg = MCInst_getOpVal(MI, OpNum);
2658
53.0k
    AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2659
53.0k
    AArch64_get_detail(MI)->is_doing_sme = true;
2660
53.0k
    return;
2661
53.0k
  } else if (p->vec_select == AARCH64_REG_INVALID) {
2662
802
    p->vec_select = MCInst_getOpVal(MI, OpNum);
2663
802
    return;
2664
802
  } else if (p->imm_index == -1) {
2665
252
    p->imm_index = MCInst_getOpVal(MI, OpNum);
2666
252
    return;
2667
252
  }
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
54.7k
{
2676
54.7k
  if (!detail_is_set(MI))
2677
0
    return;
2678
54.7k
  AArch64_check_safe_inc(MI);
2679
2680
54.7k
  AArch64_get_detail_op(MI, 0)->type = AARCH64_OP_SME;
2681
54.7k
  switch (part) {
2682
0
  default:
2683
0
    printf("Unhandled SME operand part %d\n", part);
2684
0
    CS_ASSERT_RET(0);
2685
3.21k
  case AARCH64_SME_MATRIX_TILE_LIST: {
2686
3.21k
    setup_sme_operand(MI);
2687
3.21k
    va_list args;
2688
3.21k
    va_start(args, vas);
2689
3.21k
    int Tile = va_arg(args, int); // NOLINT(clang-analyzer-valist.Uninitialized)
2690
3.21k
    va_end(args);
2691
3.21k
    AArch64_get_detail_op(MI, 0)->sme.type = AARCH64_SME_OP_TILE;
2692
3.21k
    AArch64_get_detail_op(MI, 0)->sme.tile = Tile;
2693
3.21k
    AArch64_get_detail_op(MI, 0)->vas = vas;
2694
3.21k
    AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2695
3.21k
    AArch64_get_detail(MI)->is_doing_sme = true;
2696
3.21k
    break;
2697
0
  }
2698
18.4k
  case AARCH64_SME_MATRIX_TILE:
2699
18.4k
    CS_ASSERT_RET(map_get_op_type(MI, OpNum) == CS_OP_REG);
2700
2701
18.4k
    setup_sme_operand(MI);
2702
18.4k
    AArch64_get_detail_op(MI, 0)->sme.type = AARCH64_SME_OP_TILE;
2703
18.4k
    AArch64_get_detail_op(MI, 0)->sme.tile =
2704
18.4k
      MCInst_getOpVal(MI, OpNum);
2705
18.4k
    AArch64_get_detail_op(MI, 0)->vas = vas;
2706
18.4k
    AArch64_get_detail_op(MI, 0)->access = map_get_op_access(MI, OpNum);
2707
18.4k
    AArch64_get_detail(MI)->is_doing_sme = true;
2708
18.4k
    break;
2709
16.5k
  case AARCH64_SME_MATRIX_SLICE_REG:
2710
16.5k
    CS_ASSERT_RET((map_get_op_type(MI, OpNum) & ~(CS_OP_MEM | CS_OP_BOUND)) == CS_OP_REG);
2711
16.5k
    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
16.5k
    AArch64_get_detail_op(MI, 0)->sme.type =
2715
16.5k
      AARCH64_SME_OP_TILE_VEC;
2716
16.5k
    AArch64_get_detail_op(MI, 0)->sme.slice_reg =
2717
16.5k
      MCInst_getOpVal(MI, OpNum);
2718
16.5k
    break;
2719
10.0k
  case AARCH64_SME_MATRIX_SLICE_OFF: {
2720
10.0k
    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
10.0k
    CS_ASSERT_RET(AArch64_get_detail_op(MI, 0)->type ==
2723
10.0k
           AARCH64_OP_SME);
2724
10.0k
    CS_ASSERT_RET(AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm ==
2725
10.0k
           AARCH64_SLICE_IMM_INVALID);
2726
10.0k
    va_list args;
2727
10.0k
    va_start(args, vas);
2728
10.0k
    uint16_t offset = va_arg(args, uint32_t); // NOLINT(clang-analyzer-valist.Uninitialized)
2729
10.0k
    va_end(args);
2730
10.0k
    AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm =
2731
10.0k
      offset;
2732
10.0k
    break;
2733
0
  }
2734
6.45k
  case AARCH64_SME_MATRIX_SLICE_OFF_RANGE: {
2735
6.45k
    va_list args;
2736
6.45k
    va_start(args, vas);
2737
6.45k
    uint8_t First = va_arg(args, uint32_t); // NOLINT(clang-analyzer-valist.Uninitialized)
2738
6.45k
    uint8_t Offset = va_arg(args, uint32_t); // NOLINT(clang-analyzer-valist.Uninitialized)
2739
6.45k
    va_end(args);
2740
6.45k
    AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.first =
2741
6.45k
      First;
2742
6.45k
    AArch64_get_detail_op(MI, 0)->sme.slice_offset.imm_range.offset =
2743
6.45k
      Offset;
2744
6.45k
    AArch64_get_detail_op(MI, 0)->sme.has_range_offset = true;
2745
6.45k
    break;
2746
0
  }
2747
54.7k
  }
2748
54.7k
}
2749
2750
static void insert_op(MCInst *MI, unsigned index, cs_aarch64_op op)
2751
14.3k
{
2752
14.3k
  if (!detail_is_set(MI)) {
2753
0
    return;
2754
0
  }
2755
2756
14.3k
  AArch64_check_safe_inc(MI);
2757
14.3k
  cs_aarch64_op *ops = AArch64_get_detail(MI)->operands;
2758
14.3k
  int i = AArch64_get_detail(MI)->op_count;
2759
14.3k
  if (index == -1) {
2760
14.3k
    ops[i] = op;
2761
14.3k
    AArch64_inc_op_count(MI);
2762
14.3k
    return;
2763
14.3k
  }
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.85k
{
2797
2.85k
  if (!detail_is_set(MI))
2798
0
    return;
2799
2800
2.85k
  AArch64_check_safe_inc(MI);
2801
2802
2.85k
  cs_aarch64_op op;
2803
2.85k
  AArch64_setup_op(&op);
2804
2.85k
  op.type = AARCH64_OP_REG;
2805
2.85k
  op.reg = Reg;
2806
2.85k
  op.access = access;
2807
2808
2.85k
  insert_op(MI, index, op);
2809
2.85k
}
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
3.23k
{
2816
3.23k
  if (!detail_is_set(MI))
2817
0
    return;
2818
3.23k
  AArch64_check_safe_inc(MI);
2819
2820
3.23k
  cs_aarch64_op op;
2821
3.23k
  AArch64_setup_op(&op);
2822
3.23k
  op.type = AARCH64_OP_IMM;
2823
3.23k
  op.imm = Imm;
2824
3.23k
  op.access = CS_AC_READ;
2825
2826
3.23k
  insert_op(MI, index, op);
2827
3.23k
}
2828
2829
void AArch64_insert_detail_op_sys(MCInst *MI, unsigned index, aarch64_sysop sys_op,
2830
             aarch64_op_type type)
2831
4.95k
{
2832
4.95k
  if (!detail_is_set(MI))
2833
0
    return;
2834
4.95k
  AArch64_check_safe_inc(MI);
2835
2836
4.95k
  cs_aarch64_op op;
2837
4.95k
  AArch64_setup_op(&op);
2838
4.95k
  op.type = type;
2839
4.95k
  op.sysop = sys_op;
2840
4.95k
  if (op.sysop.sub_type == AARCH64_OP_EXACTFPIMM) {
2841
4.82k
    op.fp = aarch64_exact_fp_to_fp(op.sysop.imm.exactfpimm);
2842
4.82k
  }
2843
4.95k
  insert_op(MI, index, op);
2844
4.95k
}
2845
2846
2847
void AArch64_insert_detail_op_sme(MCInst *MI, unsigned index, aarch64_op_sme sme_op)
2848
3.29k
{
2849
3.29k
  if (!detail_is_set(MI))
2850
0
    return;
2851
3.29k
  AArch64_check_safe_inc(MI);
2852
2853
3.29k
  cs_aarch64_op op;
2854
3.29k
  AArch64_setup_op(&op);
2855
3.29k
  op.type = AARCH64_OP_SME;
2856
3.29k
  op.sme = sme_op;
2857
3.29k
  insert_op(MI, index, op);
2858
3.29k
}
2859
2860
#endif