Coverage Report

Created: 2026-06-15 06:41

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
4.59k
#define CONCAT(a, b) CONCAT_(a, b)
54
4.59k
#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
32.9k
{
69
32.9k
  bool IsRVE = RISCV_getFeatureBits(Inst->csh->mode, RISCV_FeatureRVE);
70
71
32.9k
  if (RegNo >= 32 || (IsRVE && RegNo >= 16))
72
0
    return MCDisassembler_Fail;
73
74
32.9k
  MCRegister Reg = RISCV_X0 + RegNo;
75
32.9k
  MCOperand_CreateReg0(Inst, (Reg));
76
32.9k
  return MCDisassembler_Success;
77
32.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
3.08k
{
95
3.08k
  if (RegNo >= 32)
96
0
    return MCDisassembler_Fail;
97
98
3.08k
  MCRegister Reg = RISCV_F0_H + RegNo;
99
3.08k
  MCOperand_CreateReg0(Inst, (Reg));
100
3.08k
  return MCDisassembler_Success;
101
3.08k
}
102
103
static DecodeStatus DecodeFPR32RegisterClass(MCInst *Inst, uint32_t RegNo,
104
               uint64_t Address,
105
               const void *Decoder)
106
36.3k
{
107
36.3k
  if (RegNo >= 32)
108
0
    return MCDisassembler_Fail;
109
110
36.3k
  MCRegister Reg = RISCV_F0_F + RegNo;
111
36.3k
  MCOperand_CreateReg0(Inst, (Reg));
112
36.3k
  return MCDisassembler_Success;
113
36.3k
}
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
29.7k
{
131
29.7k
  if (RegNo >= 32)
132
0
    return MCDisassembler_Fail;
133
134
29.7k
  MCRegister Reg = RISCV_F0_D + RegNo;
135
29.7k
  MCOperand_CreateReg0(Inst, (Reg));
136
29.7k
  return MCDisassembler_Success;
137
29.7k
}
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
12.7k
{
155
12.7k
  if (RegNo == 0) {
156
9
    return MCDisassembler_Fail;
157
9
  }
158
159
12.6k
  return DecodeGPRRegisterClass(Inst, RegNo, Address, Decoder);
160
12.7k
}
161
162
static DecodeStatus DecodeGPRNoX0X2RegisterClass(MCInst *Inst, uint64_t RegNo,
163
             uint32_t Address,
164
             const void *Decoder)
165
1.24k
{
166
1.24k
  if (RegNo == 2) {
167
0
    return MCDisassembler_Fail;
168
0
  }
169
170
1.24k
  return DecodeGPRNoX0RegisterClass(Inst, RegNo, Address, Decoder);
171
1.24k
}
172
173
static DecodeStatus DecodeGPRCRegisterClass(MCInst *Inst, uint32_t RegNo,
174
              uint64_t Address,
175
              const void *Decoder)
176
17.5k
{
177
17.5k
  if (RegNo >= 8)
178
0
    return MCDisassembler_Fail;
179
180
17.5k
  MCRegister Reg = RISCV_X8 + RegNo;
181
17.5k
  MCOperand_CreateReg0(Inst, (Reg));
182
17.5k
  return MCDisassembler_Success;
183
17.5k
}
184
185
static DecodeStatus DecodeGPRPairRegisterClass(MCInst *Inst, uint32_t RegNo,
186
                 uint64_t Address,
187
                 const void *Decoder)
188
691
{
189
691
  if (RegNo >= 32 || RegNo & 1)
190
3
    return MCDisassembler_Fail;
191
192
688
  MCRegister Reg = RISCV_X0 + RegNo;
193
688
  MCOperand_CreateReg0(Inst, (Reg));
194
688
  return MCDisassembler_Success;
195
691
}
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.00k
{
213
6.00k
  if (RegNo >= 32)
214
0
    return MCDisassembler_Fail;
215
216
6.00k
  MCRegister Reg = RISCV_V0 + RegNo;
217
6.00k
  MCOperand_CreateReg0(Inst, (Reg));
218
6.00k
  return MCDisassembler_Success;
219
6.00k
}
220
221
static DecodeStatus DecodeVRM2RegisterClass(MCInst *Inst, uint32_t RegNo,
222
              uint64_t Address,
223
              const void *Decoder)
224
267
{
225
267
  if (RegNo >= 32 || RegNo % 2)
226
2
    return MCDisassembler_Fail;
227
228
265
  MCRegister Reg = MCRegisterInfo_getMatchingSuperReg(
229
265
    Inst->MRI, RISCV_V0 + RegNo, RISCV_sub_vrm1_0,
230
265
    MCRegisterInfo_getRegClass(Inst->MRI, RISCV_VRM2RegClassID));
231
232
265
  MCOperand_CreateReg0(Inst, (Reg));
233
265
  return MCDisassembler_Success;
234
267
}
235
236
static DecodeStatus DecodeVRM4RegisterClass(MCInst *Inst, uint32_t RegNo,
237
              uint64_t Address,
238
              const void *Decoder)
239
82
{
240
82
  if (RegNo >= 32 || RegNo % 4)
241
3
    return MCDisassembler_Fail;
242
243
79
  MCRegister Reg = MCRegisterInfo_getMatchingSuperReg(
244
79
    Inst->MRI, RISCV_V0 + RegNo, RISCV_sub_vrm1_0,
245
79
    MCRegisterInfo_getRegClass(Inst->MRI, RISCV_VRM4RegClassID));
246
247
79
  MCOperand_CreateReg0(Inst, (Reg));
248
79
  return MCDisassembler_Success;
249
82
}
250
251
static DecodeStatus DecodeVRM8RegisterClass(MCInst *Inst, uint32_t RegNo,
252
              uint64_t Address,
253
              const void *Decoder)
254
316
{
255
316
  if (RegNo >= 32 || RegNo % 8)
256
3
    return MCDisassembler_Fail;
257
258
313
  MCRegister Reg = MCRegisterInfo_getMatchingSuperReg(
259
313
    Inst->MRI, RISCV_V0 + RegNo, RISCV_sub_vrm1_0,
260
313
    MCRegisterInfo_getRegClass(Inst->MRI, RISCV_VRM8RegClassID));
261
262
313
  MCOperand_CreateReg0(Inst, (Reg));
263
313
  return MCDisassembler_Success;
264
316
}
265
266
static DecodeStatus decodeVMaskReg(MCInst *Inst, uint64_t RegNo,
267
           uint64_t Address, const void *Decoder)
268
2.29k
{
269
2.29k
  if (RegNo > 2) {
270
0
    return MCDisassembler_Fail;
271
0
  }
272
2.29k
  MCRegister Reg = (RegNo == 0) ? RISCV_V0 : RISCV_NoRegister;
273
274
2.29k
  MCOperand_CreateReg0(Inst, (Reg));
275
2.29k
  return MCDisassembler_Success;
276
2.29k
}
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
12.9k
  { \
283
12.9k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
12.9k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
12.9k
    return MCDisassembler_Success; \
286
12.9k
  }
RISCVDisassembler.c:decodeUImmOperand_8
Line
Count
Source
282
2.45k
  { \
283
2.45k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
2.45k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
2.45k
    return MCDisassembler_Success; \
286
2.45k
  }
RISCVDisassembler.c:decodeUImmOperand_9
Line
Count
Source
282
1.01k
  { \
283
1.01k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
1.01k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
1.01k
    return MCDisassembler_Success; \
286
1.01k
  }
RISCVDisassembler.c:decodeUImmOperand_7
Line
Count
Source
282
2.15k
  { \
283
2.15k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
2.15k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
2.15k
    return MCDisassembler_Success; \
286
2.15k
  }
RISCVDisassembler.c:decodeUImmOperand_2
Line
Count
Source
282
1.05k
  { \
283
1.05k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
1.05k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
1.05k
    return MCDisassembler_Success; \
286
1.05k
  }
RISCVDisassembler.c:decodeUImmOperand_4
Line
Count
Source
282
464
  { \
283
464
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
464
    MCOperand_CreateImm0(Inst, (Imm)); \
285
464
    return MCDisassembler_Success; \
286
464
  }
RISCVDisassembler.c:decodeUImmOperand_20
Line
Count
Source
282
398
  { \
283
398
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
398
    MCOperand_CreateImm0(Inst, (Imm)); \
285
398
    return MCDisassembler_Success; \
286
398
  }
RISCVDisassembler.c:decodeUImmOperand_5
Line
Count
Source
282
1.46k
  { \
283
1.46k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
1.46k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
1.46k
    return MCDisassembler_Success; \
286
1.46k
  }
RISCVDisassembler.c:decodeUImmOperand_6
Line
Count
Source
282
242
  { \
283
242
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
242
    MCOperand_CreateImm0(Inst, (Imm)); \
285
242
    return MCDisassembler_Success; \
286
242
  }
RISCVDisassembler.c:decodeUImmOperand_11
Line
Count
Source
282
497
  { \
283
497
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
497
    MCOperand_CreateImm0(Inst, (Imm)); \
285
497
    return MCDisassembler_Success; \
286
497
  }
RISCVDisassembler.c:decodeUImmOperand_10
Line
Count
Source
282
2.12k
  { \
283
2.12k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
2.12k
    MCOperand_CreateImm0(Inst, (Imm)); \
285
2.12k
    return MCDisassembler_Success; \
286
2.12k
  }
RISCVDisassembler.c:decodeUImmOperand_12
Line
Count
Source
282
986
  { \
283
986
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
986
    MCOperand_CreateImm0(Inst, (Imm)); \
285
986
    return MCDisassembler_Success; \
286
986
  }
RISCVDisassembler.c:decodeUImmOperand_3
Line
Count
Source
282
57
  { \
283
57
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
284
57
    MCOperand_CreateImm0(Inst, (Imm)); \
285
57
    return MCDisassembler_Success; \
286
57
  }
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.00k
  { \
306
2.00k
    if (Imm == 0) \
307
2.00k
      return MCDisassembler_Fail; \
308
2.00k
    return CONCAT(decodeUImmOperand, N)(Inst, Imm, Address, \
309
1.99k
                Decoder); \
310
2.00k
  }
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
6.79k
  { \
318
6.79k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
6.79k
\
320
6.79k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
6.79k
    return MCDisassembler_Success; \
322
6.79k
  }
RISCVDisassembler.c:decodeSImmOperand_6
Line
Count
Source
317
4.35k
  { \
318
4.35k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
4.35k
\
320
4.35k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
4.35k
    return MCDisassembler_Success; \
322
4.35k
  }
RISCVDisassembler.c:decodeSImmOperand_10
Line
Count
Source
317
191
  { \
318
191
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
191
\
320
191
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
191
    return MCDisassembler_Success; \
322
191
  }
RISCVDisassembler.c:decodeSImmOperand_12
Line
Count
Source
317
1.93k
  { \
318
1.93k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
1.93k
\
320
1.93k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
1.93k
    return MCDisassembler_Success; \
322
1.93k
  }
RISCVDisassembler.c:decodeSImmOperand_5
Line
Count
Source
317
314
  { \
318
314
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
319
314
\
320
314
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm), N))); \
321
314
    return MCDisassembler_Success; \
322
314
  }
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
2.26k
  { \
333
2.26k
    if (Imm == 0) \
334
2.26k
      return MCDisassembler_Fail; \
335
2.26k
    return CONCAT(decodeSImmOperand, N)(Inst, Imm, Address, \
336
2.26k
                Decoder); \
337
2.26k
  }
RISCVDisassembler.c:decodeSImmNonZeroOperand_6
Line
Count
Source
332
2.07k
  { \
333
2.07k
    if (Imm == 0) \
334
2.07k
      return MCDisassembler_Fail; \
335
2.07k
    return CONCAT(decodeSImmOperand, N)(Inst, Imm, Address, \
336
2.07k
                Decoder); \
337
2.07k
  }
RISCVDisassembler.c:decodeSImmNonZeroOperand_10
Line
Count
Source
332
192
  { \
333
192
    if (Imm == 0) \
334
192
      return MCDisassembler_Fail; \
335
192
    return CONCAT(decodeSImmOperand, N)(Inst, Imm, Address, \
336
191
                Decoder); \
337
192
  }
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
2.80k
  { \
346
2.80k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
2.80k
\
348
2.80k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
2.80k
    return MCDisassembler_Success; \
350
2.80k
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_12
Line
Count
Source
345
1.07k
  { \
346
1.07k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
1.07k
\
348
1.07k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
1.07k
    return MCDisassembler_Success; \
350
1.07k
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_9
Line
Count
Source
345
1.20k
  { \
346
1.20k
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
1.20k
\
348
1.20k
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
1.20k
    return MCDisassembler_Success; \
350
1.20k
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_13
Line
Count
Source
345
193
  { \
346
193
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
193
\
348
193
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
193
    return MCDisassembler_Success; \
350
193
  }
RISCVDisassembler.c:decodeSImmOperandAndLsl1_21
Line
Count
Source
345
338
  { \
346
338
    CS_ASSERT(isUIntN(N, Imm) && "Invalid immediate"); \
347
338
\
348
338
    MCOperand_CreateImm0(Inst, (SignExtend64((Imm << 1), N))); \
349
338
    return MCDisassembler_Success; \
350
338
  }
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
1.24k
{
359
1.24k
  CS_ASSERT(isUIntN(6, Imm) && "Invalid immediate");
360
1.24k
  if (Imm > 31) {
361
425
    Imm = (SignExtend64((Imm), 6) & 0xfffff);
362
425
  }
363
1.24k
  MCOperand_CreateImm0(Inst, (Imm));
364
1.24k
  return MCDisassembler_Success;
365
1.24k
}
366
367
static DecodeStatus decodeFRMArg(MCInst *Inst, uint32_t Imm, int64_t Address,
368
         const void *Decoder)
369
25.5k
{
370
25.5k
  CS_ASSERT(isUIntN(3, Imm) && "Invalid immediate");
371
25.5k
  if (!RISCVFPRndMode_isValidRoundingMode(Imm))
372
24
    return MCDisassembler_Fail;
373
374
25.5k
  MCOperand_CreateImm0(Inst, (Imm));
375
25.5k
  return MCDisassembler_Success;
376
25.5k
}
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
1.58k
{
415
1.58k
  CS_ASSERT(isUIntN(6, Imm) && "Invalid immediate");
416
417
1.58k
  if (!RISCV_getFeatureBits(Inst->csh->mode, RISCV_Feature64Bit) &&
418
418
      !isUIntN(5, Imm))
419
7
    return MCDisassembler_Fail;
420
421
1.58k
  MCOperand_CreateImm0(Inst, (Imm));
422
1.58k
  return MCDisassembler_Success;
423
1.58k
}
424
425
static DecodeStatus decodeUImmLog2XLenNonZeroOperand(MCInst *Inst, uint32_t Imm,
426
                 int64_t Address,
427
                 const void *Decoder)
428
1.50k
{
429
1.50k
  if (Imm == 0)
430
0
    return MCDisassembler_Fail;
431
1.50k
  return decodeUImmLog2XLenOperand(Inst, Imm, Address, Decoder);
432
1.50k
}
433
434
#include "RISCVGenDisassemblerTables.inc"
435
436
static DecodeStatus decodeRVCInstrRdRs1ImmZero(MCInst *Inst, uint32_t Insn,
437
                 uint64_t Address,
438
                 const void *Decoder)
439
106
{
440
106
  uint32_t Rd = fieldFromInstruction_4(Insn, 7, 5);
441
106
  DecodeStatus Result =
442
106
    DecodeGPRNoX0RegisterClass(Inst, Rd, Address, Decoder);
443
106
  (void)Result;
444
106
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid register");
445
106
  MCInst_addOperand2(Inst, (MCInst_getOperand(Inst, (0))));
446
106
  MCOperand_CreateImm0(Inst, (0));
447
106
  return MCDisassembler_Success;
448
106
}
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
206
{
464
206
  MCOperand_CreateReg0(Inst, (RISCV_X0));
465
206
  uint32_t SImm6 = fieldFromInstruction_4(Insn, 12, 1) << 5 |
466
206
       fieldFromInstruction_4(Insn, 2, 5);
467
206
  DecodeStatus Result =
468
206
    CONCAT(decodeSImmOperand, 6)(Inst, SImm6, Address, Decoder);
469
206
  (void)Result;
470
206
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid immediate");
471
206
  return MCDisassembler_Success;
472
206
}
473
474
static DecodeStatus decodeRVCInstrRdRs1UImm(MCInst *Inst, uint32_t Insn,
475
              uint64_t Address,
476
              const void *Decoder)
477
125
{
478
125
  MCOperand_CreateReg0(Inst, (RISCV_X0));
479
125
  MCInst_addOperand2(Inst, (MCInst_getOperand(Inst, (0))));
480
125
  uint32_t UImm6 = fieldFromInstruction_4(Insn, 12, 1) << 5 |
481
125
       fieldFromInstruction_4(Insn, 2, 5);
482
125
  DecodeStatus Result =
483
125
    CONCAT(decodeUImmOperand, 6)(Inst, UImm6, Address, Decoder);
484
125
  (void)Result;
485
125
  CS_ASSERT(Result == MCDisassembler_Success && "Invalid immediate");
486
125
  return MCDisassembler_Success;
487
125
}
488
489
static DecodeStatus decodeRVCInstrRdRs2(MCInst *Inst, uint32_t Insn,
490
          uint64_t Address, const void *Decoder)
491
35
{
492
35
  uint32_t Rd = fieldFromInstruction_4(Insn, 7, 5);
493
35
  uint32_t Rs2 = fieldFromInstruction_4(Insn, 2, 5);
494
35
  DecodeGPRRegisterClass(Inst, Rd, Address, Decoder);
495
35
  DecodeGPRRegisterClass(Inst, Rs2, Address, Decoder);
496
35
  return MCDisassembler_Success;
497
35
}
498
499
static DecodeStatus decodeRVCInstrRdRs1Rs2(MCInst *Inst, uint32_t Insn,
500
             uint64_t Address,
501
             const void *Decoder)
502
41
{
503
41
  uint32_t Rd = fieldFromInstruction_4(Insn, 7, 5);
504
41
  uint32_t Rs2 = fieldFromInstruction_4(Insn, 2, 5);
505
41
  DecodeGPRRegisterClass(Inst, Rd, Address, Decoder);
506
41
  MCInst_addOperand2(Inst, (MCInst_getOperand(Inst, (0))));
507
41
  DecodeGPRRegisterClass(Inst, Rs2, Address, Decoder);
508
41
  return MCDisassembler_Success;
509
41
}
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
181
{
550
181
  uint32_t Rs1 = fieldFromInstruction_4(Insn, 0, 5);
551
181
  uint32_t Rs2 = fieldFromInstruction_4(Insn, 5, 5);
552
181
  DecodeGPRRegisterClass(Inst, Rs1, Address, Decoder);
553
181
  DecodeGPRRegisterClass(Inst, Rs2, Address, Decoder);
554
181
  return MCDisassembler_Success;
555
181
}
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
23.5k
{
568
23.5k
  const MCInstrDesc *MCID = MCInstrDesc_get(MCInst_getOpcode(MI),
569
23.5k
              RISCVDescs.Insts,
570
23.5k
              ARR_SIZE(RISCVDescs.Insts));
571
23.5k
  MCOperand SPReg;
572
23.5k
  SPReg.MachineOperandType = kRegister;
573
23.5k
  SPReg.Kind = kRegister;
574
23.5k
  SPReg.RegVal = RISCV_X2;
575
78.1k
  for (unsigned i = 0; i < MCID->NumOperands; i++)
576
54.5k
    if (MCID->OpInfo[i].RegClass == RISCV_SPRegClassID)
577
4.27k
      MCInst_insert0(MI, i, &SPReg);
578
23.5k
}
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
37.6k
{
584
  // TODO: This will need modification when supporting instruction set
585
  // extensions with instructions > 32-bits (up to 176 bits wide).
586
37.6k
  uint32_t Insn;
587
37.6k
  DecodeStatus Result;
588
589
37.6k
#define TRY_TO_DECODE_WITH_ADDITIONAL_OPERATION( \
590
37.6k
  width, FEATURE_CHECKS, DECODER_TABLE, DESC, ADDITIONAL_OPERATION) \
591
493k
  do { \
592
534k
    if (FEATURE_CHECKS) { \
593
127k
      Result = decodeInstruction_##width( \
594
127k
        DECODER_TABLE, MI, Insn, Address, NULL); \
595
127k
      if (Result != MCDisassembler_Fail) { \
596
37.0k
        ADDITIONAL_OPERATION; \
597
37.0k
        return Result; \
598
37.0k
      } \
599
127k
    } \
600
493k
  } while (false)
601
37.6k
#define TRY_TO_DECODE_AND_ADD_SP(width, FEATURE_CHECKS, DECODER_TABLE, DESC) \
602
46.8k
  TRY_TO_DECODE_WITH_ADDITIONAL_OPERATION( \
603
46.8k
    width, FEATURE_CHECKS, DECODER_TABLE, DESC, addSPOperands(MI))
604
37.6k
#define TRY_TO_DECODE(width, FEATURE_CHECKS, DECODER_TABLE, DESC) \
605
446k
  TRY_TO_DECODE_WITH_ADDITIONAL_OPERATION( \
606
446k
    width, FEATURE_CHECKS, DECODER_TABLE, DESC, (void)NULL)
607
37.6k
#define TRY_TO_DECODE_FEATURE(width, FEATURE, DECODER_TABLE, DESC) \
608
407k
  TRY_TO_DECODE(width, RISCV_getFeatureBits(MI->csh->mode, FEATURE), \
609
407k
          DECODER_TABLE, DESC)
610
611
  // It's a 32 bit instruction if bit 0 and 1 are 1.
612
37.6k
  if ((Bytes[0] & 0x3) == 0x3) {
613
13.7k
    if (BytesLen < 4) {
614
116
      *Size = 0;
615
116
      return MCDisassembler_Fail;
616
116
    }
617
13.6k
    *Size = 4;
618
619
13.6k
    Insn = readBytes32(MI, Bytes);
620
621
13.6k
    TRY_TO_DECODE(4,
622
13.6k
            RISCV_getFeatureBits(MI->csh->mode,
623
13.6k
               RISCV_FeatureStdExtZdinx) &&
624
13.6k
              !RISCV_getFeatureBits(MI->csh->mode,
625
13.6k
                  RISCV_Feature64Bit),
626
13.6k
            DecoderTableRV32Zdinx32,
627
13.6k
            "RV32Zdinx table (Double in Integer and rv32)");
628
13.6k
    TRY_TO_DECODE(4,
629
13.6k
            RISCV_getFeatureBits(MI->csh->mode,
630
13.6k
               RISCV_FeatureStdExtZacas) &&
631
13.6k
              !RISCV_getFeatureBits(MI->csh->mode,
632
13.6k
                  RISCV_Feature64Bit),
633
13.6k
            DecoderTableRV32Zacas32,
634
13.6k
            "RV32Zacas table (Compare-And-Swap and rv32)");
635
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureStdExtZfinx,
636
13.6k
              DecoderTableRVZfinx32,
637
13.6k
              "RVZfinx table (Float in Integer)");
638
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXVentanaCondOps,
639
13.6k
              DecoderTableXVentana32,
640
13.6k
              "Ventana custom opcode table");
641
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadBa,
642
13.6k
              DecoderTableXTHeadBa32,
643
13.6k
              "XTHeadBa custom opcode table");
644
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadBb,
645
13.6k
              DecoderTableXTHeadBb32,
646
13.6k
              "XTHeadBb custom opcode table");
647
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadBs,
648
13.6k
              DecoderTableXTHeadBs32,
649
13.6k
              "XTHeadBs custom opcode table");
650
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadCondMov,
651
13.6k
              DecoderTableXTHeadCondMov32,
652
13.6k
              "XTHeadCondMov custom opcode table");
653
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadCmo,
654
13.6k
              DecoderTableXTHeadCmo32,
655
13.6k
              "XTHeadCmo custom opcode table");
656
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadFMemIdx,
657
13.6k
              DecoderTableXTHeadFMemIdx32,
658
13.6k
              "XTHeadFMemIdx custom opcode table");
659
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadMac,
660
13.6k
              DecoderTableXTHeadMac32,
661
13.6k
              "XTHeadMac custom opcode table");
662
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadMemIdx,
663
13.6k
              DecoderTableXTHeadMemIdx32,
664
13.6k
              "XTHeadMemIdx custom opcode table");
665
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadMemPair,
666
13.6k
              DecoderTableXTHeadMemPair32,
667
13.6k
              "XTHeadMemPair custom opcode table");
668
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadSync,
669
13.6k
              DecoderTableXTHeadSync32,
670
13.6k
              "XTHeadSync custom opcode table");
671
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXTHeadVdot,
672
13.6k
              DecoderTableXTHeadVdot32,
673
13.6k
              "XTHeadVdot custom opcode table");
674
13.6k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXSfvcp,
675
13.6k
              DecoderTableXSfvcp32,
676
13.6k
              "SiFive VCIX custom opcode table");
677
13.6k
    TRY_TO_DECODE_FEATURE(
678
13.6k
      4, RISCV_FeatureVendorXSfvqmaccdod,
679
13.6k
      DecoderTableXSfvqmaccdod32,
680
13.6k
      "SiFive Matrix Multiplication (2x8 and 8x2) Instruction opcode table");
681
13.6k
    TRY_TO_DECODE_FEATURE(
682
13.6k
      4, RISCV_FeatureVendorXSfvqmaccqoq,
683
13.6k
      DecoderTableXSfvqmaccqoq32,
684
13.6k
      "SiFive Matrix Multiplication (4x8 and 8x4) Instruction opcode table");
685
13.6k
    TRY_TO_DECODE_FEATURE(
686
13.6k
      4, RISCV_FeatureVendorXSfvfwmaccqqq,
687
13.6k
      DecoderTableXSfvfwmaccqqq32,
688
13.6k
      "SiFive Matrix Multiplication Instruction opcode table");
689
13.6k
    TRY_TO_DECODE_FEATURE(
690
13.6k
      4, RISCV_FeatureVendorXSfvfnrclipxfqf,
691
13.6k
      DecoderTableXSfvfnrclipxfqf32,
692
13.6k
      "SiFive FP32-to-int8 Ranged Clip Instructions opcode table");
693
13.6k
    TRY_TO_DECODE_FEATURE(
694
13.5k
      4, RISCV_FeatureVendorXCVbitmanip,
695
13.5k
      DecoderTableXCVbitmanip32,
696
13.5k
      "CORE-V Bit Manipulation custom opcode table");
697
13.5k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCVelw,
698
13.5k
              DecoderTableXCVelw32,
699
13.5k
              "CORE-V Event load custom opcode table");
700
13.5k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCVmac,
701
13.3k
              DecoderTableXCVmac32,
702
13.3k
              "CORE-V MAC custom opcode table");
703
13.3k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCVmem,
704
12.7k
              DecoderTableXCVmem32,
705
12.7k
              "CORE-V MEM custom opcode table");
706
12.7k
    TRY_TO_DECODE_FEATURE(4, RISCV_FeatureVendorXCValu,
707
12.6k
              DecoderTableXCValu32,
708
12.6k
              "CORE-V ALU custom opcode table");
709
12.6k
    TRY_TO_DECODE_FEATURE(
710
12.3k
      4, RISCV_FeatureVendorXCVsimd, DecoderTableXCVsimd32,
711
12.3k
      "CORE-V SIMD extensions custom opcode table");
712
12.3k
    TRY_TO_DECODE_FEATURE(
713
12.3k
      4, RISCV_FeatureVendorXCVbi, DecoderTableXCVbi32,
714
12.3k
      "CORE-V Immediate Branching custom opcode table");
715
12.3k
    TRY_TO_DECODE(4, true, DecoderTable32, "RISCV32 table");
716
717
217
    return MCDisassembler_Fail;
718
12.3k
  }
719
720
23.9k
  if (BytesLen < 2) {
721
139
    *Size = 0;
722
139
    return MCDisassembler_Fail;
723
139
  }
724
23.7k
  *Size = 2;
725
726
23.7k
  Insn = readBytes16(MI, Bytes);
727
23.7k
  TRY_TO_DECODE_AND_ADD_SP(
728
23.1k
    2, !RISCV_getFeatureBits(MI->csh->mode, RISCV_Feature64Bit),
729
23.1k
    DecoderTableRISCV32Only_16,
730
23.1k
    "RISCV32Only_16 table (16-bit Instruction)");
731
23.1k
  TRY_TO_DECODE_FEATURE(2, RISCV_FeatureStdExtZicfiss,
732
23.1k
            DecoderTableZicfiss16,
733
23.1k
            "RVZicfiss table (Shadow Stack)");
734
23.1k
  TRY_TO_DECODE_FEATURE(2, RISCV_FeatureStdExtZcmt, DecoderTableRVZcmt16,
735
23.1k
            "Zcmt table (16-bit Table Jump Instructions)");
736
23.1k
  TRY_TO_DECODE_FEATURE(
737
23.1k
    2, RISCV_FeatureStdExtZcmp, DecoderTableRVZcmp16,
738
23.1k
    "Zcmp table (16-bit Push/Pop & Double Move Instructions)");
739
23.1k
  TRY_TO_DECODE_AND_ADD_SP(2, true, DecoderTable16,
740
170
         "RISCV_C table (16-bit Instruction)");
741
742
170
  return MCDisassembler_Fail;
743
23.1k
}
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
37.6k
{
749
37.6k
  RISCV_init_cs_detail(MI);
750
37.6k
  MI->MRI = (MCRegisterInfo *)Info;
751
37.6k
  DecodeStatus ret =
752
37.6k
    RISCV_getInstruction(MI, Size, Bytes, ByteLen, Address, NULL);
753
754
37.6k
  MCInst_handleWriteback(MI, RISCVDescs.Insts,
755
37.6k
             ARR_SIZE(RISCVDescs.Insts));
756
757
37.6k
  return ret != MCDisassembler_Fail;
758
37.6k
}