Coverage Report

Created: 2026-06-15 06:41

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonev5/arch/AArch64/AArch64InstPrinter.c
Line
Count
Source
1
//==-- AArch64InstPrinter.cpp - Convert AArch64 MCInst to assembly syntax --==//
2
//
3
//                     The LLVM Compiler Infrastructure
4
//
5
// This file is distributed under the University of Illinois Open Source
6
// License. See LICENSE.TXT for details.
7
//
8
//===----------------------------------------------------------------------===//
9
//
10
// This class prints an AArch64 MCInst to a .s file.
11
//
12
//===----------------------------------------------------------------------===//
13
14
/* Capstone Disassembly Engine */
15
/* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2016 */
16
17
#ifdef CAPSTONE_HAS_ARM64
18
19
#include <capstone/platform.h>
20
#include <stdio.h>
21
#include <stdlib.h>
22
23
#include "AArch64InstPrinter.h"
24
#include "AArch64Disassembler.h"
25
#include "AArch64BaseInfo.h"
26
#include "../../utils.h"
27
#include "../../MCInst.h"
28
#include "../../SStream.h"
29
#include "../../MCRegisterInfo.h"
30
#include "../../MathExtras.h"
31
32
#include "AArch64Mapping.h"
33
#include "AArch64AddressingModes.h"
34
35
#define GET_REGINFO_ENUM
36
#include "AArch64GenRegisterInfo.inc"
37
38
#define GET_INSTRINFO_ENUM
39
#include "AArch64GenInstrInfo.inc"
40
41
#include "AArch64GenSubtargetInfo.inc"
42
43
44
static const char *getRegisterName(unsigned RegNo, unsigned AltIdx);
45
static void printOperand(MCInst *MI, unsigned OpNum, SStream *O);
46
static bool printSysAlias(MCInst *MI, SStream *O);
47
static char *printAliasInstr(MCInst *MI, SStream *OS, MCRegisterInfo *MRI);
48
static void printInstruction(MCInst *MI, SStream *O);
49
static void printShifter(MCInst *MI, unsigned OpNum, SStream *O);
50
static void printCustomAliasOperand(MCInst *MI, uint64_t Address, unsigned OpIdx,
51
    unsigned PrintMethodIdx, SStream *OS);
52
53
54
static cs_ac_type get_op_access(cs_struct *h, unsigned int id, unsigned int index)
55
1.19M
{
56
1.19M
#ifndef CAPSTONE_DIET
57
1.19M
  const uint8_t *arr = AArch64_get_op_access(h, id);
58
59
1.19M
  if (arr[index] == CS_AC_IGNORE)
60
0
    return 0;
61
62
1.19M
  return arr[index];
63
#else
64
  return 0;
65
#endif
66
1.19M
}
67
68
static void op_addImm(MCInst *MI, int v)
69
3.46k
{
70
3.46k
  if (MI->csh->detail) {
71
3.46k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
72
3.46k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = v;
73
3.46k
    MI->flat_insn->detail->arm64.op_count++;
74
3.46k
  }
75
3.46k
}
76
77
static void set_sme_index(MCInst *MI, bool status)
78
11.5k
{
79
  // Doing SME Index operand
80
11.5k
  MI->csh->doing_SME_Index = status;
81
82
11.5k
  if (MI->csh->detail != CS_OPT_ON)
83
0
    return;
84
85
11.5k
  if (status) {
86
8.02k
    unsigned prevOpNum = MI->flat_insn->detail->arm64.op_count - 1; 
87
8.02k
    unsigned Reg = MCOperand_getReg(MCInst_getOperand(MI, prevOpNum));
88
    // Replace previous SME register operand with an OP_SME_INDEX operand
89
8.02k
    MI->flat_insn->detail->arm64.operands[prevOpNum].type = ARM64_OP_SME_INDEX;
90
8.02k
    MI->flat_insn->detail->arm64.operands[prevOpNum].sme_index.reg = Reg;
91
8.02k
    MI->flat_insn->detail->arm64.operands[prevOpNum].sme_index.base = ARM64_REG_INVALID;
92
8.02k
    MI->flat_insn->detail->arm64.operands[prevOpNum].sme_index.disp = 0;
93
8.02k
  }
94
11.5k
}
95
96
static void set_mem_access(MCInst *MI, bool status)
97
407k
{
98
  // If status == false, check if this is meant for SME_index
99
407k
  if(!status && MI->csh->doing_SME_Index) {
100
4.52k
    MI->csh->doing_SME_Index = status;
101
4.52k
    return;
102
4.52k
  }
103
104
  // Doing Memory Operation
105
403k
  MI->csh->doing_mem = status;
106
107
108
403k
  if (MI->csh->detail != CS_OPT_ON)
109
0
    return;
110
111
403k
  if (status) {
112
201k
#ifndef CAPSTONE_DIET
113
201k
    uint8_t access;
114
201k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
115
201k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
116
201k
    MI->ac_idx++;
117
201k
#endif
118
201k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_MEM;
119
201k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.base = ARM64_REG_INVALID;
120
201k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.index = ARM64_REG_INVALID;
121
201k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.disp = 0;
122
201k
  } else {
123
    // done, create the next operand slot
124
201k
    MI->flat_insn->detail->arm64.op_count++;
125
201k
  }
126
403k
}
127
128
void AArch64_printInst(MCInst *MI, SStream *O, void *Info)
129
412k
{
130
  // Check for special encodings and print the canonical alias instead.
131
412k
  unsigned Opcode = MCInst_getOpcode(MI);
132
412k
  int LSB, Width;
133
412k
  char *mnem;
134
135
  // printf(">>> opcode = %u\n", MCInst_getOpcode(MI));
136
137
412k
  if (Opcode == AArch64_SYSxt && printSysAlias(MI, O))
138
2.00k
    return;
139
140
  // SBFM/UBFM should print to a nicer aliased form if possible.
141
410k
  if (Opcode == AArch64_SBFMXri || Opcode == AArch64_SBFMWri ||
142
408k
      Opcode == AArch64_UBFMXri || Opcode == AArch64_UBFMWri) {
143
3.91k
    bool IsSigned = (Opcode == AArch64_SBFMXri || Opcode == AArch64_SBFMWri);
144
3.91k
    bool Is64Bit = (Opcode == AArch64_SBFMXri || Opcode == AArch64_UBFMXri);
145
146
3.91k
    MCOperand *Op0 = MCInst_getOperand(MI, 0);
147
3.91k
    MCOperand *Op1 = MCInst_getOperand(MI, 1);
148
3.91k
    MCOperand *Op2 = MCInst_getOperand(MI, 2);
149
3.91k
    MCOperand *Op3 = MCInst_getOperand(MI, 3);
150
151
3.91k
    if (MCOperand_isImm(Op2) && MCOperand_getImm(Op2) == 0 && MCOperand_isImm(Op3)) {
152
2.75k
      const char *AsmMnemonic = NULL;
153
154
2.75k
      switch (MCOperand_getImm(Op3)) {
155
539
        default:
156
539
          break;
157
158
751
        case 7:
159
751
          if (IsSigned)
160
497
            AsmMnemonic = "sxtb";
161
254
          else if (!Is64Bit)
162
115
            AsmMnemonic = "uxtb";
163
751
          break;
164
165
855
        case 15:
166
855
          if (IsSigned)
167
557
            AsmMnemonic = "sxth";
168
298
          else if (!Is64Bit)
169
233
            AsmMnemonic = "uxth";
170
855
          break;
171
172
608
        case 31:
173
          // *xtw is only valid for signed 64-bit operations.
174
608
          if (Is64Bit && IsSigned)
175
297
            AsmMnemonic = "sxtw";
176
608
          break;
177
2.75k
      }
178
179
2.75k
      if (AsmMnemonic) {
180
1.69k
        SStream_concat(O, "%s\t%s, %s", AsmMnemonic,
181
1.69k
            getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName),
182
1.69k
            getRegisterName(getWRegFromXReg(MCOperand_getReg(Op1)), AArch64_NoRegAltName));
183
184
1.69k
        if (MI->csh->detail) {
185
1.69k
#ifndef CAPSTONE_DIET
186
1.69k
          uint8_t access;
187
1.69k
          access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
188
1.69k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
189
1.69k
          MI->ac_idx++;
190
1.69k
#endif
191
1.69k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
192
1.69k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
193
1.69k
          MI->flat_insn->detail->arm64.op_count++;
194
1.69k
#ifndef CAPSTONE_DIET
195
1.69k
          access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
196
1.69k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
197
1.69k
          MI->ac_idx++;
198
1.69k
#endif
199
1.69k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
200
1.69k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = getWRegFromXReg(MCOperand_getReg(Op1));
201
1.69k
          MI->flat_insn->detail->arm64.op_count++;
202
1.69k
        }
203
204
1.69k
        MCInst_setOpcodePub(MI, AArch64_map_insn(AsmMnemonic));
205
206
1.69k
        return;
207
1.69k
      }
208
2.75k
    }
209
210
    // All immediate shifts are aliases, implemented using the Bitfield
211
    // instruction. In all cases the immediate shift amount shift must be in
212
    // the range 0 to (reg.size -1).
213
2.21k
    if (MCOperand_isImm(Op2) && MCOperand_isImm(Op3)) {
214
2.21k
      const char *AsmMnemonic = NULL;
215
2.21k
      int shift = 0;
216
2.21k
      int immr = (int)MCOperand_getImm(Op2);
217
2.21k
      int imms = (int)MCOperand_getImm(Op3);
218
219
2.21k
      if (Opcode == AArch64_UBFMWri && imms != 0x1F && ((imms + 1) == immr)) {
220
112
        AsmMnemonic = "lsl";
221
112
        shift = 31 - imms;
222
2.10k
      } else if (Opcode == AArch64_UBFMXri && imms != 0x3f &&
223
481
          ((imms + 1 == immr))) {
224
96
        AsmMnemonic = "lsl";
225
96
        shift = 63 - imms;
226
2.00k
      } else if (Opcode == AArch64_UBFMWri && imms == 0x1f) {
227
236
        AsmMnemonic = "lsr";
228
236
        shift = immr;
229
1.77k
      } else if (Opcode == AArch64_UBFMXri && imms == 0x3f) {
230
51
        AsmMnemonic = "lsr";
231
51
        shift = immr;
232
1.71k
      } else if (Opcode == AArch64_SBFMWri && imms == 0x1f) {
233
11
        AsmMnemonic = "asr";
234
11
        shift = immr;
235
1.70k
      } else if (Opcode == AArch64_SBFMXri && imms == 0x3f) {
236
320
        AsmMnemonic = "asr";
237
320
        shift = immr;
238
320
      }
239
240
2.21k
      if (AsmMnemonic) {
241
826
        SStream_concat(O, "%s\t%s, %s, ", AsmMnemonic,
242
826
            getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName),
243
826
            getRegisterName(MCOperand_getReg(Op1), AArch64_NoRegAltName));
244
245
826
        printInt32Bang(O, shift);
246
247
826
        MCInst_setOpcodePub(MI, AArch64_map_insn(AsmMnemonic));
248
249
826
        if (MI->csh->detail) {
250
826
#ifndef CAPSTONE_DIET
251
826
          uint8_t access;
252
826
          access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
253
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
254
826
          MI->ac_idx++;
255
826
#endif
256
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
257
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
258
826
          MI->flat_insn->detail->arm64.op_count++;
259
826
#ifndef CAPSTONE_DIET
260
826
          access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
261
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
262
826
          MI->ac_idx++;
263
826
#endif
264
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
265
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op1);
266
826
          MI->flat_insn->detail->arm64.op_count++;
267
826
#ifndef CAPSTONE_DIET
268
826
          access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
269
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
270
826
          MI->ac_idx++;
271
826
#endif
272
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
273
826
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = shift;
274
826
          MI->flat_insn->detail->arm64.op_count++;
275
826
        }
276
277
826
        return;
278
826
      }
279
2.21k
    }
280
281
    // SBFIZ/UBFIZ aliases
282
1.38k
    if (MCOperand_getImm(Op2) > MCOperand_getImm(Op3)) {
283
791
      SStream_concat(O, "%s\t%s, %s, ", (IsSigned ? "sbfiz" : "ubfiz"),
284
791
          getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName),
285
791
          getRegisterName(MCOperand_getReg(Op1), AArch64_NoRegAltName));
286
287
791
      printInt32Bang(O, (int)((Is64Bit ? 64 : 32) - MCOperand_getImm(Op2)));
288
289
791
      SStream_concat0(O, ", ");
290
291
791
      printInt32Bang(O, (int)MCOperand_getImm(Op3) + 1);
292
293
791
      MCInst_setOpcodePub(MI, AArch64_map_insn(IsSigned ? "sbfiz" : "ubfiz"));
294
295
791
      if (MI->csh->detail) {
296
791
#ifndef CAPSTONE_DIET
297
791
        uint8_t access;
298
791
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
299
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
300
791
        MI->ac_idx++;
301
791
#endif
302
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
303
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
304
791
        MI->flat_insn->detail->arm64.op_count++;
305
791
#ifndef CAPSTONE_DIET
306
791
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
307
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
308
791
        MI->ac_idx++;
309
791
#endif
310
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
311
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op1);
312
791
        MI->flat_insn->detail->arm64.op_count++;
313
791
#ifndef CAPSTONE_DIET
314
791
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
315
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
316
791
        MI->ac_idx++;
317
791
#endif
318
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
319
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = (Is64Bit ? 64 : 32) - (int)MCOperand_getImm(Op2);
320
791
        MI->flat_insn->detail->arm64.op_count++;
321
791
#ifndef CAPSTONE_DIET
322
791
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
323
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
324
791
        MI->ac_idx++;
325
791
#endif
326
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
327
791
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = MCOperand_getImm(Op3) + 1;
328
791
        MI->flat_insn->detail->arm64.op_count++;
329
791
      }
330
331
791
      return;
332
791
    }
333
334
    // Otherwise SBFX/UBFX is the preferred form
335
597
    SStream_concat(O, "%s\t%s, %s, ", (IsSigned ? "sbfx" : "ubfx"),
336
597
        getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName),
337
597
        getRegisterName(MCOperand_getReg(Op1), AArch64_NoRegAltName));
338
339
597
    printInt32Bang(O, (int)MCOperand_getImm(Op2));
340
597
    SStream_concat0(O, ", ");
341
597
    printInt32Bang(O, (int)MCOperand_getImm(Op3) - (int)MCOperand_getImm(Op2) + 1);
342
343
597
    MCInst_setOpcodePub(MI, AArch64_map_insn(IsSigned ? "sbfx" : "ubfx"));
344
345
597
    if (MI->csh->detail) {
346
597
#ifndef CAPSTONE_DIET
347
597
      uint8_t access;
348
597
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
349
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
350
597
      MI->ac_idx++;
351
597
#endif
352
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
353
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
354
597
      MI->flat_insn->detail->arm64.op_count++;
355
597
#ifndef CAPSTONE_DIET
356
597
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
357
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
358
597
      MI->ac_idx++;
359
597
#endif
360
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
361
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op1);
362
597
      MI->flat_insn->detail->arm64.op_count++;
363
597
#ifndef CAPSTONE_DIET
364
597
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
365
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
366
597
      MI->ac_idx++;
367
597
#endif
368
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
369
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = MCOperand_getImm(Op2);
370
597
      MI->flat_insn->detail->arm64.op_count++;
371
597
#ifndef CAPSTONE_DIET
372
597
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
373
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
374
597
      MI->ac_idx++;
375
597
#endif
376
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
377
597
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = MCOperand_getImm(Op3) - MCOperand_getImm(Op2) + 1;
378
597
      MI->flat_insn->detail->arm64.op_count++;
379
597
    }
380
381
597
    return;
382
1.38k
  }
383
384
406k
  if (Opcode == AArch64_BFMXri || Opcode == AArch64_BFMWri) {
385
1.73k
    MCOperand *Op0 = MCInst_getOperand(MI, 0); // Op1 == Op0
386
1.73k
    MCOperand *Op2 = MCInst_getOperand(MI, 2);
387
1.73k
    int ImmR = (int)MCOperand_getImm(MCInst_getOperand(MI, 3));
388
1.73k
    int ImmS = (int)MCOperand_getImm(MCInst_getOperand(MI, 4));
389
390
1.73k
    if ((MCOperand_getReg(Op2) == AArch64_WZR || MCOperand_getReg(Op2) == AArch64_XZR) &&
391
1.02k
        (ImmR == 0 || ImmS < ImmR)) {
392
      // BFC takes precedence over its entire range, sligtly differently to BFI.
393
681
      int BitWidth = Opcode == AArch64_BFMXri ? 64 : 32;
394
681
      int LSB = (BitWidth - ImmR) % BitWidth;
395
681
      int Width = ImmS + 1;
396
397
681
      SStream_concat(O, "bfc\t%s, ",
398
681
          getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName));
399
400
681
      printInt32Bang(O, LSB);
401
681
      SStream_concat0(O, ", ");
402
681
      printInt32Bang(O, Width);
403
681
      MCInst_setOpcodePub(MI, AArch64_map_insn("bfc"));
404
405
681
      if (MI->csh->detail) {
406
681
#ifndef CAPSTONE_DIET
407
681
        uint8_t access;
408
681
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
409
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
410
681
        MI->ac_idx++;
411
681
#endif
412
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
413
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
414
681
        MI->flat_insn->detail->arm64.op_count++;
415
416
681
#ifndef CAPSTONE_DIET
417
681
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
418
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
419
681
        MI->ac_idx++;
420
681
#endif
421
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
422
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = LSB;
423
681
        MI->flat_insn->detail->arm64.op_count++;
424
681
#ifndef CAPSTONE_DIET
425
681
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
426
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
427
681
        MI->ac_idx++;
428
681
#endif
429
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
430
681
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Width;
431
681
        MI->flat_insn->detail->arm64.op_count++;
432
681
      }
433
434
681
      return;
435
1.05k
    } else if (ImmS < ImmR) {
436
      // BFI alias
437
376
      int BitWidth = Opcode == AArch64_BFMXri ? 64 : 32;
438
376
      LSB = (BitWidth - ImmR) % BitWidth;
439
376
      Width = ImmS + 1;
440
441
376
      SStream_concat(O, "bfi\t%s, %s, ",
442
376
          getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName),
443
376
          getRegisterName(MCOperand_getReg(Op2), AArch64_NoRegAltName));
444
445
376
      printInt32Bang(O, LSB);
446
376
      SStream_concat0(O, ", ");
447
376
      printInt32Bang(O, Width);
448
449
376
      MCInst_setOpcodePub(MI, AArch64_map_insn("bfi"));
450
451
376
      if (MI->csh->detail) {
452
376
#ifndef CAPSTONE_DIET
453
376
        uint8_t access;
454
376
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
455
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
456
376
        MI->ac_idx++;
457
376
#endif
458
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
459
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
460
376
        MI->flat_insn->detail->arm64.op_count++;
461
376
#ifndef CAPSTONE_DIET
462
376
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
463
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
464
376
        MI->ac_idx++;
465
376
#endif
466
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
467
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op2);
468
376
        MI->flat_insn->detail->arm64.op_count++;
469
376
#ifndef CAPSTONE_DIET
470
376
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
471
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
472
376
        MI->ac_idx++;
473
376
#endif
474
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
475
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = LSB;
476
376
        MI->flat_insn->detail->arm64.op_count++;
477
376
#ifndef CAPSTONE_DIET
478
376
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
479
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
480
376
        MI->ac_idx++;
481
376
#endif
482
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
483
376
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Width;
484
376
        MI->flat_insn->detail->arm64.op_count++;
485
376
      }
486
487
376
      return;
488
376
    }
489
490
678
    LSB = ImmR;
491
678
    Width = ImmS - ImmR + 1;
492
    // Otherwise BFXIL the preferred form
493
678
    SStream_concat(O, "bfxil\t%s, %s, ",
494
678
        getRegisterName(MCOperand_getReg(Op0), AArch64_NoRegAltName),
495
678
        getRegisterName(MCOperand_getReg(Op2), AArch64_NoRegAltName));
496
497
678
    printInt32Bang(O, LSB);
498
678
    SStream_concat0(O, ", ");
499
678
    printInt32Bang(O, Width);
500
501
678
    MCInst_setOpcodePub(MI, AArch64_map_insn("bfxil"));
502
503
678
    if (MI->csh->detail) {
504
678
#ifndef CAPSTONE_DIET
505
678
      uint8_t access;
506
678
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
507
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
508
678
      MI->ac_idx++;
509
678
#endif
510
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
511
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op0);
512
678
      MI->flat_insn->detail->arm64.op_count++;
513
678
#ifndef CAPSTONE_DIET
514
678
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
515
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
516
678
      MI->ac_idx++;
517
678
#endif
518
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
519
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(Op2);
520
678
      MI->flat_insn->detail->arm64.op_count++;
521
678
#ifndef CAPSTONE_DIET
522
678
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
523
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
524
678
      MI->ac_idx++;
525
678
#endif
526
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
527
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = LSB;
528
678
      MI->flat_insn->detail->arm64.op_count++;
529
678
#ifndef CAPSTONE_DIET
530
678
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
531
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
532
678
      MI->ac_idx++;
533
678
#endif
534
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
535
678
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Width;
536
678
      MI->flat_insn->detail->arm64.op_count++;
537
678
    }
538
539
678
    return;
540
1.73k
  }
541
542
  // MOVZ, MOVN and "ORR wzr, #imm" instructions are aliases for MOV, but their
543
  // domains overlap so they need to be prioritized. The chain is "MOVZ lsl #0 >
544
  // MOVZ lsl #N > MOVN lsl #0 > MOVN lsl #N > ORR". The highest instruction
545
  // that can represent the move is the MOV alias, and the rest get printed
546
  // normally.
547
404k
  if ((Opcode == AArch64_MOVZXi || Opcode == AArch64_MOVZWi) &&
548
1.20k
      MCOperand_isImm(MCInst_getOperand(MI, 1)) && MCOperand_isImm(MCInst_getOperand(MI, 2))) {
549
1.20k
    int RegWidth = Opcode == AArch64_MOVZXi ? 64 : 32;
550
1.20k
    int Shift = MCOperand_getImm(MCInst_getOperand(MI, 2));
551
1.20k
    uint64_t Value = (uint64_t)MCOperand_getImm(MCInst_getOperand(MI, 1)) << Shift;
552
553
1.20k
    if (isMOVZMovAlias(Value, Shift,
554
1.20k
          Opcode == AArch64_MOVZXi ? 64 : 32)) {
555
941
      SStream_concat(O, "mov\t%s, ", getRegisterName(MCOperand_getReg(MCInst_getOperand(MI, 0)), AArch64_NoRegAltName));
556
557
941
      printInt64Bang(O, SignExtend64(Value, RegWidth));
558
559
941
      if (MI->csh->detail) {
560
941
#ifndef CAPSTONE_DIET
561
941
        uint8_t access;
562
941
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
563
941
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
564
941
        MI->ac_idx++;
565
941
#endif
566
941
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
567
941
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(MCInst_getOperand(MI, 0));
568
941
        MI->flat_insn->detail->arm64.op_count++;
569
570
941
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
571
941
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = SignExtend64(Value, RegWidth);
572
941
        MI->flat_insn->detail->arm64.op_count++;
573
941
      }
574
575
941
      MCInst_setOpcodePub(MI, AArch64_map_insn("mov"));
576
577
941
      return;
578
941
    }
579
1.20k
  }
580
581
404k
  if ((Opcode == AArch64_MOVNXi || Opcode == AArch64_MOVNWi) &&
582
1.28k
      MCOperand_isImm(MCInst_getOperand(MI, 1)) && MCOperand_isImm(MCInst_getOperand(MI, 2))) {
583
1.28k
    int RegWidth = Opcode == AArch64_MOVNXi ? 64 : 32;
584
1.28k
    int Shift = MCOperand_getImm(MCInst_getOperand(MI, 2));
585
1.28k
    uint64_t Value = ~((uint64_t)MCOperand_getImm(MCInst_getOperand(MI, 1)) << Shift);
586
587
1.28k
    if (RegWidth == 32)
588
372
      Value = Value & 0xffffffff;
589
590
1.28k
    if (AArch64_AM_isMOVNMovAlias(Value, Shift, RegWidth)) {
591
1.12k
      SStream_concat(O, "mov\t%s, ", getRegisterName(MCOperand_getReg(MCInst_getOperand(MI, 0)), AArch64_NoRegAltName));
592
593
1.12k
      printInt64Bang(O, SignExtend64(Value, RegWidth));
594
595
1.12k
      if (MI->csh->detail) {
596
1.12k
#ifndef CAPSTONE_DIET
597
1.12k
        uint8_t access;
598
1.12k
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
599
1.12k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
600
1.12k
        MI->ac_idx++;
601
1.12k
#endif
602
1.12k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
603
1.12k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(MCInst_getOperand(MI, 0));
604
1.12k
        MI->flat_insn->detail->arm64.op_count++;
605
606
1.12k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
607
1.12k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = SignExtend64(Value, RegWidth);
608
1.12k
        MI->flat_insn->detail->arm64.op_count++;
609
1.12k
      }
610
611
1.12k
      MCInst_setOpcodePub(MI, AArch64_map_insn("mov"));
612
613
1.12k
      return;
614
1.12k
    }
615
1.28k
  }
616
617
402k
  if ((Opcode == AArch64_ORRXri || Opcode == AArch64_ORRWri) &&
618
1.79k
      (MCOperand_getReg(MCInst_getOperand(MI, 1)) == AArch64_XZR ||
619
1.49k
       MCOperand_getReg(MCInst_getOperand(MI, 1)) == AArch64_WZR) &&
620
650
      MCOperand_isImm(MCInst_getOperand(MI, 2))) {
621
650
    int RegWidth = Opcode == AArch64_ORRXri ? 64 : 32;
622
650
    uint64_t Value = AArch64_AM_decodeLogicalImmediate(
623
650
        MCOperand_getImm(MCInst_getOperand(MI, 2)), RegWidth);
624
650
    SStream_concat(O, "mov\t%s, ", getRegisterName(MCOperand_getReg(MCInst_getOperand(MI, 0)), AArch64_NoRegAltName));
625
626
650
    printInt64Bang(O, SignExtend64(Value, RegWidth));
627
628
650
    if (MI->csh->detail) {
629
650
#ifndef CAPSTONE_DIET
630
650
      uint8_t access;
631
650
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
632
650
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
633
650
      MI->ac_idx++;
634
650
#endif
635
650
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
636
650
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(MCInst_getOperand(MI, 0));
637
650
      MI->flat_insn->detail->arm64.op_count++;
638
639
650
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
640
650
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = SignExtend64(Value, RegWidth);
641
650
      MI->flat_insn->detail->arm64.op_count++;
642
650
    }
643
644
650
    MCInst_setOpcodePub(MI, AArch64_map_insn("mov"));
645
646
650
    return;
647
650
  }
648
649
  // Instruction TSB is specified as a one operand instruction, but 'csync' is
650
  // not encoded, so for printing it is treated as a special case here:
651
402k
  if (Opcode == AArch64_TSB) {
652
186
    SStream_concat0(O, "tsb\tcsync");
653
186
    MCInst_setOpcodePub(MI, AArch64_map_insn("tsb"));
654
186
    return;
655
186
  }
656
657
402k
  MI->MRI = Info;
658
659
402k
  mnem = printAliasInstr(MI, O, (MCRegisterInfo *)Info);
660
402k
  if (mnem) {
661
48.8k
    MCInst_setOpcodePub(MI, AArch64_map_insn(mnem));
662
48.8k
    cs_mem_free(mnem);
663
664
48.8k
    switch(MCInst_getOpcode(MI)) {
665
29.0k
      default: break;
666
29.0k
      case AArch64_LD1i8_POST:
667
423
        arm64_op_addImm(MI, 1);
668
423
        break;
669
113
      case AArch64_LD1i16_POST:
670
113
        arm64_op_addImm(MI, 2);
671
113
        break;
672
368
      case AArch64_LD1i32_POST:
673
368
        arm64_op_addImm(MI, 4);
674
368
        break;
675
129
      case AArch64_LD1Onev1d_POST:
676
501
      case AArch64_LD1Onev2s_POST:
677
645
      case AArch64_LD1Onev4h_POST:
678
895
      case AArch64_LD1Onev8b_POST:
679
1.62k
      case AArch64_LD1i64_POST:
680
1.62k
        arm64_op_addImm(MI, 8);
681
1.62k
        break;
682
94
      case AArch64_LD1Onev16b_POST:
683
662
      case AArch64_LD1Onev2d_POST:
684
814
      case AArch64_LD1Onev4s_POST:
685
959
      case AArch64_LD1Onev8h_POST:
686
1.17k
      case AArch64_LD1Twov1d_POST:
687
1.24k
      case AArch64_LD1Twov2s_POST:
688
1.42k
      case AArch64_LD1Twov4h_POST:
689
2.07k
      case AArch64_LD1Twov8b_POST:
690
2.07k
        arm64_op_addImm(MI, 16);
691
2.07k
        break;
692
71
      case AArch64_LD1Threev1d_POST:
693
202
      case AArch64_LD1Threev2s_POST:
694
248
      case AArch64_LD1Threev4h_POST:
695
654
      case AArch64_LD1Threev8b_POST:
696
654
        arm64_op_addImm(MI, 24);
697
654
        break;
698
117
      case AArch64_LD1Fourv1d_POST:
699
185
      case AArch64_LD1Fourv2s_POST:
700
237
      case AArch64_LD1Fourv4h_POST:
701
504
      case AArch64_LD1Fourv8b_POST:
702
719
      case AArch64_LD1Twov16b_POST:
703
953
      case AArch64_LD1Twov2d_POST:
704
1.13k
      case AArch64_LD1Twov4s_POST:
705
1.21k
      case AArch64_LD1Twov8h_POST:
706
1.21k
        arm64_op_addImm(MI, 32);
707
1.21k
        break;
708
469
      case AArch64_LD1Threev16b_POST:
709
591
      case AArch64_LD1Threev2d_POST:
710
1.15k
      case AArch64_LD1Threev4s_POST:
711
1.34k
      case AArch64_LD1Threev8h_POST:
712
1.34k
         arm64_op_addImm(MI, 48);
713
1.34k
         break;
714
280
      case AArch64_LD1Fourv16b_POST:
715
376
      case AArch64_LD1Fourv2d_POST:
716
885
      case AArch64_LD1Fourv4s_POST:
717
1.90k
      case AArch64_LD1Fourv8h_POST:
718
1.90k
        arm64_op_addImm(MI, 64);
719
1.90k
        break;
720
72
      case AArch64_UMOVvi64:
721
72
        arm64_op_addVectorArrSpecifier(MI, ARM64_VAS_1D);
722
72
        break;
723
71
      case AArch64_UMOVvi32:
724
71
        arm64_op_addVectorArrSpecifier(MI, ARM64_VAS_1S);
725
71
        break;
726
76
      case AArch64_INSvi8gpr:
727
303
      case AArch64_DUP_ZI_B:
728
483
      case AArch64_CPY_ZPmI_B:
729
692
      case AArch64_CPY_ZPzI_B:
730
845
      case AArch64_CPY_ZPmV_B:
731
1.43k
      case AArch64_CPY_ZPmR_B:
732
1.57k
      case AArch64_DUP_ZR_B:
733
1.57k
        if (MI->csh->detail) {
734
1.57k
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1B;
735
1.57k
        }
736
1.57k
        break;
737
75
      case AArch64_INSvi16gpr:
738
116
      case AArch64_DUP_ZI_H:
739
242
      case AArch64_CPY_ZPmI_H:
740
352
      case AArch64_CPY_ZPzI_H:
741
686
      case AArch64_CPY_ZPmV_H:
742
759
      case AArch64_CPY_ZPmR_H:
743
895
      case AArch64_DUP_ZR_H:
744
990
      case AArch64_FCPY_ZPmI_H:
745
1.32k
      case AArch64_FDUP_ZI_H:
746
1.32k
        if (MI->csh->detail) {
747
1.32k
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1H;
748
1.32k
        }
749
1.32k
        break;
750
161
      case AArch64_INSvi32gpr:
751
232
      case AArch64_DUP_ZI_S:
752
544
      case AArch64_CPY_ZPmI_S:
753
631
      case AArch64_CPY_ZPzI_S:
754
671
      case AArch64_CPY_ZPmV_S:
755
785
      case AArch64_CPY_ZPmR_S:
756
880
      case AArch64_DUP_ZR_S:
757
1.16k
      case AArch64_FCPY_ZPmI_S:
758
1.24k
      case AArch64_FDUP_ZI_S:
759
1.24k
        if (MI->csh->detail) {
760
1.24k
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1S;
761
1.24k
        }
762
1.24k
        break;
763
151
      case AArch64_INSvi64gpr:
764
230
      case AArch64_DUP_ZI_D:
765
335
      case AArch64_CPY_ZPmI_D:
766
561
      case AArch64_CPY_ZPzI_D:
767
764
      case AArch64_CPY_ZPmV_D:
768
914
      case AArch64_CPY_ZPmR_D:
769
1.04k
      case AArch64_DUP_ZR_D:
770
1.13k
      case AArch64_FCPY_ZPmI_D:
771
1.21k
      case AArch64_FDUP_ZI_D:
772
1.21k
        if (MI->csh->detail) {
773
1.21k
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1D;
774
1.21k
        }
775
1.21k
        break;
776
47
      case AArch64_INSvi8lane:
777
442
      case AArch64_ORR_PPzPP:
778
581
      case AArch64_ORRS_PPzPP:
779
581
        if (MI->csh->detail) {
780
581
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1B;
781
581
          MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1B;
782
581
        }
783
581
        break;
784
137
      case AArch64_INSvi16lane:
785
137
        if (MI->csh->detail) {
786
137
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1H;
787
137
          MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1H;
788
137
        }
789
137
         break;
790
231
      case AArch64_INSvi32lane:
791
231
        if (MI->csh->detail) {
792
231
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1S;
793
231
          MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1S;
794
231
        }
795
231
        break;
796
83
      case AArch64_INSvi64lane:
797
153
      case AArch64_ORR_ZZZ:
798
153
        if (MI->csh->detail) {
799
153
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1D;
800
153
          MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1D;
801
153
        }
802
153
        break;
803
104
      case AArch64_ORRv16i8:
804
172
      case AArch64_NOTv16i8:
805
172
        if (MI->csh->detail) {
806
172
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_16B;
807
172
          MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_16B;
808
172
        }
809
172
        break;
810
43
      case AArch64_ORRv8i8:
811
109
      case AArch64_NOTv8i8:
812
109
        if (MI->csh->detail) {
813
109
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_8B;
814
109
          MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_8B;
815
109
        }
816
109
        break;
817
383
      case AArch64_AND_PPzPP:
818
457
      case AArch64_ANDS_PPzPP:
819
493
      case AArch64_EOR_PPzPP:
820
562
      case AArch64_EORS_PPzPP:
821
725
      case AArch64_SEL_PPPP:
822
1.70k
      case AArch64_SEL_ZPZZ_B:
823
1.70k
        if (MI->csh->detail) {
824
1.70k
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1B;
825
1.70k
          MI->flat_insn->detail->arm64.operands[2].vas = ARM64_VAS_1B;
826
1.70k
        }
827
1.70k
        break;
828
37
      case AArch64_SEL_ZPZZ_D:
829
37
        if (MI->csh->detail) {
830
37
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1D;
831
37
          MI->flat_insn->detail->arm64.operands[2].vas = ARM64_VAS_1D;
832
37
        }
833
37
        break;
834
212
      case AArch64_SEL_ZPZZ_H:
835
212
        if (MI->csh->detail) {
836
212
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1H;
837
212
          MI->flat_insn->detail->arm64.operands[2].vas = ARM64_VAS_1H;
838
212
        }
839
212
        break;
840
298
      case AArch64_SEL_ZPZZ_S:
841
298
        if (MI->csh->detail) {
842
298
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1S;
843
298
          MI->flat_insn->detail->arm64.operands[2].vas = ARM64_VAS_1S;
844
298
        }
845
298
        break;
846
95
      case AArch64_DUP_ZZI_B:
847
95
        if (MI->csh->detail) {
848
95
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1B;
849
95
          if (MI->flat_insn->detail->arm64.op_count == 1) {
850
0
            arm64_op_addReg(MI, ARM64_REG_B0 + MCOperand_getReg(MCInst_getOperand(MI, 1)) - ARM64_REG_Z0);
851
95
          } else {
852
95
            MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1B;
853
95
          }
854
95
        }
855
95
        break;
856
176
      case AArch64_DUP_ZZI_D:
857
176
        if (MI->csh->detail) {
858
176
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1D;
859
176
          if (MI->flat_insn->detail->arm64.op_count == 1) {
860
0
            arm64_op_addReg(MI, ARM64_REG_D0 + MCOperand_getReg(MCInst_getOperand(MI, 1)) - ARM64_REG_Z0);
861
176
          } else {
862
176
            MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1D;
863
176
          }
864
176
        }
865
176
        break;
866
95
      case AArch64_DUP_ZZI_H:
867
95
        if (MI->csh->detail) {
868
95
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1H;
869
95
          if (MI->flat_insn->detail->arm64.op_count == 1) {
870
0
            arm64_op_addReg(MI, ARM64_REG_H0 + MCOperand_getReg(MCInst_getOperand(MI, 1)) - ARM64_REG_Z0);
871
95
          } else {
872
95
            MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1H;
873
95
          }
874
95
        }
875
95
        break;
876
115
      case AArch64_DUP_ZZI_Q:
877
115
        if (MI->csh->detail) {
878
115
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1Q;
879
115
          if (MI->flat_insn->detail->arm64.op_count == 1) {
880
0
            arm64_op_addReg(MI, ARM64_REG_Q0 + MCOperand_getReg(MCInst_getOperand(MI, 1)) - ARM64_REG_Z0);
881
115
          } else {
882
115
            MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1Q;
883
115
          }
884
115
         }
885
115
         break;
886
329
      case AArch64_DUP_ZZI_S:
887
329
        if (MI->csh->detail) {
888
329
          MI->flat_insn->detail->arm64.operands[0].vas = ARM64_VAS_1S;
889
329
          if (MI->flat_insn->detail->arm64.op_count == 1) {
890
0
            arm64_op_addReg(MI, ARM64_REG_S0 + MCOperand_getReg(MCInst_getOperand(MI, 1)) - ARM64_REG_Z0);
891
329
          } else {
892
329
             MI->flat_insn->detail->arm64.operands[1].vas = ARM64_VAS_1S;
893
329
          }
894
329
        }
895
329
        break;
896
      // Hacky detail filling of SMSTART and SMSTOP alias'
897
55
      case AArch64_MSRpstatesvcrImm1:{
898
55
        if(MI->csh->detail){
899
55
          MI->flat_insn->detail->arm64.op_count = 2;
900
55
#ifndef CAPSTONE_DIET
901
55
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
902
55
          MI->ac_idx++;
903
55
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
904
55
          MI->ac_idx++;
905
55
#endif
906
55
          MI->flat_insn->detail->arm64.operands[0].type = ARM64_OP_SVCR;
907
55
          MI->flat_insn->detail->arm64.operands[0].sys = (unsigned)ARM64_SYSREG_SVCR;
908
55
          MI->flat_insn->detail->arm64.operands[0].svcr = lookupSVCRByEncoding(MCOperand_getImm(MCInst_getOperand(MI, 0)))->Encoding;
909
55
          MI->flat_insn->detail->arm64.operands[1].type = ARM64_OP_IMM;
910
55
          MI->flat_insn->detail->arm64.operands[1].imm = MCOperand_getImm(MCInst_getOperand(MI, 1));
911
55
        }
912
55
        break;
913
725
      }
914
48.8k
    }
915
353k
  } else {
916
353k
    printInstruction(MI, O);
917
353k
  }
918
402k
}
919
920
static bool printSysAlias(MCInst *MI, SStream *O)
921
5.42k
{
922
  // unsigned Opcode = MCInst_getOpcode(MI);
923
  //assert(Opcode == AArch64_SYSxt && "Invalid opcode for SYS alias!");
924
925
5.42k
  const char *Ins;
926
5.42k
  uint16_t Encoding;
927
5.42k
  bool NeedsReg;
928
5.42k
  char Name[64];
929
5.42k
  MCOperand *Op1 = MCInst_getOperand(MI, 0);
930
5.42k
  MCOperand *Cn = MCInst_getOperand(MI, 1);
931
5.42k
  MCOperand *Cm = MCInst_getOperand(MI, 2);
932
5.42k
  MCOperand *Op2 = MCInst_getOperand(MI, 3);
933
934
5.42k
  unsigned Op1Val = (unsigned)MCOperand_getImm(Op1);
935
5.42k
  unsigned CnVal = (unsigned)MCOperand_getImm(Cn);
936
5.42k
  unsigned CmVal = (unsigned)MCOperand_getImm(Cm);
937
5.42k
  unsigned Op2Val = (unsigned)MCOperand_getImm(Op2);
938
939
5.42k
  Encoding = Op2Val;
940
5.42k
  Encoding |= CmVal << 3;
941
5.42k
  Encoding |= CnVal << 7;
942
5.42k
  Encoding |= Op1Val << 11;
943
944
5.42k
  if (CnVal == 7) {
945
4.48k
    switch (CmVal) {
946
180
      default:
947
180
        return false;
948
949
      // IC aliases
950
833
      case 1: case 5: {
951
833
        const IC *IC = lookupICByEncoding(Encoding);
952
        // if (!IC || !IC->haveFeatures(STI.getFeatureBits()))
953
833
        if (!IC)
954
208
          return false;
955
956
625
        NeedsReg = IC->NeedsReg;
957
625
        Ins = "ic";
958
625
        strncpy(Name, IC->Name, sizeof(Name) - 1);
959
625
      }
960
0
      break;
961
962
      // DC aliases
963
2.31k
      case 4: case 6: case 10: case 11: case 12: case 14: {
964
2.31k
        const DC *DC = lookupDCByEncoding(Encoding);
965
        // if (!DC || !DC->haveFeatures(STI.getFeatureBits()))
966
2.31k
        if (!DC)
967
1.92k
          return false;
968
969
397
        NeedsReg = true;
970
397
        Ins = "dc";
971
397
        strncpy(Name, DC->Name, sizeof(Name) - 1);
972
397
      }
973
0
      break;
974
975
      // AT aliases
976
1.14k
      case 8: case 9: {
977
1.14k
        const AT *AT = lookupATByEncoding(Encoding);
978
        // if (!AT || !AT->haveFeatures(STI.getFeatureBits()))
979
1.14k
        if (!AT)
980
344
          return false;
981
982
805
        NeedsReg = true;
983
805
        Ins = "at";
984
805
        strncpy(Name, AT->Name, sizeof(Name) - 1);
985
805
      }
986
0
      break;
987
4.48k
    }
988
4.48k
  } else if (CnVal == 8) {
989
    // TLBI aliases
990
513
    const TLBI *TLBI = lookupTLBIByEncoding(Encoding);
991
    // if (!TLBI || !TLBI->haveFeatures(STI.getFeatureBits()))
992
513
    if (!TLBI)
993
337
      return false;
994
995
176
    NeedsReg = TLBI->NeedsReg;
996
176
    Ins = "tlbi";
997
176
    strncpy(Name, TLBI->Name, sizeof(Name) - 1);
998
176
  } else
999
428
    return false;
1000
1001
2.00k
  SStream_concat(O, "%s\t%s", Ins, Name);
1002
1003
2.00k
  if (NeedsReg) {
1004
1.36k
    SStream_concat(O, ", %s", getRegisterName(MCOperand_getReg(MCInst_getOperand(MI, 4)), AArch64_NoRegAltName));
1005
1.36k
  }
1006
1007
2.00k
  MCInst_setOpcodePub(MI, AArch64_map_insn(Ins));
1008
1009
2.00k
  if (MI->csh->detail) {
1010
#if 0
1011
#ifndef CAPSTONE_DIET
1012
    uint8_t access;
1013
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1014
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1015
    MI->ac_idx++;
1016
#endif
1017
#endif
1018
2.00k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
1019
2.00k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = AArch64_map_sys_op(Name);
1020
2.00k
    MI->flat_insn->detail->arm64.op_count++;
1021
1022
2.00k
    if (NeedsReg) {
1023
1.36k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1024
1.36k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(MCInst_getOperand(MI, 4));
1025
1.36k
      MI->flat_insn->detail->arm64.op_count++;
1026
1.36k
    }
1027
2.00k
  }
1028
1029
2.00k
  return true;
1030
5.42k
}
1031
1032
static void printOperand(MCInst *MI, unsigned OpNum, SStream *O)
1033
572k
{
1034
572k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1035
1036
572k
  if (MCOperand_isReg(Op)) {
1037
494k
    unsigned Reg = MCOperand_getReg(Op);
1038
1039
494k
    SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
1040
1041
494k
    if (MI->csh->detail) {
1042
494k
      if (MI->csh->doing_mem) {
1043
226k
        if (MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.base == ARM64_REG_INVALID) {
1044
199k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.base = Reg;
1045
199k
        }
1046
26.5k
        else if (MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.index == ARM64_REG_INVALID) {
1047
26.5k
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.index = Reg;
1048
26.5k
        }
1049
268k
      } else if (MI->csh->doing_SME_Index) {
1050
        // Access op_count-1 as We want to add info to previous operand, not create a new one
1051
8.02k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count-1].sme_index.base = Reg;
1052
260k
      } else {
1053
260k
#ifndef CAPSTONE_DIET
1054
260k
        uint8_t access;
1055
1056
260k
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1057
260k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1058
260k
        MI->ac_idx++;
1059
260k
#endif
1060
260k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1061
260k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
1062
260k
        MI->flat_insn->detail->arm64.op_count++;
1063
260k
      }
1064
494k
    }
1065
494k
  } else if (MCOperand_isImm(Op)) {
1066
78.6k
    int64_t imm = MCOperand_getImm(Op);
1067
1068
78.6k
    if (MI->Opcode == AArch64_ADR) {
1069
4.40k
      imm += MI->address;
1070
4.40k
      printUInt64Bang(O, imm);
1071
74.2k
    } else {
1072
74.2k
      if (MI->csh->doing_mem) {
1073
21.8k
        if (MI->csh->imm_unsigned) {
1074
0
          printUInt64Bang(O, imm);
1075
21.8k
        } else {
1076
21.8k
          printInt64Bang(O, imm);
1077
21.8k
        }
1078
21.8k
      } else
1079
52.3k
        printUInt64Bang(O, imm);
1080
74.2k
    }
1081
1082
78.6k
    if (MI->csh->detail) {
1083
78.6k
      if (MI->csh->doing_mem) {
1084
21.8k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.disp = (int32_t)imm;
1085
56.7k
      } else if (MI->csh->doing_SME_Index) {
1086
        // Access op_count-1 as We want to add info to previous operand, not create a new one
1087
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count-1].sme_index.disp = (int32_t)imm; 
1088
56.7k
      } else {
1089
56.7k
#ifndef CAPSTONE_DIET
1090
56.7k
        uint8_t access;
1091
1092
56.7k
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1093
56.7k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1094
56.7k
#endif
1095
56.7k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1096
56.7k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = imm;
1097
56.7k
        MI->flat_insn->detail->arm64.op_count++;
1098
56.7k
      }
1099
78.6k
    }
1100
78.6k
  }
1101
572k
}
1102
1103
static void printImm(MCInst *MI, unsigned OpNum, SStream *O)
1104
7.07k
{
1105
7.07k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1106
7.07k
  printUInt64Bang(O, MCOperand_getImm(Op));
1107
1108
7.07k
  if (MI->csh->detail) {
1109
7.07k
#ifndef CAPSTONE_DIET
1110
7.07k
    uint8_t access;
1111
7.07k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1112
7.07k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1113
7.07k
    MI->ac_idx++;
1114
7.07k
#endif
1115
7.07k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1116
7.07k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = MCOperand_getImm(Op);
1117
7.07k
    MI->flat_insn->detail->arm64.op_count++;
1118
7.07k
  }
1119
7.07k
}
1120
1121
static void printImmHex(MCInst *MI, unsigned OpNum, SStream *O)
1122
104
{
1123
104
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1124
104
  printUInt64Bang(O, MCOperand_getImm(Op));
1125
1126
104
  if (MI->csh->detail) {
1127
104
#ifndef CAPSTONE_DIET
1128
104
    uint8_t access;
1129
104
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1130
104
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1131
104
    MI->ac_idx++;
1132
104
#endif
1133
104
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1134
104
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = MCOperand_getImm(Op);
1135
104
    MI->flat_insn->detail->arm64.op_count++;
1136
104
  }
1137
104
}
1138
1139
2.71k
static void printSImm(MCInst *MI, unsigned OpNo, SStream *O, int Size) {
1140
2.71k
  MCOperand *Op = MCInst_getOperand(MI, OpNo);
1141
2.71k
  if (Size == 8)
1142
2.02k
  printInt64Bang(O, (signed char) MCOperand_getImm(Op));
1143
694
  else if (Size == 16)
1144
694
  printInt64Bang(O, (signed short) MCOperand_getImm(Op));
1145
0
  else
1146
0
    printInt64Bang(O, MCOperand_getImm(Op));
1147
1148
2.71k
  if (MI->csh->detail) {
1149
2.71k
#ifndef CAPSTONE_DIET
1150
2.71k
    uint8_t access;
1151
2.71k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1152
2.71k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1153
2.71k
    MI->ac_idx++;
1154
2.71k
#endif
1155
2.71k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1156
2.71k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = MCOperand_getImm(Op);
1157
2.71k
    MI->flat_insn->detail->arm64.op_count++;
1158
2.71k
  }
1159
2.71k
}
1160
1161
static void printPostIncOperand(MCInst *MI, unsigned OpNum, SStream *O,
1162
    unsigned Imm)
1163
44.0k
{
1164
44.0k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1165
1166
44.0k
  if (MCOperand_isReg(Op)) {
1167
44.0k
    unsigned Reg = MCOperand_getReg(Op);
1168
44.0k
    if (Reg == AArch64_XZR) {
1169
0
      printInt32Bang(O, Imm);
1170
1171
0
      if (MI->csh->detail) {
1172
0
#ifndef CAPSTONE_DIET
1173
0
        uint8_t access;
1174
1175
0
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1176
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1177
0
        MI->ac_idx++;
1178
0
#endif
1179
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1180
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Imm;
1181
0
        MI->flat_insn->detail->arm64.op_count++;
1182
0
      }
1183
44.0k
    } else {
1184
44.0k
      SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
1185
1186
44.0k
      if (MI->csh->detail) {
1187
44.0k
#ifndef CAPSTONE_DIET
1188
44.0k
        uint8_t access;
1189
1190
44.0k
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1191
44.0k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1192
44.0k
        MI->ac_idx++;
1193
44.0k
#endif
1194
44.0k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1195
44.0k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
1196
44.0k
        MI->flat_insn->detail->arm64.op_count++;
1197
44.0k
      }
1198
44.0k
    }
1199
44.0k
  }
1200
  //llvm_unreachable("unknown operand kind in printPostIncOperand64");
1201
44.0k
}
1202
1203
static void printVRegOperand(MCInst *MI, unsigned OpNum, SStream *O)
1204
74.9k
{
1205
74.9k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1206
  //assert(Op.isReg() && "Non-register vreg operand!");
1207
74.9k
  unsigned Reg = MCOperand_getReg(Op);
1208
1209
74.9k
  SStream_concat0(O, getRegisterName(Reg, AArch64_vreg));
1210
1211
74.9k
  if (MI->csh->detail) {
1212
74.9k
#ifndef CAPSTONE_DIET
1213
74.9k
    uint8_t access;
1214
74.9k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1215
74.9k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1216
74.9k
    MI->ac_idx++;
1217
74.9k
#endif
1218
74.9k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1219
74.9k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = AArch64_map_vregister(Reg);
1220
74.9k
    MI->flat_insn->detail->arm64.op_count++;
1221
74.9k
  }
1222
74.9k
}
1223
1224
static void printSysCROperand(MCInst *MI, unsigned OpNum, SStream *O)
1225
7.18k
{
1226
7.18k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1227
7.18k
  unsigned Val = (unsigned)MCOperand_getImm(Op);
1228
  //assert(Op.isImm() && "System instruction C[nm] operands must be immediates!");
1229
7.18k
  SStream_concat(O, "c%u", Val);
1230
1231
7.18k
  if (MI->csh->detail) {
1232
7.18k
#ifndef CAPSTONE_DIET
1233
7.18k
    uint8_t access;
1234
1235
7.18k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1236
7.18k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1237
7.18k
    MI->ac_idx++;
1238
7.18k
#endif
1239
7.18k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_CIMM;
1240
7.18k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
1241
7.18k
    MI->flat_insn->detail->arm64.op_count++;
1242
7.18k
  }
1243
7.18k
}
1244
1245
static void printAddSubImm(MCInst *MI, unsigned OpNum, SStream *O)
1246
4.04k
{
1247
4.04k
  MCOperand *MO = MCInst_getOperand(MI, OpNum);
1248
4.04k
  if (MCOperand_isImm(MO)) {
1249
4.04k
    unsigned Val = (MCOperand_getImm(MO) & 0xfff);
1250
    //assert(Val == MO.getImm() && "Add/sub immediate out of range!");
1251
4.04k
    unsigned Shift = AArch64_AM_getShiftValue((int)MCOperand_getImm(MCInst_getOperand(MI, OpNum + 1)));
1252
1253
4.04k
    printInt32Bang(O, Val);
1254
1255
4.04k
    if (MI->csh->detail) {
1256
4.04k
#ifndef CAPSTONE_DIET
1257
4.04k
      uint8_t access;
1258
1259
4.04k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1260
4.04k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1261
4.04k
      MI->ac_idx++;
1262
4.04k
#endif
1263
4.04k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1264
4.04k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
1265
4.04k
      MI->flat_insn->detail->arm64.op_count++;
1266
4.04k
    }
1267
1268
4.04k
    if (Shift != 0)
1269
2.24k
      printShifter(MI, OpNum + 1, O);
1270
4.04k
  }
1271
4.04k
}
1272
1273
static void printLogicalImm32(MCInst *MI, unsigned OpNum, SStream *O)
1274
4.82k
{
1275
4.82k
  int64_t Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1276
1277
4.82k
  Val = AArch64_AM_decodeLogicalImmediate(Val, 32);
1278
4.82k
  printUInt32Bang(O, (int)Val);
1279
1280
4.82k
  if (MI->csh->detail) {
1281
4.82k
#ifndef CAPSTONE_DIET
1282
4.82k
    uint8_t access;
1283
1284
4.82k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1285
4.82k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1286
4.82k
    MI->ac_idx++;
1287
4.82k
#endif
1288
4.82k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1289
4.82k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
1290
4.82k
    MI->flat_insn->detail->arm64.op_count++;
1291
4.82k
  }
1292
4.82k
}
1293
1294
static void printLogicalImm64(MCInst *MI, unsigned OpNum, SStream *O)
1295
3.77k
{
1296
3.77k
  int64_t Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1297
3.77k
  Val = AArch64_AM_decodeLogicalImmediate(Val, 64);
1298
1299
3.77k
  switch(MI->flat_insn->id) {
1300
1.71k
    default:
1301
1.71k
      printInt64Bang(O, Val);
1302
1.71k
      break;
1303
1304
544
    case ARM64_INS_ORR:
1305
1.74k
    case ARM64_INS_AND:
1306
2.05k
    case ARM64_INS_EOR:
1307
2.05k
    case ARM64_INS_TST:
1308
      // do not print number in negative form
1309
2.05k
      if (Val >= 0 && Val <= HEX_THRESHOLD)
1310
110
        SStream_concat(O, "#%u", (int)Val);
1311
1.94k
      else
1312
1.94k
        SStream_concat(O, "#0x%"PRIx64, Val);
1313
2.05k
      break;
1314
3.77k
  }
1315
1316
3.77k
  if (MI->csh->detail) {
1317
3.77k
#ifndef CAPSTONE_DIET
1318
3.77k
    uint8_t access;
1319
1320
3.77k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1321
3.77k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1322
3.77k
    MI->ac_idx++;
1323
3.77k
#endif
1324
3.77k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1325
3.77k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = (int64_t)Val;
1326
3.77k
    MI->flat_insn->detail->arm64.op_count++;
1327
3.77k
  }
1328
3.77k
}
1329
1330
static void printShifter(MCInst *MI, unsigned OpNum, SStream *O)
1331
17.5k
{
1332
17.5k
  unsigned Val = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1333
1334
  // LSL #0 should not be printed.
1335
17.5k
  if (AArch64_AM_getShiftType(Val) == AArch64_AM_LSL &&
1336
9.79k
      AArch64_AM_getShiftValue(Val) == 0)
1337
1.62k
    return;
1338
1339
15.9k
  SStream_concat(O, ", %s ", AArch64_AM_getShiftExtendName(AArch64_AM_getShiftType(Val)));
1340
15.9k
  printInt32BangDec(O, AArch64_AM_getShiftValue(Val));
1341
1342
15.9k
  if (MI->csh->detail) {
1343
15.9k
    arm64_shifter shifter = ARM64_SFT_INVALID;
1344
1345
15.9k
    switch(AArch64_AM_getShiftType(Val)) {
1346
0
      default:  // never reach
1347
8.16k
      case AArch64_AM_LSL:
1348
8.16k
        shifter = ARM64_SFT_LSL;
1349
8.16k
        break;
1350
1351
2.65k
      case AArch64_AM_LSR:
1352
2.65k
        shifter = ARM64_SFT_LSR;
1353
2.65k
        break;
1354
1355
2.44k
      case AArch64_AM_ASR:
1356
2.44k
        shifter = ARM64_SFT_ASR;
1357
2.44k
        break;
1358
1359
1.83k
      case AArch64_AM_ROR:
1360
1.83k
        shifter = ARM64_SFT_ROR;
1361
1.83k
        break;
1362
1363
841
      case AArch64_AM_MSL:
1364
841
        shifter = ARM64_SFT_MSL;
1365
841
        break;
1366
15.9k
    }
1367
1368
15.9k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].shift.type = shifter;
1369
15.9k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].shift.value = AArch64_AM_getShiftValue(Val);
1370
15.9k
  }
1371
15.9k
}
1372
1373
static void printShiftedRegister(MCInst *MI, unsigned OpNum, SStream *O)
1374
9.79k
{
1375
9.79k
  SStream_concat0(O, getRegisterName(MCOperand_getReg(MCInst_getOperand(MI, OpNum)), AArch64_NoRegAltName));
1376
1377
9.79k
  if (MI->csh->detail) {
1378
9.79k
#ifndef CAPSTONE_DIET
1379
9.79k
    uint8_t access;
1380
9.79k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1381
9.79k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1382
9.79k
    MI->ac_idx++;
1383
9.79k
#endif
1384
9.79k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1385
9.79k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
1386
9.79k
    MI->flat_insn->detail->arm64.op_count++;
1387
9.79k
  }
1388
1389
9.79k
  printShifter(MI, OpNum + 1, O);
1390
9.79k
}
1391
1392
static void printArithExtend(MCInst *MI, unsigned OpNum, SStream *O)
1393
4.58k
{
1394
4.58k
  unsigned Val = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1395
4.58k
  AArch64_AM_ShiftExtendType ExtType = AArch64_AM_getArithExtendType(Val);
1396
4.58k
  unsigned ShiftVal = AArch64_AM_getArithShiftValue(Val);
1397
1398
  // If the destination or first source register operand is [W]SP, print
1399
  // UXTW/UXTX as LSL, and if the shift amount is also zero, print nothing at
1400
  // all.
1401
4.58k
  if (ExtType == AArch64_AM_UXTW || ExtType == AArch64_AM_UXTX) {
1402
2.37k
    unsigned Dest = MCOperand_getReg(MCInst_getOperand(MI, 0));
1403
2.37k
    unsigned Src1 = MCOperand_getReg(MCInst_getOperand(MI, 1));
1404
1405
2.37k
    if (((Dest == AArch64_SP || Src1 == AArch64_SP) &&
1406
511
          ExtType == AArch64_AM_UXTX) ||
1407
2.22k
        ((Dest == AArch64_WSP || Src1 == AArch64_WSP) &&
1408
768
         ExtType == AArch64_AM_UXTW)) {
1409
223
      if (ShiftVal != 0) {
1410
223
        SStream_concat0(O, ", lsl ");
1411
223
        printInt32Bang(O, ShiftVal);
1412
1413
223
        if (MI->csh->detail) {
1414
223
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].shift.type = ARM64_SFT_LSL;
1415
223
          MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].shift.value = ShiftVal;
1416
223
        }
1417
223
      }
1418
1419
223
      return;
1420
223
    }
1421
2.37k
  }
1422
1423
4.36k
  SStream_concat(O, ", %s", AArch64_AM_getShiftExtendName(ExtType));
1424
1425
4.36k
  if (MI->csh->detail) {
1426
4.36k
    arm64_extender ext = ARM64_EXT_INVALID;
1427
4.36k
    switch(ExtType) {
1428
0
      default:  // never reach
1429
1430
452
      case AArch64_AM_UXTB:
1431
452
        ext = ARM64_EXT_UXTB;
1432
452
        break;
1433
1434
554
      case AArch64_AM_UXTH:
1435
554
        ext = ARM64_EXT_UXTH;
1436
554
        break;
1437
1438
1.28k
      case AArch64_AM_UXTW:
1439
1.28k
        ext = ARM64_EXT_UXTW;
1440
1.28k
        break;
1441
1442
869
      case AArch64_AM_UXTX:
1443
869
        ext = ARM64_EXT_UXTX;
1444
869
        break;
1445
1446
100
      case AArch64_AM_SXTB:
1447
100
        ext = ARM64_EXT_SXTB;
1448
100
        break;
1449
1450
529
      case AArch64_AM_SXTH:
1451
529
        ext = ARM64_EXT_SXTH;
1452
529
        break;
1453
1454
201
      case AArch64_AM_SXTW:
1455
201
        ext = ARM64_EXT_SXTW;
1456
201
        break;
1457
1458
375
      case AArch64_AM_SXTX:
1459
375
        ext = ARM64_EXT_SXTX;
1460
375
        break;
1461
4.36k
    }
1462
1463
4.36k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].ext = ext;
1464
4.36k
  }
1465
1466
4.36k
  if (ShiftVal != 0) {
1467
3.73k
    SStream_concat0(O, " ");
1468
3.73k
    printInt32Bang(O, ShiftVal);
1469
1470
3.73k
    if (MI->csh->detail) {
1471
3.73k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].shift.type = ARM64_SFT_LSL;
1472
3.73k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].shift.value = ShiftVal;
1473
3.73k
    }
1474
3.73k
  }
1475
4.36k
}
1476
1477
static void printExtendedRegister(MCInst *MI, unsigned OpNum, SStream *O)
1478
3.28k
{
1479
3.28k
  unsigned Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
1480
1481
3.28k
  SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
1482
1483
3.28k
  if (MI->csh->detail) {
1484
3.28k
#ifndef CAPSTONE_DIET
1485
3.28k
    uint8_t access;
1486
3.28k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1487
3.28k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1488
3.28k
    MI->ac_idx++;
1489
3.28k
#endif
1490
3.28k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1491
3.28k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
1492
3.28k
    MI->flat_insn->detail->arm64.op_count++;
1493
3.28k
  }
1494
1495
3.28k
  printArithExtend(MI, OpNum + 1, O);
1496
3.28k
}
1497
1498
static void printMemExtendImpl(MCInst *MI, bool SignExtend, bool DoShift, unsigned Width,
1499
             char SrcRegKind, SStream *O)
1500
26.1k
{
1501
  // sxtw, sxtx, uxtw or lsl (== uxtx)
1502
26.1k
  bool IsLSL = !SignExtend && SrcRegKind == 'x';
1503
26.1k
  if (IsLSL) {
1504
10.1k
    SStream_concat0(O, "lsl");
1505
1506
10.1k
    if (MI->csh->detail) {
1507
10.1k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].shift.type = ARM64_SFT_LSL;
1508
10.1k
    }
1509
16.0k
  } else {
1510
16.0k
    SStream_concat(O, "%cxt%c", (SignExtend ? 's' : 'u'), SrcRegKind);
1511
1512
16.0k
    if (MI->csh->detail) {
1513
16.0k
      if (!SignExtend) {
1514
7.22k
        switch(SrcRegKind) {
1515
0
          default: break;
1516
0
          case 'b':
1517
0
               MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_UXTB;
1518
0
               break;
1519
0
          case 'h':
1520
0
               MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_UXTH;
1521
0
               break;
1522
7.22k
          case 'w':
1523
7.22k
               MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_UXTW;
1524
7.22k
               break;
1525
7.22k
        }
1526
8.80k
      } else {
1527
8.80k
          switch(SrcRegKind) {
1528
0
            default: break;
1529
0
            case 'b':
1530
0
              MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_SXTB;
1531
0
              break;
1532
0
            case 'h':
1533
0
              MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_SXTH;
1534
0
              break;
1535
7.29k
            case 'w':
1536
7.29k
              MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_SXTW;
1537
7.29k
              break;
1538
1.51k
            case 'x':
1539
1.51k
              MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].ext = ARM64_EXT_SXTX;
1540
1.51k
              break;
1541
8.80k
          }
1542
8.80k
      }
1543
16.0k
    }
1544
16.0k
  }
1545
1546
26.1k
  if (DoShift || IsLSL) {
1547
17.7k
    SStream_concat(O, " #%u", Log2_32(Width / 8));
1548
1549
17.7k
    if (MI->csh->detail) {
1550
17.7k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].shift.type = ARM64_SFT_LSL;
1551
17.7k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].shift.value = Log2_32(Width / 8);
1552
17.7k
    }
1553
17.7k
  }
1554
26.1k
}
1555
1556
static void printMemExtend(MCInst *MI, unsigned OpNum, SStream *O, char SrcRegKind, unsigned Width)
1557
7.00k
{
1558
7.00k
  unsigned SignExtend = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1559
7.00k
  unsigned DoShift = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum + 1));
1560
1561
7.00k
  printMemExtendImpl(MI, SignExtend, DoShift, Width, SrcRegKind, O);
1562
7.00k
}
1563
1564
static void printRegWithShiftExtend(MCInst *MI, unsigned OpNum, SStream *O,
1565
            bool SignExtend, int ExtWidth,
1566
            char SrcRegKind, char Suffix)
1567
21.9k
{
1568
21.9k
  bool DoShift;
1569
1570
21.9k
  printOperand(MI, OpNum, O);
1571
1572
21.9k
  if (Suffix == 's' || Suffix == 'd')
1573
15.5k
    SStream_concat(O, ".%c", Suffix);
1574
1575
21.9k
  DoShift = ExtWidth != 8;
1576
21.9k
  if (SignExtend || DoShift || SrcRegKind == 'w') {
1577
19.1k
    SStream_concat0(O, ", ");
1578
19.1k
    printMemExtendImpl(MI, SignExtend, DoShift, ExtWidth, SrcRegKind, O);
1579
19.1k
  }
1580
21.9k
}
1581
1582
static void printCondCode(MCInst *MI, unsigned OpNum, SStream *O)
1583
4.61k
{
1584
4.61k
  AArch64CC_CondCode CC = (AArch64CC_CondCode)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1585
4.61k
  SStream_concat0(O, getCondCodeName(CC));
1586
1587
4.61k
  if (MI->csh->detail)
1588
4.61k
    MI->flat_insn->detail->arm64.cc = (arm64_cc)(CC + 1);
1589
4.61k
}
1590
1591
static void printInverseCondCode(MCInst *MI, unsigned OpNum, SStream *O)
1592
1.26k
{
1593
1.26k
  AArch64CC_CondCode CC = (AArch64CC_CondCode)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1594
1.26k
  SStream_concat0(O, getCondCodeName(getInvertedCondCode(CC)));
1595
1596
1.26k
  if (MI->csh->detail) {
1597
1.26k
    MI->flat_insn->detail->arm64.cc = (arm64_cc)(getInvertedCondCode(CC) + 1);
1598
1.26k
  }
1599
1.26k
}
1600
1601
static void printImmScale(MCInst *MI, unsigned OpNum, SStream *O, int Scale)
1602
29.8k
{
1603
29.8k
  int64_t val = Scale * MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1604
1605
29.8k
  printInt64Bang(O, val);
1606
1607
29.8k
  if (MI->csh->detail) {
1608
29.8k
    if (MI->csh->doing_mem) {
1609
24.3k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.disp = (int32_t)val;
1610
24.3k
    } else {
1611
5.48k
#ifndef CAPSTONE_DIET
1612
5.48k
      uint8_t access;
1613
1614
5.48k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1615
5.48k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1616
5.48k
      MI->ac_idx++;
1617
5.48k
#endif
1618
5.48k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1619
5.48k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = val;
1620
5.48k
      MI->flat_insn->detail->arm64.op_count++;
1621
5.48k
    }
1622
29.8k
  }
1623
29.8k
}
1624
1625
static void printUImm12Offset(MCInst *MI, unsigned OpNum, SStream *O, unsigned Scale)
1626
12.3k
{
1627
12.3k
  MCOperand *MO = MCInst_getOperand(MI, OpNum);
1628
1629
12.3k
  if (MCOperand_isImm(MO)) {
1630
12.3k
    int64_t val = Scale * MCOperand_getImm(MO);
1631
12.3k
    printInt64Bang(O, val);
1632
1633
12.3k
    if (MI->csh->detail) {
1634
12.3k
      if (MI->csh->doing_mem) {
1635
12.3k
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].mem.disp = (int32_t)val;
1636
12.3k
      } else {
1637
0
#ifndef CAPSTONE_DIET
1638
0
        uint8_t access;
1639
1640
0
        access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1641
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1642
0
        MI->ac_idx++;
1643
0
#endif
1644
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1645
0
        MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = (int)val;
1646
0
        MI->flat_insn->detail->arm64.op_count++;
1647
0
      }
1648
12.3k
    }
1649
12.3k
  }
1650
12.3k
}
1651
1652
#if 0
1653
static void printAMIndexedWB(MCInst *MI, unsigned OpNum, SStream *O, unsigned int Scale)
1654
{
1655
  MCOperand *MO = MCInst_getOperand(MI, OpNum + 1);
1656
1657
  SStream_concat(O, "[%s", getRegisterName(MCOperand_getReg(MCInst_getOperand(MI, OpNum)), AArch64_NoRegAltName));
1658
1659
  if (MCOperand_isImm(MO)) {
1660
    int64_t val = Scale * MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1661
    printInt64Bang(O, val);
1662
  // } else {
1663
  //   // assert(MO1.isExpr() && "Unexpected operand type!");
1664
  //   SStream_concat0(O, ", ");
1665
  //   MO1.getExpr()->print(O, &MAI);
1666
  }
1667
1668
  SStream_concat0(O, "]");
1669
}
1670
#endif
1671
1672
// IsSVEPrefetch = false
1673
static void printPrefetchOp(MCInst *MI, unsigned OpNum, SStream *O, bool IsSVEPrefetch)
1674
10.0k
{
1675
10.0k
  unsigned prfop = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1676
1677
10.0k
  if (IsSVEPrefetch) {
1678
7.76k
    const SVEPRFM *PRFM = lookupSVEPRFMByEncoding(prfop);
1679
7.76k
    if (PRFM)
1680
6.43k
      SStream_concat0(O, PRFM->Name);
1681
1682
7.76k
    return;
1683
7.76k
  } else {
1684
2.25k
    const PRFM *PRFM = lookupPRFMByEncoding(prfop);
1685
2.25k
    if (PRFM)
1686
1.44k
      SStream_concat0(O, PRFM->Name);
1687
1688
2.25k
    return;
1689
2.25k
  }
1690
1691
  // FIXME: set OpcodePub?
1692
1693
0
  printInt32Bang(O, prfop);
1694
1695
0
  if (MI->csh->detail) {
1696
0
#ifndef CAPSTONE_DIET
1697
0
    uint8_t access;
1698
0
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1699
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1700
0
    MI->ac_idx++;
1701
0
#endif
1702
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
1703
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = prfop;
1704
0
    MI->flat_insn->detail->arm64.op_count++;
1705
0
  }
1706
0
}
1707
1708
static void printPSBHintOp(MCInst *MI, unsigned OpNum, SStream *O)
1709
927
{
1710
927
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1711
927
  unsigned int psbhintop = MCOperand_getImm(Op);
1712
1713
927
  const PSB *PSB = lookupPSBByEncoding(psbhintop);
1714
927
  if (PSB)
1715
927
    SStream_concat0(O, PSB->Name);
1716
0
  else
1717
0
    printUInt32Bang(O, psbhintop);
1718
927
}
1719
1720
335
static void printBTIHintOp(MCInst *MI, unsigned OpNum, SStream *O) {
1721
335
  unsigned btihintop = MCOperand_getImm(MCInst_getOperand(MI, OpNum)) ^ 32;
1722
1723
335
  const BTI *BTI = lookupBTIByEncoding(btihintop);
1724
335
  if (BTI)
1725
335
  SStream_concat0(O, BTI->Name);
1726
0
  else
1727
0
  printUInt32Bang(O, btihintop);
1728
335
}
1729
1730
static void printFPImmOperand(MCInst *MI, unsigned OpNum, SStream *O)
1731
1.99k
{
1732
1.99k
  MCOperand *MO = MCInst_getOperand(MI, OpNum);
1733
1.99k
  float FPImm = MCOperand_isFPImm(MO) ? MCOperand_getFPImm(MO) : AArch64_AM_getFPImmFloat((int)MCOperand_getImm(MO));
1734
1735
  // 8 decimal places are enough to perfectly represent permitted floats.
1736
#if defined(_KERNEL_MODE)
1737
  // Issue #681: Windows kernel does not support formatting float point
1738
  SStream_concat0(O, "#<float_point_unsupported>");
1739
#else
1740
1.99k
  SStream_concat(O, "#%.8f", FPImm);
1741
1.99k
#endif
1742
1743
1.99k
  if (MI->csh->detail) {
1744
1.99k
#ifndef CAPSTONE_DIET
1745
1.99k
    uint8_t access;
1746
1747
1.99k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1748
1.99k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1749
1.99k
    MI->ac_idx++;
1750
1.99k
#endif
1751
1.99k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_FP;
1752
1.99k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].fp = FPImm;
1753
1.99k
    MI->flat_insn->detail->arm64.op_count++;
1754
1.99k
  }
1755
1.99k
}
1756
1757
//static unsigned getNextVectorRegister(unsigned Reg, unsigned Stride = 1)
1758
static unsigned getNextVectorRegister(unsigned Reg, unsigned Stride)
1759
258k
{
1760
517k
  while (Stride--) {
1761
258k
    if (Reg >= AArch64_Q0 && Reg <= AArch64_Q30) // AArch64_Q0 .. AArch64_Q30
1762
223k
      Reg += 1;
1763
34.6k
    else if (Reg == AArch64_Q31) // Vector lists can wrap around.
1764
9.32k
      Reg = AArch64_Q0;
1765
25.3k
    else if (Reg >= AArch64_Z0 && Reg <= AArch64_Z30) // AArch64_Z0 .. AArch64_Z30
1766
23.8k
      Reg += 1;
1767
1.51k
    else if (Reg == AArch64_Z31) // Vector lists can wrap around.
1768
1.51k
      Reg = AArch64_Z0;
1769
258k
  }
1770
1771
258k
  return Reg;
1772
258k
}
1773
1774
static void printGPRSeqPairsClassOperand(MCInst *MI, unsigned OpNum, SStream *O, unsigned int size)
1775
4.53k
{
1776
  // static_assert(size == 64 || size == 32,
1777
  //    "Template parameter must be either 32 or 64");
1778
4.53k
  unsigned Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
1779
4.53k
  unsigned Sube = (size == 32) ? AArch64_sube32 : AArch64_sube64;
1780
4.53k
  unsigned Subo = (size == 32) ? AArch64_subo32 : AArch64_subo64;
1781
4.53k
  unsigned Even = MCRegisterInfo_getSubReg(MI->MRI, Reg, Sube);
1782
4.53k
  unsigned Odd = MCRegisterInfo_getSubReg(MI->MRI, Reg, Subo);
1783
1784
4.53k
  SStream_concat(O, "%s, %s", getRegisterName(Even, AArch64_NoRegAltName),
1785
4.53k
      getRegisterName(Odd, AArch64_NoRegAltName));
1786
1787
4.53k
  if (MI->csh->detail) {
1788
4.53k
#ifndef CAPSTONE_DIET
1789
4.53k
    uint8_t access;
1790
1791
4.53k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1792
4.53k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1793
4.53k
    MI->ac_idx++;
1794
4.53k
#endif
1795
1796
4.53k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1797
4.53k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Even;
1798
4.53k
    MI->flat_insn->detail->arm64.op_count++;
1799
1800
4.53k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1801
4.53k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Odd;
1802
4.53k
    MI->flat_insn->detail->arm64.op_count++;
1803
4.53k
  }
1804
4.53k
}
1805
1806
static void printVectorList(MCInst *MI, unsigned OpNum, SStream *O,
1807
    char *LayoutSuffix, MCRegisterInfo *MRI, arm64_vas vas)
1808
106k
{
1809
1.55M
#define GETREGCLASS_CONTAIN0(_class, _reg) MCRegisterClass_contains(MCRegisterInfo_getRegClass(MRI, _class), _reg)
1810
106k
  unsigned Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
1811
106k
  unsigned NumRegs = 1, FirstReg, i;
1812
1813
106k
  SStream_concat0(O, "{");
1814
1815
  // Work out how many registers there are in the list (if there is an actual
1816
  // list).
1817
106k
  if (GETREGCLASS_CONTAIN0(AArch64_DDRegClassID , Reg) ||
1818
100k
      GETREGCLASS_CONTAIN0(AArch64_ZPR2RegClassID, Reg) ||
1819
99.6k
      GETREGCLASS_CONTAIN0(AArch64_QQRegClassID, Reg))
1820
24.7k
    NumRegs = 2;
1821
81.4k
  else if (GETREGCLASS_CONTAIN0(AArch64_DDDRegClassID, Reg) ||
1822
76.6k
      GETREGCLASS_CONTAIN0(AArch64_ZPR3RegClassID, Reg) ||
1823
75.2k
      GETREGCLASS_CONTAIN0(AArch64_QQQRegClassID, Reg))
1824
23.4k
    NumRegs = 3;
1825
57.9k
  else if (GETREGCLASS_CONTAIN0(AArch64_DDDDRegClassID, Reg) ||
1826
51.6k
      GETREGCLASS_CONTAIN0(AArch64_ZPR4RegClassID, Reg) ||
1827
50.5k
      GETREGCLASS_CONTAIN0(AArch64_QQQQRegClassID, Reg))
1828
26.8k
    NumRegs = 4;
1829
1830
  // Now forget about the list and find out what the first register is.
1831
106k
  if ((FirstReg = MCRegisterInfo_getSubReg(MRI, Reg, AArch64_dsub0)))
1832
16.5k
    Reg = FirstReg;
1833
89.7k
  else if ((FirstReg = MCRegisterInfo_getSubReg(MRI, Reg, AArch64_qsub0)))
1834
54.8k
    Reg = FirstReg;
1835
34.8k
  else if ((FirstReg = MCRegisterInfo_getSubReg(MRI, Reg, AArch64_zsub0)))
1836
3.73k
    Reg = FirstReg;
1837
1838
  // If it's a D-reg, we need to promote it to the equivalent Q-reg before
1839
  // printing (otherwise getRegisterName fails).
1840
106k
  if (GETREGCLASS_CONTAIN0(AArch64_FPR64RegClassID, Reg)) {
1841
18.9k
    const MCRegisterClass *FPR128RC = MCRegisterInfo_getRegClass(MRI, AArch64_FPR128RegClassID);
1842
18.9k
    Reg = MCRegisterInfo_getMatchingSuperReg(MRI, Reg, AArch64_dsub, FPR128RC);
1843
18.9k
  }
1844
1845
364k
  for (i = 0; i < NumRegs; ++i, Reg = getNextVectorRegister(Reg, 1)) {
1846
258k
    bool isZReg = GETREGCLASS_CONTAIN0(AArch64_ZPRRegClassID, Reg);
1847
258k
    if (isZReg)
1848
25.3k
      SStream_concat(O, "%s%s", getRegisterName(Reg, AArch64_NoRegAltName), LayoutSuffix);
1849
233k
    else
1850
233k
      SStream_concat(O, "%s%s", getRegisterName(Reg, AArch64_vreg), LayoutSuffix);
1851
1852
258k
    if (MI->csh->detail) {
1853
258k
#ifndef CAPSTONE_DIET
1854
258k
      uint8_t access;
1855
1856
258k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
1857
258k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
1858
258k
      MI->ac_idx++;
1859
258k
#endif
1860
258k
      unsigned regForDetail = isZReg ? Reg : AArch64_map_vregister(Reg);
1861
258k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
1862
258k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = regForDetail;
1863
258k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].vas = vas;
1864
258k
      MI->flat_insn->detail->arm64.op_count++;
1865
258k
    }
1866
1867
258k
    if (i + 1 != NumRegs)
1868
152k
      SStream_concat0(O, ", ");
1869
258k
  }
1870
1871
106k
  SStream_concat0(O, "}");
1872
106k
}
1873
1874
static void printTypedVectorList(MCInst *MI, unsigned OpNum, SStream *O, unsigned NumLanes, char LaneKind)
1875
106k
{
1876
106k
  char Suffix[32];
1877
106k
  arm64_vas vas = 0;
1878
1879
106k
  if (NumLanes) {
1880
47.3k
    cs_snprintf(Suffix, sizeof(Suffix), ".%u%c", NumLanes, LaneKind);
1881
1882
47.3k
    switch(LaneKind) {
1883
0
      default: break;
1884
14.5k
      case 'b':
1885
14.5k
        switch(NumLanes) {
1886
0
          default: break;
1887
0
          case 1:
1888
0
               vas = ARM64_VAS_1B;
1889
0
               break;
1890
0
          case 4:
1891
0
               vas = ARM64_VAS_4B;
1892
0
               break;
1893
5.08k
          case 8:
1894
5.08k
               vas = ARM64_VAS_8B;
1895
5.08k
               break;
1896
9.44k
          case 16:
1897
9.44k
               vas = ARM64_VAS_16B;
1898
9.44k
               break;
1899
14.5k
        }
1900
14.5k
        break;
1901
14.5k
      case 'h':
1902
12.9k
        switch(NumLanes) {
1903
0
          default: break;
1904
0
          case 1:
1905
0
               vas = ARM64_VAS_1H;
1906
0
               break;
1907
0
          case 2:
1908
0
               vas = ARM64_VAS_2H;
1909
0
               break;
1910
4.57k
          case 4:
1911
4.57k
               vas = ARM64_VAS_4H;
1912
4.57k
               break;
1913
8.37k
          case 8:
1914
8.37k
               vas = ARM64_VAS_8H;
1915
8.37k
               break;
1916
12.9k
        }
1917
12.9k
        break;
1918
12.9k
      case 's':
1919
12.0k
        switch(NumLanes) {
1920
0
          default: break;
1921
0
          case 1:
1922
0
               vas = ARM64_VAS_1S;
1923
0
               break;
1924
5.91k
          case 2:
1925
5.91k
               vas = ARM64_VAS_2S;
1926
5.91k
               break;
1927
6.10k
          case 4:
1928
6.10k
               vas = ARM64_VAS_4S;
1929
6.10k
               break;
1930
12.0k
        }
1931
12.0k
        break;
1932
12.0k
      case 'd':
1933
7.83k
        switch(NumLanes) {
1934
0
          default: break;
1935
3.40k
          case 1:
1936
3.40k
               vas = ARM64_VAS_1D;
1937
3.40k
               break;
1938
4.43k
          case 2:
1939
4.43k
               vas = ARM64_VAS_2D;
1940
4.43k
               break;
1941
7.83k
        }
1942
7.83k
        break;
1943
7.83k
      case 'q':
1944
0
        switch(NumLanes) {
1945
0
          default: break;
1946
0
          case 1:
1947
0
               vas = ARM64_VAS_1Q;
1948
0
               break;
1949
0
        }
1950
0
        break;
1951
47.3k
    }
1952
58.8k
  } else {
1953
58.8k
    cs_snprintf(Suffix, sizeof(Suffix), ".%c", LaneKind);
1954
1955
58.8k
    switch(LaneKind) {
1956
0
      default: break;
1957
13.0k
      case 'b':
1958
13.0k
           vas = ARM64_VAS_1B;
1959
13.0k
           break;
1960
13.4k
      case 'h':
1961
13.4k
           vas = ARM64_VAS_1H;
1962
13.4k
           break;
1963
14.0k
      case 's':
1964
14.0k
           vas = ARM64_VAS_1S;
1965
14.0k
           break;
1966
18.2k
      case 'd':
1967
18.2k
           vas = ARM64_VAS_1D;
1968
18.2k
           break;
1969
0
      case 'q':
1970
0
           vas = ARM64_VAS_1Q;
1971
0
           break;
1972
58.8k
    }
1973
58.8k
  }
1974
1975
106k
  printVectorList(MI, OpNum, O, Suffix, MI->MRI, vas);
1976
106k
}
1977
1978
static void printVectorIndex(MCInst *MI, unsigned OpNum, SStream *O)
1979
56.9k
{
1980
56.9k
  SStream_concat0(O, "[");
1981
56.9k
  printInt32(O, (int)MCOperand_getImm(MCInst_getOperand(MI, OpNum)));
1982
56.9k
  SStream_concat0(O, "]");
1983
1984
56.9k
  if (MI->csh->detail) {
1985
56.9k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count - 1].vector_index = (int)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
1986
56.9k
  }
1987
56.9k
}
1988
1989
static void printAlignedLabel(MCInst *MI, unsigned OpNum, SStream *O)
1990
15.2k
{
1991
15.2k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
1992
1993
  // If the label has already been resolved to an immediate offset (say, when
1994
  // we're running the disassembler), just print the immediate.
1995
15.2k
  if (MCOperand_isImm(Op)) {
1996
15.2k
    uint64_t imm = (MCOperand_getImm(Op) * 4) + MI->address;
1997
15.2k
    printUInt64Bang(O, imm);
1998
1999
15.2k
    if (MI->csh->detail) {
2000
15.2k
#ifndef CAPSTONE_DIET
2001
15.2k
      uint8_t access;
2002
2003
15.2k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2004
15.2k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2005
15.2k
      MI->ac_idx++;
2006
15.2k
#endif
2007
15.2k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
2008
15.2k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = imm;
2009
15.2k
      MI->flat_insn->detail->arm64.op_count++;
2010
15.2k
    }
2011
15.2k
  }
2012
15.2k
}
2013
2014
static void printAdrpLabel(MCInst *MI, unsigned OpNum, SStream *O)
2015
1.91k
{
2016
1.91k
  MCOperand *Op = MCInst_getOperand(MI, OpNum);
2017
2018
1.91k
  if (MCOperand_isImm(Op)) {
2019
    // ADRP sign extends a 21-bit offset, shifts it left by 12
2020
    // and adds it to the value of the PC with its bottom 12 bits cleared
2021
1.91k
    uint64_t imm = (MCOperand_getImm(Op) * 0x1000) + (MI->address & ~0xfff);
2022
1.91k
    printUInt64Bang(O, imm);
2023
2024
1.91k
    if (MI->csh->detail) {
2025
1.91k
#ifndef CAPSTONE_DIET
2026
1.91k
      uint8_t access;
2027
2028
1.91k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2029
1.91k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2030
1.91k
      MI->ac_idx++;
2031
1.91k
#endif
2032
1.91k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
2033
1.91k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = imm;
2034
1.91k
      MI->flat_insn->detail->arm64.op_count++;
2035
1.91k
    }
2036
1.91k
  }
2037
1.91k
}
2038
2039
static void printBarrierOption(MCInst *MI, unsigned OpNum, SStream *O)
2040
982
{
2041
982
  unsigned Val = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2042
982
  unsigned Opcode = MCInst_getOpcode(MI);
2043
982
  const char *Name = NULL;
2044
2045
982
  if (Opcode == AArch64_ISB) {
2046
221
    const ISB *ISB = lookupISBByEncoding(Val);
2047
221
    Name = ISB ? ISB->Name : NULL;
2048
761
  } else if (Opcode == AArch64_TSB) {
2049
0
    const TSB *TSB = lookupTSBByEncoding(Val);
2050
0
    Name = TSB ? TSB->Name : NULL;
2051
761
  } else {
2052
761
    const DB *DB = lookupDBByEncoding(Val);
2053
761
    Name = DB ? DB->Name : NULL;
2054
761
  }
2055
2056
982
  if (Name) {
2057
86
    SStream_concat0(O, Name);
2058
2059
86
    if (MI->csh->detail) {
2060
86
#ifndef CAPSTONE_DIET
2061
86
      uint8_t access;
2062
2063
86
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2064
86
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2065
86
      MI->ac_idx++;
2066
86
#endif
2067
86
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_BARRIER;
2068
86
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].barrier = Val;
2069
86
      MI->flat_insn->detail->arm64.op_count++;
2070
86
    }
2071
896
  } else {
2072
896
    printUInt32Bang(O, Val);
2073
2074
896
    if (MI->csh->detail) {
2075
896
#ifndef CAPSTONE_DIET
2076
896
      uint8_t access;
2077
2078
896
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2079
896
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2080
896
      MI->ac_idx++;
2081
896
#endif
2082
896
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
2083
896
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
2084
896
      MI->flat_insn->detail->arm64.op_count++;
2085
896
    }
2086
896
  }
2087
982
}
2088
2089
215
static void printBarriernXSOption(MCInst *MI, unsigned OpNo, SStream *O) {
2090
215
  unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNo));
2091
  // assert(MI->getOpcode() == AArch64::DSBnXS);
2092
2093
215
  const char *Name = NULL;
2094
215
  const DBnXS *DB = lookupDBnXSByEncoding(Val);
2095
215
  Name = DB ? DB->Name : NULL;
2096
2097
215
  if (Name) {
2098
215
    SStream_concat0(O, Name);
2099
2100
215
    if (MI->csh->detail) {
2101
215
#ifndef CAPSTONE_DIET
2102
215
      uint8_t access;
2103
2104
215
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2105
215
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2106
215
      MI->ac_idx++;
2107
215
#endif
2108
215
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_BARRIER;
2109
215
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].barrier = Val;
2110
215
      MI->flat_insn->detail->arm64.op_count++;
2111
215
    }
2112
215
  }
2113
0
  else {
2114
0
    printUInt32Bang(O, Val);
2115
2116
0
    if (MI->csh->detail) {
2117
0
#ifndef CAPSTONE_DIET
2118
0
      uint8_t access;
2119
2120
0
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2121
0
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2122
0
      MI->ac_idx++;
2123
0
#endif
2124
0
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
2125
0
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
2126
0
      MI->flat_insn->detail->arm64.op_count++;
2127
0
    }
2128
0
  }
2129
215
}
2130
2131
static void printMRSSystemRegister(MCInst *MI, unsigned OpNum, SStream *O)
2132
2.05k
{
2133
2.05k
  unsigned Val = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2134
2.05k
  const SysReg *Reg = lookupSysRegByEncoding(Val);
2135
2136
  // Horrible hack for the one register that has identical encodings but
2137
  // different names in MSR and MRS. Because of this, one of MRS and MSR is
2138
  // going to get the wrong entry
2139
2.05k
  if (Val == ARM64_SYSREG_DBGDTRRX_EL0) {
2140
61
    SStream_concat0(O, "dbgdtrrx_el0");
2141
2142
61
    if (MI->csh->detail) {
2143
61
#ifndef CAPSTONE_DIET
2144
61
      uint8_t access;
2145
2146
61
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2147
61
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2148
61
      MI->ac_idx++;
2149
61
#endif
2150
2151
61
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
2152
61
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = Val;
2153
61
      MI->flat_insn->detail->arm64.op_count++;
2154
61
    }
2155
2156
61
    return;
2157
61
  }
2158
2159
  // Another hack for a register which has an alternative name which is not an alias,
2160
  // and is not in the Armv9-A documentation.
2161
1.99k
  if( Val == ARM64_SYSREG_VSCTLR_EL2){
2162
268
    SStream_concat0(O, "ttbr0_el2");
2163
2164
268
    if (MI->csh->detail) {
2165
268
#ifndef CAPSTONE_DIET
2166
268
      uint8_t access;
2167
2168
268
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2169
268
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2170
268
      MI->ac_idx++;
2171
268
#endif
2172
2173
268
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
2174
268
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = Val;
2175
268
      MI->flat_insn->detail->arm64.op_count++;
2176
268
    }
2177
2178
268
    return;
2179
268
  }
2180
2181
  // if (Reg && Reg->Readable && Reg->haveFeatures(STI.getFeatureBits()))
2182
1.72k
  if (Reg && Reg->Readable) {
2183
184
    SStream_concat0(O, Reg->Name);
2184
2185
184
    if (MI->csh->detail) {
2186
184
#ifndef CAPSTONE_DIET
2187
184
      uint8_t access;
2188
2189
184
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2190
184
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2191
184
      MI->ac_idx++;
2192
184
#endif
2193
2194
184
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
2195
184
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = Reg->Encoding;
2196
184
      MI->flat_insn->detail->arm64.op_count++;
2197
184
    }
2198
1.53k
  } else {
2199
1.53k
    char result[128];
2200
2201
1.53k
    AArch64SysReg_genericRegisterString(Val, result);
2202
1.53k
    SStream_concat0(O, result);
2203
2204
1.53k
    if (MI->csh->detail) {
2205
1.53k
#ifndef CAPSTONE_DIET
2206
1.53k
      uint8_t access;
2207
1.53k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2208
1.53k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2209
1.53k
      MI->ac_idx++;
2210
1.53k
#endif
2211
1.53k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG_MRS;
2212
1.53k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Val;
2213
1.53k
      MI->flat_insn->detail->arm64.op_count++;
2214
1.53k
    }
2215
1.53k
  }
2216
1.72k
}
2217
2218
static void printMSRSystemRegister(MCInst *MI, unsigned OpNum, SStream *O)
2219
4.68k
{
2220
4.68k
  unsigned Val = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2221
4.68k
  const SysReg *Reg = lookupSysRegByEncoding(Val);
2222
2223
  // Horrible hack for the one register that has identical encodings but
2224
  // different names in MSR and MRS. Because of this, one of MRS and MSR is
2225
  // going to get the wrong entry
2226
4.68k
  if (Val == ARM64_SYSREG_DBGDTRTX_EL0) {
2227
141
    SStream_concat0(O, "dbgdtrtx_el0");
2228
2229
141
    if (MI->csh->detail) {
2230
141
#ifndef CAPSTONE_DIET
2231
141
      uint8_t access;
2232
2233
141
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2234
141
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2235
141
      MI->ac_idx++;
2236
141
#endif
2237
2238
141
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
2239
141
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = Val;
2240
141
      MI->flat_insn->detail->arm64.op_count++;
2241
141
    }
2242
2243
141
    return;
2244
141
  }
2245
2246
  // Another hack for a register which has an alternative name which is not an alias,
2247
  // and is not in the Armv9-A documentation.
2248
4.54k
  if( Val == ARM64_SYSREG_VSCTLR_EL2){
2249
408
    SStream_concat0(O, "ttbr0_el2");
2250
2251
408
    if (MI->csh->detail) {
2252
408
#ifndef CAPSTONE_DIET
2253
408
      uint8_t access;
2254
2255
408
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2256
408
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2257
408
      MI->ac_idx++;
2258
408
#endif
2259
2260
408
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
2261
408
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = Val;
2262
408
      MI->flat_insn->detail->arm64.op_count++;
2263
408
    }
2264
2265
408
    return;
2266
408
  }
2267
2268
  // if (Reg && Reg->Writeable && Reg->haveFeatures(STI.getFeatureBits()))
2269
4.13k
  if (Reg && Reg->Writeable) {
2270
316
    SStream_concat0(O, Reg->Name);
2271
2272
316
    if (MI->csh->detail) {
2273
316
#ifndef CAPSTONE_DIET
2274
316
      uint8_t access;
2275
2276
316
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2277
316
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2278
316
      MI->ac_idx++;
2279
316
#endif
2280
2281
316
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SYS;
2282
316
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = Reg->Encoding;
2283
316
      MI->flat_insn->detail->arm64.op_count++;
2284
316
    }
2285
3.81k
  } else {
2286
3.81k
    char result[128];
2287
2288
3.81k
    AArch64SysReg_genericRegisterString(Val, result);
2289
3.81k
    SStream_concat0(O, result);
2290
2291
3.81k
    if (MI->csh->detail) {
2292
3.81k
#ifndef CAPSTONE_DIET
2293
3.81k
      uint8_t access;
2294
3.81k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2295
3.81k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2296
3.81k
      MI->ac_idx++;
2297
3.81k
#endif
2298
3.81k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG_MRS;
2299
3.81k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Val;
2300
3.81k
      MI->flat_insn->detail->arm64.op_count++;
2301
3.81k
    }
2302
3.81k
  }
2303
4.13k
}
2304
2305
static void printSystemPStateField(MCInst *MI, unsigned OpNum, SStream *O)
2306
534
{
2307
534
  unsigned Val = (unsigned)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2308
2309
534
  const PState *PState = lookupPStateByEncoding(Val);
2310
2311
534
  if (PState) {
2312
534
    SStream_concat0(O, PState->Name);
2313
2314
534
    if (MI->csh->detail) {
2315
534
#ifndef CAPSTONE_DIET
2316
534
      uint8_t access;
2317
534
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2318
534
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2319
534
      MI->ac_idx++;
2320
534
#endif
2321
534
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_PSTATE;
2322
534
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].pstate = Val;
2323
534
      MI->flat_insn->detail->arm64.op_count++;
2324
534
    }
2325
534
  } else {
2326
0
    printUInt32Bang(O, Val);
2327
2328
0
    if (MI->csh->detail) {
2329
0
#ifndef CAPSTONE_DIET
2330
0
      unsigned char access;
2331
2332
0
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2333
0
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2334
0
      MI->ac_idx++;
2335
0
#endif
2336
2337
0
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
2338
0
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
2339
0
      MI->flat_insn->detail->arm64.op_count++;
2340
0
    }
2341
0
  }
2342
534
}
2343
2344
static void printSIMDType10Operand(MCInst *MI, unsigned OpNum, SStream *O)
2345
1.35k
{
2346
1.35k
  uint8_t RawVal = (uint8_t)MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2347
1.35k
  uint64_t Val = AArch64_AM_decodeAdvSIMDModImmType10(RawVal);
2348
2349
1.35k
  SStream_concat(O, "#%#016llx", Val);
2350
2351
1.35k
  if (MI->csh->detail) {
2352
1.35k
#ifndef CAPSTONE_DIET
2353
1.35k
    unsigned char access;
2354
2355
1.35k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2356
1.35k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2357
1.35k
    MI->ac_idx++;
2358
1.35k
#endif
2359
1.35k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_IMM;
2360
1.35k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].imm = Val;
2361
1.35k
    MI->flat_insn->detail->arm64.op_count++;
2362
1.35k
  }
2363
1.35k
}
2364
2365
static void printComplexRotationOp(MCInst *MI, unsigned OpNum, SStream *O, int64_t Angle, int64_t Remainder)
2366
3.42k
{
2367
3.42k
  unsigned int Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2368
3.42k
  printInt64Bang(O, (Val * Angle) + Remainder);
2369
3.42k
  op_addImm(MI, (Val * Angle) + Remainder);
2370
3.42k
}
2371
2372
static void printSVCROp(MCInst *MI, unsigned OpNum, SStream *O)
2373
0
{
2374
0
  MCOperand *MO = MCInst_getOperand(MI, OpNum);
2375
    // assert(MCOperand_isImm(MO) && "Unexpected operand type!");
2376
0
    unsigned svcrop = MCOperand_getImm(MO);
2377
0
  const SVCR *svcr = lookupSVCRByEncoding(svcrop);
2378
    // assert(svcr && "Unexpected SVCR operand!");
2379
0
  SStream_concat0(O, svcr->Name);
2380
2381
0
  if (MI->csh->detail) {
2382
0
#ifndef CAPSTONE_DIET
2383
0
    uint8_t access;
2384
2385
0
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2386
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2387
0
    MI->ac_idx++;
2388
0
#endif
2389
2390
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_SVCR;
2391
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].sys = (unsigned)ARM64_SYSREG_SVCR;
2392
0
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].svcr = svcr->Encoding;
2393
0
    MI->flat_insn->detail->arm64.op_count++;
2394
0
  }
2395
0
}
2396
2397
static void printMatrix(MCInst *MI, unsigned OpNum, SStream *O, int EltSize)
2398
413
{
2399
413
  MCOperand *RegOp = MCInst_getOperand(MI, OpNum);
2400
    // assert(MCOperand_isReg(RegOp) && "Unexpected operand type!");
2401
413
  unsigned Reg = MCOperand_getReg(RegOp);
2402
2403
413
  SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
2404
413
  const char *sizeStr = "";
2405
413
    switch (EltSize) {
2406
413
    case 0:
2407
413
    sizeStr = "";
2408
413
      break;
2409
0
    case 8:
2410
0
      sizeStr = ".b";
2411
0
      break;
2412
0
    case 16:
2413
0
      sizeStr = ".h";
2414
0
      break;
2415
0
    case 32:
2416
0
      sizeStr = ".s";
2417
0
      break;
2418
0
    case 64:
2419
0
      sizeStr = ".d";
2420
0
      break;
2421
0
    case 128:
2422
0
      sizeStr = ".q";
2423
0
      break;
2424
0
    default:
2425
0
    break;
2426
    //   llvm_unreachable("Unsupported element size");
2427
413
    }
2428
413
  SStream_concat0(O, sizeStr);
2429
2430
413
  if (MI->csh->detail) {
2431
413
#ifndef CAPSTONE_DIET
2432
413
    uint8_t access;
2433
2434
413
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2435
413
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2436
413
    MI->ac_idx++;
2437
413
#endif
2438
2439
413
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
2440
413
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
2441
413
    MI->flat_insn->detail->arm64.op_count++;
2442
413
  }
2443
413
}
2444
2445
static void printMatrixIndex(MCInst *MI, unsigned OpNum, SStream *O)
2446
8.02k
{
2447
8.02k
  int64_t imm = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2448
8.02k
  printInt64(O, imm);
2449
2450
8.02k
  if (MI->csh->detail) {
2451
8.02k
    if (MI->csh->doing_SME_Index) {
2452
      // Access op_count-1 as We want to add info to previous operand, not create a new one
2453
8.02k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count-1].sme_index.disp = imm;
2454
8.02k
    }
2455
8.02k
  }
2456
8.02k
}
2457
2458
static void printMatrixTile(MCInst *MI, unsigned OpNum, SStream *O)
2459
2.50k
{
2460
2.50k
  MCOperand *RegOp = MCInst_getOperand(MI, OpNum);
2461
    // assert(MCOperand_isReg(RegOp) && "Unexpected operand type!");
2462
2.50k
  unsigned Reg = MCOperand_getReg(RegOp);
2463
2.50k
    SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
2464
2465
2.50k
  if (MI->csh->detail) {
2466
2.50k
#ifndef CAPSTONE_DIET
2467
2.50k
    uint8_t access;
2468
2469
2.50k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2470
2.50k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2471
2.50k
    MI->ac_idx++;
2472
2.50k
#endif
2473
2474
2.50k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
2475
2.50k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
2476
2.50k
    MI->flat_insn->detail->arm64.op_count++;
2477
2.50k
  }
2478
2.50k
}
2479
2480
static void printMatrixTileVector(MCInst *MI, unsigned OpNum, SStream *O, bool IsVertical)
2481
6.75k
{
2482
6.75k
  MCOperand *RegOp = MCInst_getOperand(MI, OpNum);
2483
    // assert(MCOperand_isReg(RegOp) && "Unexpected operand type!");
2484
6.75k
  unsigned Reg = MCOperand_getReg(RegOp);
2485
6.75k
#ifndef CAPSTONE_DIET
2486
6.75k
  const char *RegName = getRegisterName(Reg, AArch64_NoRegAltName);
2487
2488
6.75k
  const size_t strLn = strlen(RegName);
2489
  // +2 for extra chars, + 1 for null char \0
2490
6.75k
  char *RegNameNew = cs_mem_malloc(sizeof(char) * (strLn + 2 + 1));
2491
6.75k
  int index = 0, i;
2492
54.1k
  for (i = 0; i < (strLn + 2); i++){
2493
47.4k
    if(RegName[i] != '.'){
2494
40.6k
      RegNameNew[index] = RegName[i];
2495
40.6k
      index++;
2496
40.6k
    }
2497
6.75k
    else{
2498
6.75k
      RegNameNew[index] = IsVertical ? 'v' : 'h';
2499
6.75k
      RegNameNew[index + 1] = '.';
2500
6.75k
      index += 2;
2501
6.75k
    }
2502
47.4k
  }
2503
6.75k
  SStream_concat0(O, RegNameNew);
2504
6.75k
#endif
2505
2506
6.75k
  if (MI->csh->detail) {
2507
6.75k
#ifndef CAPSTONE_DIET
2508
6.75k
    uint8_t access;
2509
2510
6.75k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2511
6.75k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2512
6.75k
    MI->ac_idx++;
2513
6.75k
#endif
2514
2515
6.75k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
2516
6.75k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
2517
6.75k
    MI->flat_insn->detail->arm64.op_count++;
2518
6.75k
  }
2519
6.75k
#ifndef CAPSTONE_DIET
2520
6.75k
  cs_mem_free(RegNameNew);
2521
6.75k
#endif
2522
6.75k
}
2523
2524
static const unsigned MatrixZADRegisterTable[] = {
2525
  AArch64_ZAD0, AArch64_ZAD1, AArch64_ZAD2, AArch64_ZAD3,
2526
  AArch64_ZAD4, AArch64_ZAD5, AArch64_ZAD6, AArch64_ZAD7
2527
};
2528
2529
523
static void printMatrixTileList(MCInst *MI, unsigned OpNum, SStream *O){
2530
523
  unsigned MaxRegs = 8;
2531
523
  unsigned RegMask = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2532
2533
523
  unsigned NumRegs = 0, I;
2534
4.70k
  for (I = 0; I < MaxRegs; ++I)
2535
4.18k
    if ((RegMask & (1 << I)) != 0)
2536
1.31k
      ++NumRegs;
2537
2538
523
  SStream_concat0(O, "{");
2539
523
  unsigned Printed = 0, J;
2540
4.70k
  for (J = 0; J < MaxRegs; ++J) {
2541
4.18k
    unsigned Reg = RegMask & (1 << J);
2542
4.18k
    if (Reg == 0)
2543
2.87k
      continue;
2544
1.31k
    SStream_concat0(O, getRegisterName(MatrixZADRegisterTable[J], AArch64_NoRegAltName));
2545
2546
1.31k
    if (MI->csh->detail) {
2547
1.31k
#ifndef CAPSTONE_DIET
2548
1.31k
      uint8_t access;
2549
2550
1.31k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2551
1.31k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2552
1.31k
      MI->ac_idx++;
2553
1.31k
#endif
2554
2555
1.31k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
2556
1.31k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = MatrixZADRegisterTable[J];
2557
1.31k
      MI->flat_insn->detail->arm64.op_count++;
2558
1.31k
    }
2559
2560
1.31k
    if (Printed + 1 != NumRegs)
2561
793
      SStream_concat0(O, ", ");
2562
1.31k
    ++Printed;
2563
1.31k
  }
2564
523
  SStream_concat0(O, "}");
2565
523
}
2566
2567
static void printSVEPattern(MCInst *MI, unsigned OpNum, SStream *O)
2568
4.80k
{
2569
4.80k
  unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2570
2571
4.80k
  const SVEPREDPAT *Pat = lookupSVEPREDPATByEncoding(Val);
2572
4.80k
  if (Pat)
2573
3.37k
    SStream_concat0(O, Pat->Name);
2574
1.42k
  else
2575
1.42k
    printUInt32Bang(O, Val);
2576
4.80k
}
2577
2578
// default suffix = 0
2579
static void printSVERegOp(MCInst *MI, unsigned OpNum, SStream *O, char suffix)
2580
185k
{
2581
185k
  unsigned int Reg;
2582
2583
#if 0
2584
  switch (suffix) {
2585
    case 0:
2586
    case 'b':
2587
    case 'h':
2588
    case 's':
2589
    case 'd':
2590
    case 'q':
2591
      break;
2592
    default:
2593
      // llvm_unreachable("Invalid kind specifier.");
2594
  }
2595
#endif
2596
2597
185k
  Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
2598
2599
185k
  if (MI->csh->detail) {
2600
185k
#ifndef CAPSTONE_DIET
2601
185k
      uint8_t access;
2602
2603
185k
      access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2604
185k
      MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2605
185k
      MI->ac_idx++;
2606
185k
#endif
2607
185k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
2608
185k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
2609
185k
    MI->flat_insn->detail->arm64.op_count++;
2610
185k
  }
2611
2612
185k
  SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
2613
2614
185k
  if (suffix != '\0')
2615
120k
    SStream_concat(O, ".%c", suffix);
2616
185k
}
2617
2618
static void printImmSVE16(int16_t Val, SStream *O)
2619
1.00k
{
2620
1.00k
  printUInt32Bang(O, Val);
2621
1.00k
}
2622
2623
static void printImmSVE32(int32_t Val, SStream *O)
2624
1.74k
{
2625
1.74k
  printUInt32Bang(O, Val);
2626
1.74k
}
2627
2628
static void printImmSVE64(int64_t Val, SStream *O)
2629
1.36k
{
2630
1.36k
  printUInt64Bang(O, Val);
2631
1.36k
}
2632
2633
static void printImm8OptLsl32(MCInst *MI, unsigned OpNum, SStream *O)
2634
1.82k
{
2635
1.82k
  unsigned UnscaledVal = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2636
1.82k
  unsigned Shift = MCOperand_getImm(MCInst_getOperand(MI, OpNum + 1));
2637
1.82k
  uint32_t Val;
2638
2639
  // assert(AArch64_AM::getShiftType(Shift) == AArch64_AM::LSL &&
2640
  //  "Unexepected shift type!");
2641
2642
  // #0 lsl #8 is never pretty printed
2643
1.82k
  if ((UnscaledVal == 0) && (AArch64_AM_getShiftValue(Shift) != 0)) {
2644
78
    printUInt32Bang(O, UnscaledVal);
2645
78
    printShifter(MI, OpNum + 1, O);
2646
78
    return;
2647
78
  }
2648
2649
1.74k
  Val = UnscaledVal * (1 << AArch64_AM_getShiftValue(Shift));
2650
1.74k
  printImmSVE32(Val, O);
2651
1.74k
}
2652
2653
static void printImm8OptLsl64(MCInst *MI, unsigned OpNum, SStream *O)
2654
799
{
2655
799
  unsigned UnscaledVal = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2656
799
  unsigned Shift = MCOperand_getImm(MCInst_getOperand(MI, OpNum + 1));
2657
799
  uint64_t Val;
2658
2659
  // assert(AArch64_AM::getShiftType(Shift) == AArch64_AM::LSL &&
2660
  //  "Unexepected shift type!");
2661
2662
  // #0 lsl #8 is never pretty printed
2663
799
  if ((UnscaledVal == 0) && (AArch64_AM_getShiftValue(Shift) != 0)) {
2664
361
    printUInt32Bang(O, UnscaledVal);
2665
361
    printShifter(MI, OpNum + 1, O);
2666
361
    return;
2667
361
  }
2668
2669
438
  Val = UnscaledVal * (1 << AArch64_AM_getShiftValue(Shift));
2670
438
  printImmSVE64(Val, O);
2671
438
}
2672
2673
static void printSVELogicalImm16(MCInst *MI, unsigned OpNum, SStream *O)
2674
602
{
2675
602
  uint64_t Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2676
602
  uint64_t PrintVal = AArch64_AM_decodeLogicalImmediate(Val, 64);
2677
2678
  // Prefer the default format for 16bit values, hex otherwise.
2679
602
  printImmSVE16(PrintVal, O);
2680
602
}
2681
2682
static void printSVELogicalImm32(MCInst *MI, unsigned OpNum, SStream *O)
2683
1.22k
{
2684
1.22k
  uint64_t Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2685
1.22k
  uint64_t PrintVal = AArch64_AM_decodeLogicalImmediate(Val, 64);
2686
2687
  // Prefer the default format for 16bit values, hex otherwise.
2688
1.22k
  if ((uint16_t)PrintVal == (uint32_t)PrintVal)
2689
403
    printImmSVE16(PrintVal, O);
2690
817
  else
2691
817
    printUInt64Bang(O, PrintVal);
2692
1.22k
}
2693
2694
static void printSVELogicalImm64(MCInst *MI, unsigned OpNum, SStream *O)
2695
922
{
2696
922
  uint64_t Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2697
922
  uint64_t PrintVal = AArch64_AM_decodeLogicalImmediate(Val, 64);
2698
2699
922
  printImmSVE64(PrintVal, O);
2700
922
}
2701
2702
static void printZPRasFPR(MCInst *MI, unsigned OpNum, SStream *O, int Width)
2703
2.85k
{
2704
2.85k
  unsigned int Base, Reg;
2705
2706
2.85k
  switch (Width) {
2707
0
    default: // llvm_unreachable("Unsupported width");
2708
300
    case 8:   Base = AArch64_B0; break;
2709
744
    case 16:  Base = AArch64_H0; break;
2710
761
    case 32:  Base = AArch64_S0; break;
2711
938
    case 64:  Base = AArch64_D0; break;
2712
112
    case 128: Base = AArch64_Q0; break;
2713
2.85k
  }
2714
2715
2.85k
  Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum)) - AArch64_Z0 + Base;
2716
2717
2.85k
  SStream_concat0(O, getRegisterName(Reg, AArch64_NoRegAltName));
2718
2719
2.85k
  if (MI->csh->detail) {
2720
2.85k
#ifndef CAPSTONE_DIET
2721
2.85k
    uint8_t access;
2722
2723
2.85k
    access = get_op_access(MI->csh, MCInst_getOpcode(MI), MI->ac_idx);
2724
2.85k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].access = access;
2725
2.85k
    MI->ac_idx++;
2726
2.85k
#endif
2727
2.85k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].type = ARM64_OP_REG;
2728
2.85k
    MI->flat_insn->detail->arm64.operands[MI->flat_insn->detail->arm64.op_count].reg = Reg;
2729
2.85k
    MI->flat_insn->detail->arm64.op_count++;
2730
2.85k
  }
2731
2.85k
}
2732
2733
static void printExactFPImm(MCInst *MI, unsigned OpNum, SStream *O, unsigned ImmIs0, unsigned ImmIs1)
2734
696
{
2735
696
  const ExactFPImm *Imm0Desc = lookupExactFPImmByEnum(ImmIs0);
2736
696
  const ExactFPImm *Imm1Desc = lookupExactFPImmByEnum(ImmIs1);
2737
696
  unsigned Val = MCOperand_getImm(MCInst_getOperand(MI, OpNum));
2738
2739
696
  SStream_concat0(O, Val ? Imm1Desc->Repr : Imm0Desc->Repr);
2740
696
}
2741
2742
static void printGPR64as32(MCInst *MI, unsigned OpNum, SStream *O)
2743
5.88k
{
2744
5.88k
  unsigned int Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
2745
2746
5.88k
  SStream_concat0(O, getRegisterName(getWRegFromXReg(Reg), AArch64_NoRegAltName));
2747
5.88k
}
2748
2749
static void printGPR64x8(MCInst *MI, unsigned OpNum, SStream *O) 
2750
754
{
2751
754
    unsigned int Reg = MCOperand_getReg(MCInst_getOperand(MI, OpNum));
2752
2753
754
    SStream_concat0(O, getRegisterName(MCRegisterInfo_getSubReg(MI->MRI, Reg, AArch64_x8sub_0), AArch64_NoRegAltName));
2754
754
}
2755
2756
#define PRINT_ALIAS_INSTR
2757
#include "AArch64GenAsmWriter.inc"
2758
#include "AArch64GenRegisterName.inc"
2759
2760
void AArch64_post_printer(csh handle, cs_insn *flat_insn, char *insn_asm, MCInst *mci)
2761
412k
{
2762
412k
  if (((cs_struct *)handle)->detail != CS_OPT_ON)
2763
0
    return;
2764
2765
412k
  if (mci->csh->detail) {
2766
412k
    unsigned opcode = MCInst_getOpcode(mci);
2767
2768
412k
    switch (opcode) {
2769
329k
      default:
2770
329k
        break;
2771
329k
      case AArch64_LD1Fourv16b_POST:
2772
437
      case AArch64_LD1Fourv1d_POST:
2773
534
      case AArch64_LD1Fourv2d_POST:
2774
966
      case AArch64_LD1Fourv2s_POST:
2775
1.04k
      case AArch64_LD1Fourv4h_POST:
2776
1.70k
      case AArch64_LD1Fourv4s_POST:
2777
2.20k
      case AArch64_LD1Fourv8b_POST:
2778
3.23k
      case AArch64_LD1Fourv8h_POST:
2779
3.33k
      case AArch64_LD1Onev16b_POST:
2780
3.58k
      case AArch64_LD1Onev1d_POST:
2781
4.15k
      case AArch64_LD1Onev2d_POST:
2782
4.53k
      case AArch64_LD1Onev2s_POST:
2783
4.68k
      case AArch64_LD1Onev4h_POST:
2784
4.84k
      case AArch64_LD1Onev4s_POST:
2785
5.11k
      case AArch64_LD1Onev8b_POST:
2786
5.50k
      case AArch64_LD1Onev8h_POST:
2787
5.63k
      case AArch64_LD1Rv16b_POST:
2788
5.93k
      case AArch64_LD1Rv1d_POST:
2789
6.16k
      case AArch64_LD1Rv2d_POST:
2790
6.19k
      case AArch64_LD1Rv2s_POST:
2791
6.28k
      case AArch64_LD1Rv4h_POST:
2792
7.07k
      case AArch64_LD1Rv4s_POST:
2793
7.13k
      case AArch64_LD1Rv8b_POST:
2794
7.20k
      case AArch64_LD1Rv8h_POST:
2795
7.67k
      case AArch64_LD1Threev16b_POST:
2796
7.76k
      case AArch64_LD1Threev1d_POST:
2797
7.89k
      case AArch64_LD1Threev2d_POST:
2798
8.02k
      case AArch64_LD1Threev2s_POST:
2799
8.08k
      case AArch64_LD1Threev4h_POST:
2800
8.69k
      case AArch64_LD1Threev4s_POST:
2801
9.26k
      case AArch64_LD1Threev8b_POST:
2802
9.61k
      case AArch64_LD1Threev8h_POST:
2803
9.83k
      case AArch64_LD1Twov16b_POST:
2804
10.1k
      case AArch64_LD1Twov1d_POST:
2805
10.4k
      case AArch64_LD1Twov2d_POST:
2806
10.5k
      case AArch64_LD1Twov2s_POST:
2807
10.6k
      case AArch64_LD1Twov4h_POST:
2808
10.8k
      case AArch64_LD1Twov4s_POST:
2809
11.5k
      case AArch64_LD1Twov8b_POST:
2810
11.6k
      case AArch64_LD1Twov8h_POST:
2811
11.8k
      case AArch64_LD1i16_POST:
2812
13.5k
      case AArch64_LD1i32_POST:
2813
14.3k
      case AArch64_LD1i64_POST:
2814
15.4k
      case AArch64_LD1i8_POST:
2815
16.1k
      case AArch64_LD2Rv16b_POST:
2816
16.5k
      case AArch64_LD2Rv1d_POST:
2817
16.9k
      case AArch64_LD2Rv2d_POST:
2818
17.1k
      case AArch64_LD2Rv2s_POST:
2819
18.0k
      case AArch64_LD2Rv4h_POST:
2820
18.2k
      case AArch64_LD2Rv4s_POST:
2821
18.2k
      case AArch64_LD2Rv8b_POST:
2822
18.3k
      case AArch64_LD2Rv8h_POST:
2823
18.6k
      case AArch64_LD2Twov16b_POST:
2824
18.8k
      case AArch64_LD2Twov2d_POST:
2825
18.8k
      case AArch64_LD2Twov2s_POST:
2826
19.1k
      case AArch64_LD2Twov4h_POST:
2827
19.4k
      case AArch64_LD2Twov4s_POST:
2828
19.4k
      case AArch64_LD2Twov8b_POST:
2829
19.5k
      case AArch64_LD2Twov8h_POST:
2830
20.4k
      case AArch64_LD2i16_POST:
2831
20.8k
      case AArch64_LD2i32_POST:
2832
22.6k
      case AArch64_LD2i64_POST:
2833
23.6k
      case AArch64_LD2i8_POST:
2834
23.9k
      case AArch64_LD3Rv16b_POST:
2835
24.0k
      case AArch64_LD3Rv1d_POST:
2836
24.4k
      case AArch64_LD3Rv2d_POST:
2837
24.4k
      case AArch64_LD3Rv2s_POST:
2838
24.5k
      case AArch64_LD3Rv4h_POST:
2839
25.1k
      case AArch64_LD3Rv4s_POST:
2840
25.4k
      case AArch64_LD3Rv8b_POST:
2841
25.9k
      case AArch64_LD3Rv8h_POST:
2842
26.0k
      case AArch64_LD3Threev16b_POST:
2843
26.5k
      case AArch64_LD3Threev2d_POST:
2844
26.6k
      case AArch64_LD3Threev2s_POST:
2845
27.2k
      case AArch64_LD3Threev4h_POST:
2846
27.5k
      case AArch64_LD3Threev4s_POST:
2847
27.6k
      case AArch64_LD3Threev8b_POST:
2848
28.0k
      case AArch64_LD3Threev8h_POST:
2849
28.5k
      case AArch64_LD3i16_POST:
2850
29.2k
      case AArch64_LD3i32_POST:
2851
30.6k
      case AArch64_LD3i64_POST:
2852
31.3k
      case AArch64_LD3i8_POST:
2853
31.6k
      case AArch64_LD4Fourv16b_POST:
2854
31.6k
      case AArch64_LD4Fourv2d_POST:
2855
32.1k
      case AArch64_LD4Fourv2s_POST:
2856
32.7k
      case AArch64_LD4Fourv4h_POST:
2857
33.1k
      case AArch64_LD4Fourv4s_POST:
2858
33.4k
      case AArch64_LD4Fourv8b_POST:
2859
33.7k
      case AArch64_LD4Fourv8h_POST:
2860
34.0k
      case AArch64_LD4Rv16b_POST:
2861
34.0k
      case AArch64_LD4Rv1d_POST:
2862
34.2k
      case AArch64_LD4Rv2d_POST:
2863
35.2k
      case AArch64_LD4Rv2s_POST:
2864
35.3k
      case AArch64_LD4Rv4h_POST:
2865
35.6k
      case AArch64_LD4Rv4s_POST:
2866
36.2k
      case AArch64_LD4Rv8b_POST:
2867
36.8k
      case AArch64_LD4Rv8h_POST:
2868
37.3k
      case AArch64_LD4i16_POST:
2869
38.0k
      case AArch64_LD4i32_POST:
2870
38.4k
      case AArch64_LD4i64_POST:
2871
39.3k
      case AArch64_LD4i8_POST:
2872
39.4k
      case AArch64_LDRBBpost:
2873
39.5k
      case AArch64_LDRBpost:
2874
39.7k
      case AArch64_LDRDpost:
2875
39.9k
      case AArch64_LDRHHpost:
2876
40.2k
      case AArch64_LDRHpost:
2877
40.6k
      case AArch64_LDRQpost:
2878
40.8k
      case AArch64_LDPDpost:
2879
41.1k
      case AArch64_LDPQpost:
2880
41.4k
      case AArch64_LDPSWpost:
2881
41.6k
      case AArch64_LDPSpost:
2882
42.3k
      case AArch64_LDPWpost:
2883
42.4k
      case AArch64_LDPXpost:
2884
42.5k
      case AArch64_ST1Fourv16b_POST:
2885
42.9k
      case AArch64_ST1Fourv1d_POST:
2886
43.4k
      case AArch64_ST1Fourv2d_POST:
2887
43.5k
      case AArch64_ST1Fourv2s_POST:
2888
43.6k
      case AArch64_ST1Fourv4h_POST:
2889
43.7k
      case AArch64_ST1Fourv4s_POST:
2890
43.8k
      case AArch64_ST1Fourv8b_POST:
2891
44.9k
      case AArch64_ST1Fourv8h_POST:
2892
45.0k
      case AArch64_ST1Onev16b_POST:
2893
45.4k
      case AArch64_ST1Onev1d_POST:
2894
45.5k
      case AArch64_ST1Onev2d_POST:
2895
45.6k
      case AArch64_ST1Onev2s_POST:
2896
45.7k
      case AArch64_ST1Onev4h_POST:
2897
45.8k
      case AArch64_ST1Onev4s_POST:
2898
45.9k
      case AArch64_ST1Onev8b_POST:
2899
46.0k
      case AArch64_ST1Onev8h_POST:
2900
46.1k
      case AArch64_ST1Threev16b_POST:
2901
46.3k
      case AArch64_ST1Threev1d_POST:
2902
46.4k
      case AArch64_ST1Threev2d_POST:
2903
46.8k
      case AArch64_ST1Threev2s_POST:
2904
47.2k
      case AArch64_ST1Threev4h_POST:
2905
47.2k
      case AArch64_ST1Threev4s_POST:
2906
47.5k
      case AArch64_ST1Threev8b_POST:
2907
47.6k
      case AArch64_ST1Threev8h_POST:
2908
47.6k
      case AArch64_ST1Twov16b_POST:
2909
47.9k
      case AArch64_ST1Twov1d_POST:
2910
48.1k
      case AArch64_ST1Twov2d_POST:
2911
48.5k
      case AArch64_ST1Twov2s_POST:
2912
48.6k
      case AArch64_ST1Twov4h_POST:
2913
48.7k
      case AArch64_ST1Twov4s_POST:
2914
48.8k
      case AArch64_ST1Twov8b_POST:
2915
49.3k
      case AArch64_ST1Twov8h_POST:
2916
49.8k
      case AArch64_ST1i16_POST:
2917
50.2k
      case AArch64_ST1i32_POST:
2918
50.7k
      case AArch64_ST1i64_POST:
2919
51.3k
      case AArch64_ST1i8_POST:
2920
51.5k
      case AArch64_ST2GPostIndex:
2921
52.3k
      case AArch64_ST2Twov16b_POST:
2922
52.4k
      case AArch64_ST2Twov2d_POST:
2923
52.5k
      case AArch64_ST2Twov2s_POST:
2924
52.7k
      case AArch64_ST2Twov4h_POST:
2925
52.8k
      case AArch64_ST2Twov4s_POST:
2926
53.1k
      case AArch64_ST2Twov8b_POST:
2927
53.4k
      case AArch64_ST2Twov8h_POST:
2928
54.1k
      case AArch64_ST2i16_POST:
2929
54.3k
      case AArch64_ST2i32_POST:
2930
54.7k
      case AArch64_ST2i64_POST:
2931
55.2k
      case AArch64_ST2i8_POST:
2932
55.4k
      case AArch64_ST3Threev16b_POST:
2933
55.7k
      case AArch64_ST3Threev2d_POST:
2934
56.5k
      case AArch64_ST3Threev2s_POST:
2935
56.6k
      case AArch64_ST3Threev4h_POST:
2936
57.1k
      case AArch64_ST3Threev4s_POST:
2937
57.2k
      case AArch64_ST3Threev8b_POST:
2938
57.2k
      case AArch64_ST3Threev8h_POST:
2939
57.7k
      case AArch64_ST3i16_POST:
2940
58.4k
      case AArch64_ST3i32_POST:
2941
58.8k
      case AArch64_ST3i64_POST:
2942
59.2k
      case AArch64_ST3i8_POST:
2943
60.9k
      case AArch64_ST4Fourv16b_POST:
2944
61.1k
      case AArch64_ST4Fourv2d_POST:
2945
61.2k
      case AArch64_ST4Fourv2s_POST:
2946
61.3k
      case AArch64_ST4Fourv4h_POST:
2947
61.4k
      case AArch64_ST4Fourv4s_POST:
2948
61.5k
      case AArch64_ST4Fourv8b_POST:
2949
62.7k
      case AArch64_ST4Fourv8h_POST:
2950
63.1k
      case AArch64_ST4i16_POST:
2951
63.6k
      case AArch64_ST4i32_POST:
2952
64.0k
      case AArch64_ST4i64_POST:
2953
64.1k
      case AArch64_ST4i8_POST:
2954
64.6k
      case AArch64_STPDpost:
2955
64.9k
      case AArch64_STPQpost:
2956
65.5k
      case AArch64_STPSpost:
2957
66.3k
      case AArch64_STPWpost:
2958
66.9k
      case AArch64_STPXpost:
2959
66.9k
      case AArch64_STRBBpost:
2960
67.1k
      case AArch64_STRBpost:
2961
67.1k
      case AArch64_STRDpost:
2962
67.3k
      case AArch64_STRHHpost:
2963
67.4k
      case AArch64_STRHpost:
2964
67.6k
      case AArch64_STRQpost:
2965
67.7k
      case AArch64_STRSpost:
2966
67.8k
      case AArch64_STRWpost:
2967
67.9k
      case AArch64_STRXpost:
2968
68.1k
      case AArch64_STZ2GPostIndex:
2969
68.2k
      case AArch64_STZGPostIndex:
2970
68.2k
      case AArch64_STGPostIndex:
2971
68.2k
      case AArch64_STGPpost:
2972
68.4k
      case AArch64_LDRSBWpost:
2973
68.6k
      case AArch64_LDRSBXpost:
2974
68.7k
      case AArch64_LDRSHWpost:
2975
69.2k
      case AArch64_LDRSHXpost:
2976
69.4k
      case AArch64_LDRSWpost:
2977
69.5k
      case AArch64_LDRSpost:
2978
69.6k
      case AArch64_LDRWpost:
2979
69.9k
      case AArch64_LDRXpost:
2980
69.9k
        flat_insn->detail->arm64.writeback = true;
2981
69.9k
          flat_insn->detail->arm64.post_index = true;
2982
69.9k
        break;
2983
864
      case AArch64_LDRAAwriteback:
2984
1.01k
      case AArch64_LDRABwriteback:
2985
1.65k
      case AArch64_ST2GPreIndex:
2986
2.00k
      case AArch64_LDPDpre:
2987
2.27k
      case AArch64_LDPQpre:
2988
2.94k
      case AArch64_LDPSWpre:
2989
3.09k
      case AArch64_LDPSpre:
2990
3.21k
      case AArch64_LDPWpre:
2991
3.64k
      case AArch64_LDPXpre:
2992
4.06k
      case AArch64_LDRBBpre:
2993
4.44k
      case AArch64_LDRBpre:
2994
4.88k
      case AArch64_LDRDpre:
2995
4.99k
      case AArch64_LDRHHpre:
2996
5.39k
      case AArch64_LDRHpre:
2997
5.57k
      case AArch64_LDRQpre:
2998
5.81k
      case AArch64_LDRSBWpre:
2999
5.92k
      case AArch64_LDRSBXpre:
3000
6.16k
      case AArch64_LDRSHWpre:
3001
6.24k
      case AArch64_LDRSHXpre:
3002
6.33k
      case AArch64_LDRSWpre:
3003
6.55k
      case AArch64_LDRSpre:
3004
6.66k
      case AArch64_LDRWpre:
3005
6.81k
      case AArch64_LDRXpre:
3006
7.30k
      case AArch64_STGPreIndex:
3007
8.00k
      case AArch64_STPDpre:
3008
8.55k
      case AArch64_STPQpre:
3009
9.13k
      case AArch64_STPSpre:
3010
9.45k
      case AArch64_STPWpre:
3011
10.1k
      case AArch64_STPXpre:
3012
10.7k
      case AArch64_STRBBpre:
3013
11.0k
      case AArch64_STRBpre:
3014
11.3k
      case AArch64_STRDpre:
3015
11.7k
      case AArch64_STRHHpre:
3016
11.9k
      case AArch64_STRHpre:
3017
12.1k
      case AArch64_STRQpre:
3018
12.2k
      case AArch64_STRSpre:
3019
12.3k
      case AArch64_STRWpre:
3020
12.5k
      case AArch64_STRXpre:
3021
13.2k
      case AArch64_STZ2GPreIndex:
3022
13.4k
      case AArch64_STZGPreIndex:
3023
13.4k
      case AArch64_STGPpre:
3024
        flat_insn->detail->arm64.writeback = true;
3025
13.4k
        break;
3026
412k
    }
3027
412k
  }
3028
412k
}
3029
3030
#endif