Coverage Report

Created: 2025-07-04 06:11

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