Coverage Report

Created: 2026-05-30 06:22

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonenext/arch/RISCV/RISCVDisassembler.c
Line
Count
Source
1
/* Capstone Disassembly Engine, http://www.capstone-engine.org */
2
/* By Nguyen Anh Quynh <aquynh@gmail.com>, 2013-2022, */
3
/*    Rot127 <unisono@quyllur.org> 2022-2023 */
4
/* Automatically translated source file from LLVM. */
5
6
/* LLVM-commit: <commit> */
7
/* LLVM-tag: <tag> */
8
9
/* Only small edits allowed. */
10
/* For multiple similar edits, please create a Patch for the translator. */
11
12
/* Capstone's C++ file translator: */
13
/* https://github.com/capstone-engine/capstone/tree/next/suite/auto-sync */
14
15
//===-- RISCVDisassembler.cpp - Disassembler for RISC-V -------------------===//
16
//
17
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
18
// See https://llvm.org/LICENSE.txt for license information.
19
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
20
//
21
//===----------------------------------------------------------------------===//
22
//
23
// This file implements the RISCVDisassembler class.
24
//
25
//===----------------------------------------------------------------------===//
26
27
#include <stdio.h>
28
#include <string.h>
29
#include <stdlib.h>
30
#include <capstone/platform.h>
31
32
#include "../../MCInst.h"
33
#include "../../MathExtras.h"
34
#include "../../MCInstPrinter.h"
35
#include "../../MCDisassembler.h"
36
#include "../../MCFixedLenDisassembler.h"
37
#include "../../cs_priv.h"
38
#include "../../utils.h"
39
#include "RISCVDisassemblerExtension.h"
40
#include "RISCVBaseInfo.h"
41
#include "RISCVMapping.h"
42
43
#define GET_SUBTARGETINFO_ENUM
44
#include "RISCVGenSubtargetInfo.inc"
45
46
#define GET_REGINFO_ENUM
47
#include "RISCVGenRegisterInfo.inc"
48
49
#define GET_INSTRINFO_ENUM
50
#define GET_INSTRINFO_MC_DESC
51
#include "RISCVGenInstrInfo.inc"
52
53
6.71k
#define CONCAT(a, b) CONCAT_(a, b)
54
6.71k
#define CONCAT_(a, b) a##_##b
55
56
#define DEBUG_TYPE "riscv-disassembler"
57
58
DecodeStatus RISCV_getInstruction(MCInst *Instr, uint16_t *Size,
59
          const uint8_t *Bytes, size_t BytesLen,
60
          uint64_t Address, SStream *CStream);
61
void addSPOperands(MCInst *MI);
62
;
63
// end anonymous namespace
64
65
static DecodeStatus DecodeGPRRegisterClass(MCInst *Inst, uint32_t RegNo,
66
             uint64_t Address,
67
             const void *Decoder)
68
45.9k
{
69
45.9k
  bool IsRVE = RISCV_getFeatureBits(Inst->csh->mode, RISCV_FeatureRVE);
70
71
45.9k
  if (RegNo >= 32 || (IsRVE && RegNo >= 16))
72
0
    return MCDisassembler_Fail;
73
74
45.9k
  MCRegister Reg = RISCV_X0 + RegNo;
75
45.9k
  MCOperand_CreateReg0(Inst, (Reg));
76
45.9k
  return MCDisassembler_Success;
77
45.9k
}
78
79
static DecodeStatus DecodeGPRX1X5RegisterClass(MCInst *Inst, uint32_t RegNo,
80
                 uint64_t Address,
81
                 const void *Decoder)
82
0
{
83
0
  MCRegister Reg = RISCV_X0 + RegNo;
84
0
  if (Reg != RISCV_X1 && Reg != RISCV_X5)
85
0
    return MCDisassembler_Fail;
86
87
0
  MCOperand_CreateReg0(Inst, (Reg));
88
0
  return MCDisassembler_Success;
89
0
}
90
91
static DecodeStatus DecodeFPR16RegisterClass(MCInst *Inst, uint32_t RegNo,
92
               uint64_t Address,
93
               const void *Decoder)
94
4.52k
{
95
4.52k
  if (RegNo >= 32)
96
0
    return MCDisassembler_Fail;
97
98
4.52k
  MCRegister Reg = RISCV_F0_H + RegNo;
99
4.52k
  MCOperand_CreateReg0(Inst, (Reg));
100
4.52k
  return MCDisassembler_Success;
101
4.52k
}
102
103
static DecodeStatus DecodeFPR32RegisterClass(MCInst *Inst, uint32_t RegNo,
104
               uint64_t Address,
105
               const void *Decoder)
106
1.59k
{
107
1.59k
  if (RegNo >= 32)
108
0
    return MCDisassembler_Fail;
109
110
1.59k
  MCRegister Reg = RISCV_F0_F + RegNo;
111
1.59k
  MCOperand_CreateReg0(Inst, (Reg));
112
1.59k
  return MCDisassembler_Success;
113
1.59k
}
114
115
static DecodeStatus DecodeFPR32CRegisterClass(MCInst *Inst, uint32_t RegNo,
116
                uint64_t Address,
117
                const void *Decoder)
118
0
{
119
0
  if (RegNo >= 8) {
120
0
    return MCDisassembler_Fail;
121
0
  }
122
0
  MCRegister Reg = RISCV_F8_F + RegNo;
123
0
  MCOperand_CreateReg0(Inst, (Reg));
124
0
  return MCDisassembler_Success;
125
0
}
126
127
static DecodeStatus DecodeFPR64RegisterClass(MCInst *Inst, uint32_t RegNo,
128
               uint64_t Address,
129
               const void *Decoder)
130
0
{
131
0
  if (RegNo >= 32)
132
0
    return MCDisassembler_Fail;
133
134
0
  MCRegister Reg = RISCV_F0_D + RegNo;
135
0
  MCOperand_CreateReg0(Inst, (Reg));
136
0
  return MCDisassembler_Success;
137
0
}
138
139
static DecodeStatus DecodeFPR64CRegisterClass(MCInst *Inst, uint32_t RegNo,
140
                uint64_t Address,
141
                const void *Decoder)
142
0
{
143
0
  if (RegNo >= 8) {
144
0
    return MCDisassembler_Fail;
145
0
  }
146
0
  MCRegister Reg = RISCV_F8_D + RegNo;
147
0
  MCOperand_CreateReg0(Inst, (Reg));
148
0
  return MCDisassembler_Success;
149
0
}
150
151
static DecodeStatus DecodeGPRNoX0RegisterClass(MCInst *Inst, uint32_t RegNo,
152
                 uint64_t Address,
153
                 const void *Decoder)
154
18.6k
{
155
18.6k
  if (RegNo == 0) {
156
14
    return MCDisassembler_Fail;
157
14
  }
158
159
18.6k
  return DecodeGPRRegisterClass(Inst, RegNo, Address, Decoder);
160
18.6k
}
161
162
static DecodeStatus DecodeGPRNoX0X2RegisterClass(MCInst *Inst, uint64_t RegNo,
163
             uint32_t Address,
164
             const void *Decoder)
165
2.25k
{
166
2.25k
  if (RegNo == 2) {
167
0
    return MCDisassembler_Fail;
168
0
  }
169
170
2.25k
  return DecodeGPRNoX0RegisterClass(Inst, RegNo, Address, Decoder);
171
2.25k
}
172
173
static DecodeStatus DecodeGPRCRegisterClass(MCInst *Inst, uint32_t RegNo,
174
              uint64_t Address,
175
              const void *Decoder)
176
26.5k
{
177
26.5k
  if (RegNo >= 8)
178
0
    return MCDisassembler_Fail;
179
180
26.5k
  MCRegister Reg = RISCV_X8 + RegNo;
181
26.5k
  MCOperand_CreateReg0(Inst, (Reg));
182
26.5k
  return MCDisassembler_Success;
183
26.5k
}
184
185
static DecodeStatus DecodeGPRPairRegisterClass(MCInst *Inst, uint32_t RegNo,
186
                 uint64_t Address,
187
                 const void *Decoder)
188
118
{
189
118
  if (RegNo >= 32 || RegNo & 1)
190
2
    return MCDisassembler_Fail;
191
192
116
  MCRegister Reg = RISCV_X0 + RegNo;
193
116
  MCOperand_CreateReg0(Inst, (Reg));
194
116
  return MCDisassembler_Success;
195
118
}
196
197
static DecodeStatus DecodeSR07RegisterClass(MCInst *Inst, uint64_t RegNo,
198
              uint64_t Address,
199
              const void *Decoder)
200
0
{
201
0
  if (RegNo >= 8)
202
0
    return MCDisassembler_Fail;
203
204
0
  MCRegister Reg = (RegNo < 2) ? (RegNo + RISCV_X8) :
205
0
               (RegNo - 2 + RISCV_X18);
206
0
  MCOperand_CreateReg0(Inst, (Reg));
207
0
  return MCDisassembler_Success;
208
0
}
209
210
static DecodeStatus DecodeVRRegisterClass(MCInst *Inst, uint32_t RegNo,
211
            uint64_t Address, const void *Decoder)
212
6.33k
{
213
6.33k
  if (RegNo >= 32)
214
0
    return MCDisassembler_Fail;
215
216
6.33k
  MCRegister Reg = RISCV_V0 + RegNo;
217
6.33k
  MCOperand_CreateReg0(Inst, (Reg));
218
6.33k
  return MCDisassembler_Success;
219
6.33k
}
220
221
static DecodeStatus DecodeVRM2RegisterClass(MCInst *Inst, uint32_t RegNo,
222
              uint64_t Address,
223
              const void *Decoder)
224
285
{
225
285
  if (RegNo >= 32 || RegNo % 2)
226
2
    return MCDisassembler_Fail;
227
228
283
  MCRegister Reg = MCRegisterInfo_getMatchingSuperReg(
229
283
    Inst->MRI, RISCV_V0 + RegNo, RISCV_sub_vrm1_0,
230
283
    MCRegisterInfo_getRegClass(Inst->MRI, RISCV_VRM2RegClassID));
231
232
283
  MCOperand_CreateReg0(Inst, (Reg));
233
283
  return MCDisassembler_Success;
234
285
}
235
236
static DecodeStatus DecodeVRM4RegisterClass(MCInst *Inst, uint32_t RegNo,
237
              uint64_t Address,
238
              const void *Decoder)
239
117
{
240
117
  if (RegNo >= 32 || RegNo % 4)
241
3
    return MCDisassembler_Fail;
242
243
114
  MCRegister Reg = MCRegisterInfo_getMatchingSuperReg(
244
114
    Inst->MRI, RISCV_V0 + RegNo, RISCV_sub_vrm1_0,
245
114
    MCRegisterInfo_getRegClass(Inst->MRI, RISCV_VRM4RegClassID));
246
247
114
  MCOperand_CreateReg0(Inst, (Reg));
248
114
  return MCDisassembler_Success;
249
117
}
250
251
static DecodeStatus DecodeVRM8RegisterClass(MCInst *Inst, uint32_t RegNo,
252
              uint64_t Address,
253
              const void *Decoder)
254
320
{
255
320
  if (RegNo >= 32 || RegNo % 8)
256
4
    return MCDisassembler_Fail;
257
258
316
  MCRegister Reg = MCRegisterInfo_getMatchingSuperReg(
259
316
    Inst->MRI, RISCV_V0 + RegNo, RISCV_sub_vrm1_0,
260
316
    MCRegisterInfo_getRegClass(Inst->MRI, RISCV_VRM8RegClassID));
261
262
316
  MCOperand_CreateReg0(Inst, (Reg));
263
316
  return MCDisassembler_Success;
264
320
}
265
266
static DecodeStatus decodeVMaskReg(MCInst *Inst, uint64_t RegNo,
267
           uint64_t Address, const void *Decoder)
268
2.18k
{
269
2.18k
  if (RegNo > 2) {
270
0
    return MCDisassembler_Fail;
271
0
  }
272
2.18k
  MCRegister Reg = (RegNo == 0) ? RISCV_V0 : RISCV_NoRegister;
273
274
2.18k
  MCOperand_CreateReg0(Inst, (Reg));
275
2.18k
  return MCDisassembler_Success;
276
2.18k
}
277
278
#define DEFINE_decodeUImmOperand(N) \
279
  static DecodeStatus CONCAT(decodeUImmOperand, \
280
           N)(MCInst * Inst, uint32_t Imm, \
281
              int64_t Address, const void *Decoder) \
282
19.9k
  { \
283
19.9k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
19.9k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
19.9k
    return MCDisassembler_Success; \
286
19.9k
  }
RISCVDisassembler.c:decodeUImmOperand_8
Line
Count
Source
282
3.84k
  { \
283
3.84k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
3.84k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
3.84k
    return MCDisassembler_Success; \
286
3.84k
  }
RISCVDisassembler.c:decodeUImmOperand_9
Line
Count
Source
282
1.50k
  { \
283
1.50k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
1.50k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
1.50k
    return MCDisassembler_Success; \
286
1.50k
  }
RISCVDisassembler.c:decodeUImmOperand_7
Line
Count
Source
282
3.20k
  { \
283
3.20k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
3.20k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
3.20k
    return MCDisassembler_Success; \
286
3.20k
  }
RISCVDisassembler.c:decodeUImmOperand_2
Line
Count
Source
282
2.07k
  { \
283
2.07k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
2.07k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
2.07k
    return MCDisassembler_Success; \
286
2.07k
  }
RISCVDisassembler.c:decodeUImmOperand_4
Line
Count
Source
282
801
  { \
283
801
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
801
    MCOperand_CreateImm0(Inst, (Imm)); \
285
801
    return MCDisassembler_Success; \
286
801
  }
RISCVDisassembler.c:decodeUImmOperand_20
Line
Count
Source
282
583
  { \
283
583
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
583
    MCOperand_CreateImm0(Inst, (Imm)); \
285
583
    return MCDisassembler_Success; \
286
583
  }
RISCVDisassembler.c:decodeUImmOperand_5
Line
Count
Source
282
2.10k
  { \
283
2.10k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
2.10k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
2.10k
    return MCDisassembler_Success; \
286
2.10k
  }
RISCVDisassembler.c:decodeUImmOperand_6
Line
Count
Source
282
851
  { \
283
851
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
851
    MCOperand_CreateImm0(Inst, (Imm)); \
285
851
    return MCDisassembler_Success; \
286
851
  }
RISCVDisassembler.c:decodeUImmOperand_11
Line
Count
Source
282
685
  { \
283
685
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
685
    MCOperand_CreateImm0(Inst, (Imm)); \
285
685
    return MCDisassembler_Success; \
286
685
  }
RISCVDisassembler.c:decodeUImmOperand_10
Line
Count
Source
282
2.92k
  { \
283
2.92k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
2.92k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
2.92k
    return MCDisassembler_Success; \
286
2.92k
  }
RISCVDisassembler.c:decodeUImmOperand_12
Line
Count
Source
282
1.29k
  { \
283
1.29k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
1.29k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
1.29k
    return MCDisassembler_Success; \
286
1.29k
  }
RISCVDisassembler.c:decodeUImmOperand_3
Line
Count
Source
282
83
  { \
283
83
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
83
    MCOperand_CreateImm0(Inst, (Imm)); \
285
83
    return MCDisassembler_Success; \
286
83
  }
Unexecuted instantiation: RISCVDisassembler.c:decodeUImmOperand_1
287
DEFINE_decodeUImmOperand(6);
288
DEFINE_decodeUImmOperand(2);
289
DEFINE_decodeUImmOperand(8);
290
DEFINE_decodeUImmOperand(9);
291
DEFINE_decodeUImmOperand(7);
292
DEFINE_decodeUImmOperand(4);
293
DEFINE_decodeUImmOperand(20);
294
DEFINE_decodeUImmOperand(5);
295
DEFINE_decodeUImmOperand(11);
296
DEFINE_decodeUImmOperand(10);
297
DEFINE_decodeUImmOperand(12);
298
DEFINE_decodeUImmOperand(3);
299
DEFINE_decodeUImmOperand(1);
300
301
#define DEFINE_decodeUImmNonZeroOperand(N) \
302
  static DecodeStatus CONCAT(decodeUImmNonZeroOperand, \
303
           N)(MCInst * Inst, uint32_t Imm, \
304
              int64_t Address, const void *Decoder) \
305
2.74k
  { \
306
2.74k
    if (Imm == 0) \
307
2.74k
      return MCDisassembler_Fail; \
308
2.74k
    return CONCAT(decodeUImmOperand, N)(Inst, Imm, Address, \
309
2.73k
                Decoder); \
310
2.74k
  }
311
DEFINE_decodeUImmNonZeroOperand(10);
312
313
#define DEFINE_decodeSImmOperand(N) \
314
  static DecodeStatus CONCAT(decodeSImmOperand, \
315
           N)(MCInst * Inst, uint32_t Imm, \
316
              int64_t Address, const void *Decoder) \
317
10.1k
  { \
318
10.1k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
10.1k
\
320
10.1k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
10.1k
    return MCDisassembler_Success; \
322
10.1k
  }
RISCVDisassembler.c:decodeSImmOperand_6
Line
Count
Source
317
6.40k
  { \
318
6.40k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
6.40k
\
320
6.40k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
6.40k
    return MCDisassembler_Success; \
322
6.40k
  }
RISCVDisassembler.c:decodeSImmOperand_10
Line
Count
Source
317
346
  { \
318
346
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
346
\
320
346
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
346
    return MCDisassembler_Success; \
322
346
  }
RISCVDisassembler.c:decodeSImmOperand_12
Line
Count
Source
317
3.16k
  { \
318
3.16k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
3.16k
\
320
3.16k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
3.16k
    return MCDisassembler_Success; \
322
3.16k
  }
RISCVDisassembler.c:decodeSImmOperand_5
Line
Count
Source
317
277
  { \
318
277
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
277
\
320
277
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
277
    return MCDisassembler_Success; \
322
277
  }
323
DEFINE_decodeSImmOperand(6);
324
DEFINE_decodeSImmOperand(12);
325
DEFINE_decodeSImmOperand(5);
326
DEFINE_decodeSImmOperand(10);
327
328
#define DEFINE_decodeSImmNonZeroOperand(N) \
329
  static DecodeStatus CONCAT(decodeSImmNonZeroOperand, \
330
           N)(MCInst * Inst, uint32_t Imm, \
331
              int64_t Address, const void *Decoder) \
332
3.09k
  { \
333
3.09k
    if (Imm == 0) \
334
3.09k
      return MCDisassembler_Fail; \
335
3.09k
    return CONCAT(decodeSImmOperand, N)(Inst, Imm, Address, \
336
3.09k
                Decoder); \
337
3.09k
  }
RISCVDisassembler.c:decodeSImmNonZeroOperand_6
Line
Count
Source
332
2.74k
  { \
333
2.74k
    if (Imm == 0) \
334
2.74k
      return MCDisassembler_Fail; \
335
2.74k
    return CONCAT(decodeSImmOperand, N)(Inst, Imm, Address, \
336
2.74k
                Decoder); \
337
2.74k
  }
RISCVDisassembler.c:decodeSImmNonZeroOperand_10
Line
Count
Source
332
346
  { \
333
346
    if (Imm == 0) \
334
346
      return MCDisassembler_Fail; \
335
346
    return CONCAT(decodeSImmOperand, N)(Inst, Imm, Address, \
336
346
                Decoder); \
337
346
  }
338
DEFINE_decodeSImmNonZeroOperand(6);
339
DEFINE_decodeSImmNonZeroOperand(10);
340
341
#define DEFINE_decodeSImmOperandAndLsl1(N) \
342
  static DecodeStatus CONCAT(decodeSImmOperandAndLsl1, \
343
           N)(MCInst * Inst, uint32_t Imm, \
344
              int64_t Address, const void *Decoder) \
345
4.22k
  { \
346
4.22k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
4.22k
\
348
4.22k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
4.22k
    return MCDisassembler_Success; \
350
4.22k
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_12
Line
Count
Source
345
1.59k
  { \
346
1.59k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
1.59k
\
348
1.59k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
1.59k
    return MCDisassembler_Success; \
350
1.59k
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_9
Line
Count
Source
345
2.12k
  { \
346
2.12k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
2.12k
\
348
2.12k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
2.12k
    return MCDisassembler_Success; \
350
2.12k
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_13
Line
Count
Source
345
243
  { \
346
243
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
243
\
348
243
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
243
    return MCDisassembler_Success; \
350
243
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_21
Line
Count
Source
345
269
  { \
346
269
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
269
\
348
269
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
269
    return MCDisassembler_Success; \
350
269
  }
351
DEFINE_decodeSImmOperandAndLsl1(12);
352
DEFINE_decodeSImmOperandAndLsl1(9);
353
DEFINE_decodeSImmOperandAndLsl1(13);
354
DEFINE_decodeSImmOperandAndLsl1(21);
355
356
static DecodeStatus decodeCLUIImmOperand(MCInst *Inst, uint32_t Imm,
357
           int64_t Address, const void *Decoder)
358
2.25k
{
359
2.25k
  CS_ASSERT(isUIntN(6, Imm) && "Invalid immediate");
360
2.25k
  if (Imm > 31) {
361
1.36k
    Imm = (SignExtend64((Imm), 6) & 0xfffff);
362
1.36k
  }
363
2.25k
  MCOperand_CreateImm0(Inst, (Imm));
364
2.25k
  return MCDisassembler_Success;
365
2.25k
}
366
367
static DecodeStatus decodeFRMArg(MCInst *Inst, uint32_t Imm, int64_t Address,
368
         const void *Decoder)
369
2.03k
{
370
2.03k
  CS_ASSERT(isUIntN(3, Imm) && "Invalid immediate");
371
2.03k
  if (!RISCVFPRndMode_isValidRoundingMode(Imm))
372
7
    return MCDisassembler_Fail;
373
374
2.02k
  MCOperand_CreateImm0(Inst, (Imm));
375
2.02k
  return MCDisassembler_Success;
376
2.03k
}
377
378
static DecodeStatus decodeRVCInstrRdRs1ImmZero(MCInst *Inst, uint32_t Insn,
379
                 uint64_t Address,
380
                 const void *Decoder);
381
382
static DecodeStatus decodeRVCInstrRdSImm(MCInst *Inst, uint32_t Insn,
383
           uint64_t Address, const void *Decoder);
384
385
static DecodeStatus decodeRVCInstrRdRs1UImm(MCInst *Inst, uint32_t Insn,
386
              uint64_t Address,
387
              const void *Decoder);
388
389
static DecodeStatus decodeRVCInstrRdRs2(MCInst *Inst, uint32_t Insn,
390
          uint64_t Address, const void *Decoder);
391
392
static DecodeStatus decodeRVCInstrRdRs1Rs2(MCInst *Inst, uint32_t Insn,
393
             uint64_t Address,
394
             const void *Decoder);
395
396
static DecodeStatus decodeXTHeadMemPair(MCInst *Inst, uint32_t Insn,
397
          uint64_t Address, const void *Decoder);
398
399
static DecodeStatus decodeZcmpRlist(MCInst *Inst, unsigned Imm,
400
            uint64_t Address, const void *Decoder);
401
402
static DecodeStatus decodeRegReg(MCInst *Inst, uint32_t Insn, uint64_t Address,
403
         const void *Decoder);
404
405
static DecodeStatus decodeZcmpSpimm(MCInst *Inst, unsigned Imm,
406
            uint64_t Address, const void *Decoder);
407
408
static DecodeStatus decodeCSSPushPopchk(MCInst *Inst, uint32_t Insn,
409
          uint64_t Address, const void *Decoder);
410
411
static DecodeStatus decodeUImmLog2XLenOperand(MCInst *Inst, uint32_t Imm,
412
                int64_t Address,
413
                const void *Decoder)
414
2.54k
{
415
2.54k
  CS_ASSERT(isUIntN(6, Imm) && "Invalid immediate");
416
417
2.54k
  if (!RISCV_getFeatureBits(Inst->csh->mode, RISCV_Feature64Bit) &&
418
886
      !isUIntN(5, Imm))
419
6
    return MCDisassembler_Fail;
420
421
2.54k
  MCOperand_CreateImm0(Inst, (Imm));
422
2.54k
  return MCDisassembler_Success;
423
2.54k
}
424
425
static DecodeStatus decodeUImmLog2XLenNonZeroOperand(MCInst *Inst, uint32_t Imm,
426
                 int64_t Address,
427
                 const void *Decoder)
428
2.25k
{
429
2.25k
  if (Imm == 0)
430
0
    return MCDisassembler_Fail;
431
2.25k
  return decodeUImmLog2XLenOperand(Inst, Imm, Address, Decoder);
432
2.25k
}
433
434
#include "RISCVGenDisassemblerTables.inc"
435
436
static DecodeStatus decodeRVCInstrRdRs1ImmZero(MCInst *Inst, uint32_t Insn,
437
                 uint64_t Address,
438
                 const void *Decoder)
439
265
{
440
265
  uint32_t Rd = fieldFromInstruction_4(Insn, 7, 5);
441
265
  DecodeStatus Result =
442
265
    DecodeGPRNoX0RegisterClass(Inst, Rd, Address, Decoder);
443
265
  (void)Result;
444
265
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid register");
445
265
  MCInst_addOperand2(Inst, (MCInst_getOperand(Inst, (0))));
446
265
  MCOperand_CreateImm0(Inst, (0));
447
265
  return MCDisassembler_Success;
448
265
}
449
450
static DecodeStatus decodeCSSPushPopchk(MCInst *Inst, uint32_t Insn,
451
          uint64_t Address, const void *Decoder)
452
0
{
453
0
  uint32_t Rs1 = fieldFromInstruction_4(Insn, 7, 5);
454
0
  DecodeStatus Result =
455
0
    DecodeGPRX1X5RegisterClass(Inst, Rs1, Address, Decoder);
456
0
  (void)Result;
457
0
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid register");
458
0
  return MCDisassembler_Success;
459
0
}
460
461
static DecodeStatus decodeRVCInstrRdSImm(MCInst *Inst, uint32_t Insn,
462
           uint64_t Address, const void *Decoder)
463
232
{
464
232
  MCOperand_CreateReg0(Inst, (RISCV_X0));
465
232
  uint32_t SImm6 = fieldFromInstruction_4(Insn, 12, 1) << 5 |
466
232
       fieldFromInstruction_4(Insn, 2, 5);
467
232
  DecodeStatus Result =
468
232
    CONCAT(decodeSImmOperand, 6)(Inst, SImm6, Address, Decoder);
469
232
  (void)Result;
470
232
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid immediate");
471
232
  return MCDisassembler_Success;
472
232
}
473
474
static DecodeStatus decodeRVCInstrRdRs1UImm(MCInst *Inst, uint32_t Insn,
475
              uint64_t Address,
476
              const void *Decoder)
477
653
{
478
653
  MCOperand_CreateReg0(Inst, (RISCV_X0));
479
653
  MCInst_addOperand2(Inst, (MCInst_getOperand(Inst, (0))));
480
653
  uint32_t UImm6 = fieldFromInstruction_4(Insn, 12, 1) << 5 |
481
653
       fieldFromInstruction_4(Insn, 2, 5);
482
653
  DecodeStatus Result =
483
653
    CONCAT(decodeUImmOperand, 6)(Inst, UImm6, Address, Decoder);
484
653
  (void)Result;
485
653
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid immediate");
486
653
  return MCDisassembler_Success;
487
653
}
488
489
static DecodeStatus decodeRVCInstrRdRs2(MCInst *Inst, uint32_t Insn,
490
          uint64_t Address, const void *Decoder)
491
229
{
492
229
  uint32_t Rd = fieldFromInstruction_4(Insn, 7, 5);
493
229
  uint32_t Rs2 = fieldFromInstruction_4(Insn, 2, 5);
494
229
  DecodeGPRRegisterClass(Inst, Rd, Address, Decoder);
495
229
  DecodeGPRRegisterClass(Inst, Rs2, Address, Decoder);
496
229
  return MCDisassembler_Success;
497
229
}
498
499
static DecodeStatus decodeRVCInstrRdRs1Rs2(MCInst *Inst, uint32_t Insn,
500
             uint64_t Address,
501
             const void *Decoder)
502
66
{
503
66
  uint32_t Rd = fieldFromInstruction_4(Insn, 7, 5);
504
66
  uint32_t Rs2 = fieldFromInstruction_4(Insn, 2, 5);
505
66
  DecodeGPRRegisterClass(Inst, Rd, Address, Decoder);
506
66
  MCInst_addOperand2(Inst, (MCInst_getOperand(Inst, (0))));
507
66
  DecodeGPRRegisterClass(Inst, Rs2, Address, Decoder);
508
66
  return MCDisassembler_Success;
509
66
}
510
511
static DecodeStatus decodeXTHeadMemPair(MCInst *Inst, uint32_t Insn,
512
          uint64_t Address, const void *Decoder)
513
0
{
514
0
  uint32_t Rd1 = fieldFromInstruction_4(Insn, 7, 5);
515
0
  uint32_t Rs1 = fieldFromInstruction_4(Insn, 15, 5);
516
0
  uint32_t Rd2 = fieldFromInstruction_4(Insn, 20, 5);
517
0
  uint32_t UImm2 = fieldFromInstruction_4(Insn, 25, 2);
518
0
  DecodeGPRRegisterClass(Inst, Rd1, Address, Decoder);
519
0
  DecodeGPRRegisterClass(Inst, Rd2, Address, Decoder);
520
0
  DecodeGPRRegisterClass(Inst, Rs1, Address, Decoder);
521
0
  DecodeStatus Result =
522
0
    CONCAT(decodeUImmOperand, 2)(Inst, UImm2, Address, Decoder);
523
0
  (void)Result;
524
0
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid immediate");
525
526
  // Disassemble the final operand which is implicit.
527
0
  unsigned Opcode = MCInst_getOpcode(Inst);
528
0
  bool IsWordOp = (Opcode == RISCV_TH_LWD || Opcode == RISCV_TH_LWUD ||
529
0
       Opcode == RISCV_TH_SWD);
530
0
  if (IsWordOp)
531
0
    MCOperand_CreateImm0(Inst, (3));
532
0
  else
533
0
    MCOperand_CreateImm0(Inst, (4));
534
535
0
  return MCDisassembler_Success;
536
0
}
537
538
static DecodeStatus decodeZcmpRlist(MCInst *Inst, unsigned Imm,
539
            uint64_t Address, const void *Decoder)
540
0
{
541
0
  if (Imm <= 3)
542
0
    return MCDisassembler_Fail;
543
0
  MCOperand_CreateImm0(Inst, (Imm));
544
0
  return MCDisassembler_Success;
545
0
}
546
547
static DecodeStatus decodeRegReg(MCInst *Inst, uint32_t Insn, uint64_t Address,
548
         const void *Decoder)
549
105
{
550
105
  uint32_t Rs1 = fieldFromInstruction_4(Insn, 0, 5);
551
105
  uint32_t Rs2 = fieldFromInstruction_4(Insn, 5, 5);
552
105
  DecodeGPRRegisterClass(Inst, Rs1, Address, Decoder);
553
105
  DecodeGPRRegisterClass(Inst, Rs2, Address, Decoder);
554
105
  return MCDisassembler_Success;
555
105
}
556
557
static DecodeStatus decodeZcmpSpimm(MCInst *Inst, unsigned Imm,
558
            uint64_t Address, const void *Decoder)
559
0
{
560
0
  MCOperand_CreateImm0(Inst, (Imm));
561
0
  return MCDisassembler_Success;
562
0
}
563
564
// Add implied SP operand for C.*SP compressed instructions. The SP operand
565
// isn't explicitly encoded in the instruction.
566
void addSPOperands(MCInst *MI)
567
37.6k
{
568
37.6k
  const MCInstrDesc *MCID = MCInstrDesc_get(MCInst_getOpcode(MI),
569
37.6k
              RISCVDescs.Insts,
570
37.6k
              ARR_SIZE(RISCVDescs.Insts));
571
37.6k
  MCOperand SPReg;
572
37.6k
  SPReg.MachineOperandType = kRegister;
573
37.6k
  SPReg.Kind = kRegister;
574
37.6k
  SPReg.RegVal = RISCV_X2;
575
122k
  for (unsigned i = 0; i < MCID->NumOperands; i++)
576
84.6k
    if (MCID->OpInfo[i].RegClass == RISCV_SPRegClassID)
577
6.15k
      MCInst_insert0(MI, i, &SPReg);
578
37.6k
}
579
580
DecodeStatus RISCV_getInstruction(MCInst *MI, uint16_t *Size,
581
          const uint8_t *Bytes, size_t BytesLen,
582
          uint64_t Address, SStream *CS)
583
55.9k
{
584
  // TODO: This will need modification when supporting instruction set
585
  // extensions with instructions > 32-bits (up to 176 bits wide).
586
55.9k
  uint32_t Insn;
587
55.9k
  DecodeStatus Result;
588
589
55.9k
#define TRY_TO_DECODE_WITH_ADDITIONAL_OPERATION( \
590
55.9k
  width, FEATURE_CHECKS, DECODER_TABLE, DESC, ADDITIONAL_OPERATION) \
591
675k
  do { \
592
729k
    if (FEATURE_CHECKS) { \
593
174k
      Result = decodeInstruction_##width( \
594
174k
        DECODER_TABLE, MI, Insn, Address, NULL); \
595
174k
      if (Result != MCDisassembler_Fail) { \
596
55.1k
        ADDITIONAL_OPERATION; \
597
55.1k
        return Result; \
598
55.1k
      } \
599
174k
    } \
600
675k
  } while (false)
601
55.9k
#define TRY_TO_DECODE_AND_ADD_SP(width, FEATURE_CHECKS, DECODER_TABLE, DESC) \
602
74.8k
  TRY_TO_DECODE_WITH_ADDITIONAL_OPERATION( \
603
74.8k
    width, FEATURE_CHECKS, DECODER_TABLE, DESC, addSPOperands(MI))
604
55.9k
#define TRY_TO_DECODE(width, FEATURE_CHECKS, DECODER_TABLE, DESC) \
605
601k
  TRY_TO_DECODE_WITH_ADDITIONAL_OPERATION( \
606
601k
    width, FEATURE_CHECKS, DECODER_TABLE, DESC, (void)NULL)
607
55.9k
#define TRY_TO_DECODE_FEATURE(width, FEATURE, DECODER_TABLE, DESC) \
608
549k
  TRY_TO_DECODE(width, RISCV_getFeatureBits(MI->csh->mode, FEATURE), \
609
549k
          DECODER_TABLE, DESC)
610
611
  // It's a 32 bit instruction if bit 0 and 1 are 1.
612
55.9k
  if ((Bytes[0] & 0x3) == 0x3) {
613
17.8k
    if (BytesLen < 4) {
614
117
      *Size = 0;
615
117
      return MCDisassembler_Fail;
616
117
    }
617
17.7k
    *Size = 4;
618
619
17.7k
    Insn = readBytes32(MI, Bytes);
620
621
17.7k
    TRY_TO_DECODE(4,
622
17.7k
            RISCV_getFeatureBits(MI->csh->mode,
623
17.7k
               RISCV_FeatureStdExtZdinx) &&
624
17.7k
              !RISCV_getFeatureBits(MI->csh->mode,
625
17.7k
                  RISCV_Feature64Bit),
626
17.7k
            DecoderTableRV32Zdinx32,
627
17.7k
            "RV32Zdinx table (Double in Integer and rv32)");
628
17.7k
    TRY_TO_DECODE(4,
629
17.7k
            RISCV_getFeatureBits(MI->csh->mode,
630
17.7k
               RISCV_FeatureStdExtZacas) &&
631
17.7k
              !RISCV_getFeatureBits(MI->csh->mode,
632
17.7k
                  RISCV_Feature64Bit),
633
17.7k
            DecoderTableRV32Zacas32,
634
17.7k
            "RV32Zacas table (Compare-And-Swap and rv32)");
635
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureStdExtZfinx,
636
17.7k
              DecoderTableRVZfinx32,
637
17.7k
              "RVZfinx table (Float in Integer)");
638
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXVentanaCondOps,
639
17.7k
              DecoderTableXVentana32,
640
17.7k
              "Ventana custom opcode table");
641
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadBa,
642
17.7k
              DecoderTableXTHeadBa32,
643
17.7k
              "XTHeadBa custom opcode table");
644
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadBb,
645
17.7k
              DecoderTableXTHeadBb32,
646
17.7k
              "XTHeadBb custom opcode table");
647
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadBs,
648
17.7k
              DecoderTableXTHeadBs32,
649
17.7k
              "XTHeadBs custom opcode table");
650
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadCondMov,
651
17.7k
              DecoderTableXTHeadCondMov32,
652
17.7k
              "XTHeadCondMov custom opcode table");
653
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadCmo,
654
17.7k
              DecoderTableXTHeadCmo32,
655
17.7k
              "XTHeadCmo custom opcode table");
656
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadFMemIdx,
657
17.7k
              DecoderTableXTHeadFMemIdx32,
658
17.7k
              "XTHeadFMemIdx custom opcode table");
659
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadMac,
660
17.7k
              DecoderTableXTHeadMac32,
661
17.7k
              "XTHeadMac custom opcode table");
662
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadMemIdx,
663
17.7k
              DecoderTableXTHeadMemIdx32,
664
17.7k
              "XTHeadMemIdx custom opcode table");
665
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadMemPair,
666
17.7k
              DecoderTableXTHeadMemPair32,
667
17.7k
              "XTHeadMemPair custom opcode table");
668
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadSync,
669
17.7k
              DecoderTableXTHeadSync32,
670
17.7k
              "XTHeadSync custom opcode table");
671
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadVdot,
672
17.7k
              DecoderTableXTHeadVdot32,
673
17.7k
              "XTHeadVdot custom opcode table");
674
17.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXSfvcp,
675
17.7k
              DecoderTableXSfvcp32,
676
17.7k
              "SiFive VCIX custom opcode table");
677
17.7k
    TRY_TO_DECODE_FEATURE(
678
17.7k
      4, RISCV_FeatureVendorXSfvqmaccdod,
679
17.7k
      DecoderTableXSfvqmaccdod32,
680
17.7k
      "SiFive Matrix Multiplication (2x8 and 8x2) Instruction opcode table");
681
17.7k
    TRY_TO_DECODE_FEATURE(
682
17.7k
      4, RISCV_FeatureVendorXSfvqmaccqoq,
683
17.7k
      DecoderTableXSfvqmaccqoq32,
684
17.7k
      "SiFive Matrix Multiplication (4x8 and 8x4) Instruction opcode table");
685
17.7k
    TRY_TO_DECODE_FEATURE(
686
17.7k
      4, RISCV_FeatureVendorXSfvfwmaccqqq,
687
17.7k
      DecoderTableXSfvfwmaccqqq32,
688
17.7k
      "SiFive Matrix Multiplication Instruction opcode table");
689
17.7k
    TRY_TO_DECODE_FEATURE(
690
17.7k
      4, RISCV_FeatureVendorXSfvfnrclipxfqf,
691
17.7k
      DecoderTableXSfvfnrclipxfqf32,
692
17.7k
      "SiFive FP32-to-int8 Ranged Clip Instructions opcode table");
693
17.7k
    TRY_TO_DECODE_FEATURE(
694
17.6k
      4, RISCV_FeatureVendorXCVbitmanip,
695
17.6k
      DecoderTableXCVbitmanip32,
696
17.6k
      "CORE-V Bit Manipulation custom opcode table");
697
17.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCVelw,
698
17.6k
              DecoderTableXCVelw32,
699
17.6k
              "CORE-V Event load custom opcode table");
700
17.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCVmac,
701
17.1k
              DecoderTableXCVmac32,
702
17.1k
              "CORE-V MAC custom opcode table");
703
17.1k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCVmem,
704
16.5k
              DecoderTableXCVmem32,
705
16.5k
              "CORE-V MEM custom opcode table");
706
16.5k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCValu,
707
16.4k
              DecoderTableXCValu32,
708
16.4k
              "CORE-V ALU custom opcode table");
709
16.4k
    TRY_TO_DECODE_FEATURE(
710
16.0k
      4, RISCV_FeatureVendorXCVsimd, DecoderTableXCVsimd32,
711
16.0k
      "CORE-V SIMD extensions custom opcode table");
712
16.0k
    TRY_TO_DECODE_FEATURE(
713
15.9k
      4, RISCV_FeatureVendorXCVbi, DecoderTableXCVbi32,
714
15.9k
      "CORE-V Immediate Branching custom opcode table");
715
15.9k
    TRY_TO_DECODE(4, true, DecoderTable32, "RISCV32 table");
716
717
322
    return MCDisassembler_Fail;
718
15.9k
  }
719
720
38.0k
  if (BytesLen < 2) {
721
146
    *Size = 0;
722
146
    return MCDisassembler_Fail;
723
146
  }
724
37.9k
  *Size = 2;
725
726
37.9k
  Insn = readBytes16(MI, Bytes);
727
37.9k
  TRY_TO_DECODE_AND_ADD_SP(
728
36.8k
    2, !RISCV_getFeatureBits(MI->csh->mode, RISCV_Feature64Bit),
729
36.8k
    DecoderTableRISCV32Only_16,
730
36.8k
    "RISCV32Only_16 table (16-bit Instruction)");
731
36.8k
  TRY_TO_DECODE_FEATURE(2, RISCV_FeatureStdExtZicfiss,
732
36.8k
            DecoderTableZicfiss16,
733
36.8k
            "RVZicfiss table (Shadow Stack)");
734
36.8k
  TRY_TO_DECODE_FEATURE(2, RISCV_FeatureStdExtZcmt, DecoderTableRVZcmt16,
735
36.8k
            "Zcmt table (16-bit Table Jump Instructions)");
736
36.8k
  TRY_TO_DECODE_FEATURE(
737
36.8k
    2, RISCV_FeatureStdExtZcmp, DecoderTableRVZcmp16,
738
36.8k
    "Zcmp table (16-bit Push/Pop & Double Move Instructions)");
739
36.8k
  TRY_TO_DECODE_AND_ADD_SP(2, true, DecoderTable16,
740
239
         "RISCV_C table (16-bit Instruction)");
741
742
239
  return MCDisassembler_Fail;
743
36.8k
}
744
745
bool RISCV_LLVM_getInstruction(csh handle, const uint8_t *Bytes, size_t ByteLen,
746
             MCInst *MI, uint16_t *Size, uint64_t Address,
747
             void *Info)
748
55.9k
{
749
55.9k
  RISCV_init_cs_detail(MI);
750
55.9k
  MI->MRI = (MCRegisterInfo *)Info;
751
  return RISCV_getInstruction(MI, Size, Bytes, ByteLen, Address, NULL) !=
752
55.9k
         MCDisassembler_Fail;
753
55.9k
}