Coverage Report

Created: 2026-02-26 07:11

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonenext/arch/Mips/MipsInstPrinter.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
//===-- MipsInstPrinter.cpp - Convert Mips MCInst to assembly syntax ------===//
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 class prints an Mips MCInst to a .s file.
24
//
25
//===----------------------------------------------------------------------===//
26
27
#include <stdio.h>
28
#include <string.h>
29
#include <stdlib.h>
30
#include <capstone/platform.h>
31
32
#include "MipsMapping.h"
33
#include "MipsInstPrinter.h"
34
35
#define GET_SUBTARGETINFO_ENUM
36
#include "MipsGenSubtargetInfo.inc"
37
38
#define GET_INSTRINFO_ENUM
39
#include "MipsGenInstrInfo.inc"
40
41
#define GET_REGINFO_ENUM
42
#include "MipsGenRegisterInfo.inc"
43
44
19.4k
#define CONCAT(a, b) CONCAT_(a, b)
45
19.4k
#define CONCAT_(a, b) a##_##b
46
47
#define DEBUG_TYPE "asm-printer"
48
49
#define PRINT_ALIAS_INSTR
50
#include "MipsGenAsmWriter.inc"
51
52
static bool isReg(const MCInst *MI, unsigned OpNo, unsigned R)
53
8.28k
{
54
8.28k
  return MCOperand_getReg(MCInst_getOperand((MCInst *)MI, (OpNo))) == R;
55
8.28k
}
56
57
static const char *MipsFCCToString(Mips_CondCode CC)
58
0
{
59
0
  switch (CC) {
60
0
  case Mips_FCOND_F:
61
0
  case Mips_FCOND_T:
62
0
    return "f";
63
0
  case Mips_FCOND_UN:
64
0
  case Mips_FCOND_OR:
65
0
    return "un";
66
0
  case Mips_FCOND_OEQ:
67
0
  case Mips_FCOND_UNE:
68
0
    return "eq";
69
0
  case Mips_FCOND_UEQ:
70
0
  case Mips_FCOND_ONE:
71
0
    return "ueq";
72
0
  case Mips_FCOND_OLT:
73
0
  case Mips_FCOND_UGE:
74
0
    return "olt";
75
0
  case Mips_FCOND_ULT:
76
0
  case Mips_FCOND_OGE:
77
0
    return "ult";
78
0
  case Mips_FCOND_OLE:
79
0
  case Mips_FCOND_UGT:
80
0
    return "ole";
81
0
  case Mips_FCOND_ULE:
82
0
  case Mips_FCOND_OGT:
83
0
    return "ule";
84
0
  case Mips_FCOND_SF:
85
0
  case Mips_FCOND_ST:
86
0
    return "sf";
87
0
  case Mips_FCOND_NGLE:
88
0
  case Mips_FCOND_GLE:
89
0
    return "ngle";
90
0
  case Mips_FCOND_SEQ:
91
0
  case Mips_FCOND_SNE:
92
0
    return "seq";
93
0
  case Mips_FCOND_NGL:
94
0
  case Mips_FCOND_GL:
95
0
    return "ngl";
96
0
  case Mips_FCOND_LT:
97
0
  case Mips_FCOND_NLT:
98
0
    return "lt";
99
0
  case Mips_FCOND_NGE:
100
0
  case Mips_FCOND_GE:
101
0
    return "nge";
102
0
  case Mips_FCOND_LE:
103
0
  case Mips_FCOND_NLE:
104
0
    return "le";
105
0
  case Mips_FCOND_NGT:
106
0
  case Mips_FCOND_GT:
107
0
    return "ngt";
108
0
  }
109
0
  CS_ASSERT_RET_VAL(0 && "Impossible condition code!", NULL);
110
0
  return "";
111
0
}
112
113
const char *Mips_LLVM_getRegisterName(unsigned RegNo, bool noRegName);
114
115
static void printRegName(MCInst *MI, SStream *OS, MCRegister Reg)
116
221k
{
117
221k
  int syntax_opt = MI->csh->syntax;
118
221k
  if (!(syntax_opt & CS_OPT_SYNTAX_NO_DOLLAR)) {
119
221k
    SStream_concat1(OS, '$');
120
221k
  }
121
221k
  SStream_concat0(OS, Mips_LLVM_getRegisterName(
122
221k
            Reg, syntax_opt & CS_OPT_SYNTAX_NOREGNAME));
123
221k
}
124
125
static void patch_cs_printer(MCInst *MI, SStream *O)
126
121k
{
127
  // replace '# 16 bit inst' to empty.
128
121k
  SStream_replc(O, '#', 0);
129
121k
  SStream_trimls(O);
130
131
121k
  if (MI->csh->syntax & CS_OPT_SYNTAX_NO_DOLLAR) {
132
0
    char *dollar = strchr(O->buffer, '$');
133
0
    if (!dollar) {
134
0
      return;
135
0
    }
136
0
    size_t dollar_len = strlen(dollar + 1);
137
    // to include `\0`
138
0
    memmove(dollar, dollar + 1, dollar_len + 1);
139
0
  }
140
121k
}
141
142
static void patch_cs_detail_operand_reg(cs_mips_op *op, unsigned reg,
143
          unsigned access)
144
262
{
145
262
  op->type = MIPS_OP_REG;
146
262
  op->reg = reg;
147
262
  op->is_reglist = false;
148
262
  op->access = access;
149
262
}
150
151
static void patch_cs_details(MCInst *MI)
152
121k
{
153
121k
  if (!detail_is_set(MI))
154
0
    return;
155
156
121k
  cs_mips_op *op0 = NULL, *op1 = NULL, *op2 = NULL;
157
121k
  unsigned opcode = MCInst_getOpcode(MI);
158
121k
  unsigned n_ops = MCInst_getNumOperands(MI);
159
160
121k
  switch (opcode) {
161
  /* mips r2 to r5 only 64bit */
162
31
  case Mips_DSDIV: /// ddiv $$zero, $rs, $rt
163
    /* fall-thru */
164
118
  case Mips_DUDIV: /// ddivu $$zero, $rs, $rt
165
118
    if (n_ops != 2) {
166
0
      return;
167
0
    }
168
118
    Mips_inc_op_count(MI);
169
118
    op0 = Mips_get_detail_op(MI, -3);
170
118
    op1 = Mips_get_detail_op(MI, -2);
171
118
    op2 = Mips_get_detail_op(MI, -1);
172
    // move all details by one and add $zero reg
173
118
    *op2 = *op1;
174
118
    *op1 = *op0;
175
118
    patch_cs_detail_operand_reg(op0, MIPS_REG_ZERO_64, CS_AC_WRITE);
176
118
    return;
177
178
  /* mips r2 to r5 only */
179
35
  case Mips_SDIV: /// div $$zero, $rs, $rt
180
    /* fall-thru */
181
65
  case Mips_UDIV: /// divu $$zero, $rs, $rt
182
    /* fall-thru */
183
  /* microMIPS only */
184
73
  case Mips_SDIV_MM: /// div $$zero, $rs, $rt
185
    /* fall-thru */
186
144
  case Mips_UDIV_MM: /// divu $$zero, $rs, $rt
187
    /* fall-thru */
188
189
  /* MIPS16 only */
190
144
  case Mips_DivRxRy16: /// div $$zero, $rx, $ry
191
    /* fall-thru */
192
144
  case Mips_DivuRxRy16: /// divu $$zero, $rx, $ry
193
144
    if (n_ops != 2) {
194
0
      return;
195
0
    }
196
144
    Mips_inc_op_count(MI);
197
144
    op0 = Mips_get_detail_op(MI, -3);
198
144
    op1 = Mips_get_detail_op(MI, -2);
199
144
    op2 = Mips_get_detail_op(MI, -1);
200
    // move all details by one and add $zero reg
201
144
    *op2 = *op1;
202
144
    *op1 = *op0;
203
144
    patch_cs_detail_operand_reg(op0, MIPS_REG_ZERO, CS_AC_WRITE);
204
144
    return;
205
0
  case Mips_AddiuSpImm16: /// addiu $$sp, imm8
206
    /* fall-thru */
207
0
  case Mips_AddiuSpImmX16: /// addiu $$sp, imm8
208
0
    if (n_ops != 1) {
209
0
      return;
210
0
    }
211
0
    Mips_inc_op_count(MI);
212
0
    op0 = Mips_get_detail_op(MI, -2);
213
0
    op1 = Mips_get_detail_op(MI, -1);
214
    // move all details by one and add $sp reg
215
0
    *op1 = *op0;
216
0
    patch_cs_detail_operand_reg(op0, MIPS_REG_SP, CS_AC_READ_WRITE);
217
0
    return;
218
0
  case Mips_JrcRa16: /// jrc $ra
219
    /* fall-thru */
220
0
  case Mips_JrRa16: /// jr $ra
221
0
    if (n_ops > 0) {
222
0
      return;
223
0
    }
224
0
    Mips_inc_op_count(MI);
225
0
    op0 = Mips_get_detail_op(MI, -1);
226
0
    patch_cs_detail_operand_reg(op0, MIPS_REG_RA, CS_AC_READ);
227
0
    return;
228
120k
  default:
229
120k
    return;
230
121k
  }
231
121k
}
232
233
void Mips_LLVM_printInst(MCInst *MI, uint64_t Address, SStream *O)
234
121k
{
235
121k
  bool useAliasDetails = map_use_alias_details(MI);
236
121k
  if (!useAliasDetails) {
237
0
    SStream_Close(O);
238
0
    printInstruction(MI, Address, O);
239
0
    SStream_Open(O);
240
0
    map_set_fill_detail_ops(MI, false);
241
0
  }
242
243
121k
  if (printAliasInstr(MI, Address, O) || printAlias4(MI, Address, O)) {
244
7.09k
    MCInst_setIsAlias(MI, true);
245
114k
  } else {
246
114k
    printInstruction(MI, Address, O);
247
114k
  }
248
249
121k
  patch_cs_printer(MI, O);
250
121k
  patch_cs_details(MI);
251
252
121k
  if (!useAliasDetails) {
253
0
    map_set_fill_detail_ops(MI, true);
254
0
  }
255
121k
}
256
257
void printOperand(MCInst *MI, unsigned OpNo, SStream *O)
258
264k
{
259
264k
  switch (MCInst_getOpcode(MI)) {
260
264k
  default:
261
264k
    break;
262
264k
  case Mips_AND16_NM:
263
0
  case Mips_XOR16_NM:
264
0
  case Mips_OR16_NM:
265
0
    if (MCInst_getNumOperands(MI) == 2 && OpNo == 2)
266
0
      OpNo = 0; // rt, rs -> rt, rs, rt
267
0
    break;
268
0
  case Mips_ADDu4x4_NM:
269
0
  case Mips_MUL4x4_NM:
270
0
    if (MCInst_getNumOperands(MI) == 2 && OpNo > 0)
271
0
      OpNo = OpNo - 1; // rt, rs -> rt, rt, rs
272
0
    break;
273
264k
  }
274
275
264k
  MCOperand *Op = MCInst_getOperand(MI, (OpNo));
276
264k
  if (MCOperand_isReg(Op)) {
277
215k
    Mips_add_cs_detail_0(MI, Mips_OP_GROUP_Operand, OpNo);
278
215k
    printRegName(MI, O, MCOperand_getReg(Op));
279
215k
    return;
280
215k
  }
281
282
49.3k
  if (MCOperand_isImm(Op)) {
283
49.3k
    switch (MCInst_getOpcode(MI)) {
284
0
    case Mips_LI48_NM:
285
0
    case Mips_ANDI16_NM:
286
0
    case Mips_ANDI_NM:
287
0
    case Mips_ORI_NM:
288
0
    case Mips_XORI_NM:
289
0
    case Mips_TEQ_NM:
290
0
    case Mips_TNE_NM:
291
0
    case Mips_SIGRIE_NM:
292
0
    case Mips_SDBBP_NM:
293
0
    case Mips_SDBBP16_NM:
294
0
    case Mips_BREAK_NM:
295
0
    case Mips_BREAK16_NM:
296
0
    case Mips_SYSCALL_NM:
297
0
    case Mips_SYSCALL16_NM:
298
0
    case Mips_WAIT_NM:
299
0
      CONCAT(printUImm, CONCAT(32, 0))
300
0
      (MI, OpNo, O);
301
0
      break;
302
49.3k
    default:
303
49.3k
      Mips_add_cs_detail_0(MI, Mips_OP_GROUP_Operand, OpNo);
304
49.3k
      printInt64(O, MCOperand_getImm(Op));
305
49.3k
      break;
306
49.3k
    }
307
49.3k
    return;
308
49.3k
  }
309
49.3k
}
310
311
static void printJumpOperand(MCInst *MI, unsigned OpNo, SStream *O)
312
3.61k
{
313
3.61k
  Mips_add_cs_detail_0(MI, Mips_OP_GROUP_JumpOperand, OpNo);
314
3.61k
  MCOperand *Op = MCInst_getOperand(MI, (OpNo));
315
3.61k
  if (MCOperand_isReg(Op))
316
0
    return printRegName(MI, O, MCOperand_getReg(Op));
317
318
  // only the upper bits are needed.
319
3.61k
  uint64_t Base = MI->address & ~0x0fffffffull;
320
3.61k
  uint64_t Target = MCOperand_getImm(Op);
321
3.61k
  printInt64(O, Base | Target);
322
3.61k
}
323
324
static void printBranchOperand(MCInst *MI, uint64_t Address, unsigned OpNo,
325
             SStream *O)
326
23.3k
{
327
23.3k
  Mips_add_cs_detail_0(MI, Mips_OP_GROUP_BranchOperand, OpNo);
328
23.3k
  MCOperand *Op = MCInst_getOperand(MI, (OpNo));
329
23.3k
  if (MCOperand_isReg(Op))
330
823
    return printRegName(MI, O, MCOperand_getReg(Op));
331
332
22.5k
  uint64_t Target = Address + MCOperand_getImm(Op);
333
22.5k
  printInt64(O, Target);
334
22.5k
}
335
336
#define DEFINE_printUImm(Bits) \
337
  static void CONCAT(printUImm, CONCAT(Bits, 0))(MCInst * MI, int opNum, \
338
                   SStream *O) \
339
18.8k
  { \
340
18.8k
    Mips_add_cs_detail_0( \
341
18.8k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
18.8k
      opNum); \
343
18.8k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
18.8k
    if (MCOperand_isImm(MO)) { \
345
18.8k
      uint64_t Imm = MCOperand_getImm(MO); \
346
18.8k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
18.8k
      printUInt64(O, Imm); \
348
18.8k
      return; \
349
18.8k
    } \
350
18.8k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_10_0
Line
Count
Source
339
1.36k
  { \
340
1.36k
    Mips_add_cs_detail_0( \
341
1.36k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
1.36k
      opNum); \
343
1.36k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
1.36k
    if (MCOperand_isImm(MO)) { \
345
1.36k
      uint64_t Imm = MCOperand_getImm(MO); \
346
1.36k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
1.36k
      printUInt64(O, Imm); \
348
1.36k
      return; \
349
1.36k
    } \
350
1.36k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_4_0
Line
Count
Source
339
2.97k
  { \
340
2.97k
    Mips_add_cs_detail_0( \
341
2.97k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
2.97k
      opNum); \
343
2.97k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
2.97k
    if (MCOperand_isImm(MO)) { \
345
2.97k
      uint64_t Imm = MCOperand_getImm(MO); \
346
2.97k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
2.97k
      printUInt64(O, Imm); \
348
2.97k
      return; \
349
2.97k
    } \
350
2.97k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_5_0
Line
Count
Source
339
3.57k
  { \
340
3.57k
    Mips_add_cs_detail_0( \
341
3.57k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
3.57k
      opNum); \
343
3.57k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
3.57k
    if (MCOperand_isImm(MO)) { \
345
3.57k
      uint64_t Imm = MCOperand_getImm(MO); \
346
3.57k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
3.57k
      printUInt64(O, Imm); \
348
3.57k
      return; \
349
3.57k
    } \
350
3.57k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
Unexecuted instantiation: MipsInstPrinter.c:printUImm_26_0
MipsInstPrinter.c:printUImm_8_0
Line
Count
Source
339
1.38k
  { \
340
1.38k
    Mips_add_cs_detail_0( \
341
1.38k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
1.38k
      opNum); \
343
1.38k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
1.38k
    if (MCOperand_isImm(MO)) { \
345
1.38k
      uint64_t Imm = MCOperand_getImm(MO); \
346
1.38k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
1.38k
      printUInt64(O, Imm); \
348
1.38k
      return; \
349
1.38k
    } \
350
1.38k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
Unexecuted instantiation: MipsInstPrinter.c:printUImm_12_0
MipsInstPrinter.c:printUImm_20_0
Line
Count
Source
339
263
  { \
340
263
    Mips_add_cs_detail_0( \
341
263
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
263
      opNum); \
343
263
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
263
    if (MCOperand_isImm(MO)) { \
345
263
      uint64_t Imm = MCOperand_getImm(MO); \
346
263
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
263
      printUInt64(O, Imm); \
348
263
      return; \
349
263
    } \
350
263
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_16_0
Line
Count
Source
339
3.64k
  { \
340
3.64k
    Mips_add_cs_detail_0( \
341
3.64k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
3.64k
      opNum); \
343
3.64k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
3.64k
    if (MCOperand_isImm(MO)) { \
345
3.64k
      uint64_t Imm = MCOperand_getImm(MO); \
346
3.64k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
3.64k
      printUInt64(O, Imm); \
348
3.64k
      return; \
349
3.64k
    } \
350
3.64k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
Unexecuted instantiation: MipsInstPrinter.c:printUImm_32_0
MipsInstPrinter.c:printUImm_7_0
Line
Count
Source
339
21
  { \
340
21
    Mips_add_cs_detail_0( \
341
21
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
21
      opNum); \
343
21
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
21
    if (MCOperand_isImm(MO)) { \
345
21
      uint64_t Imm = MCOperand_getImm(MO); \
346
21
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
21
      printUInt64(O, Imm); \
348
21
      return; \
349
21
    } \
350
21
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_2_0
Line
Count
Source
339
973
  { \
340
973
    Mips_add_cs_detail_0( \
341
973
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
973
      opNum); \
343
973
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
973
    if (MCOperand_isImm(MO)) { \
345
973
      uint64_t Imm = MCOperand_getImm(MO); \
346
973
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
973
      printUInt64(O, Imm); \
348
973
      return; \
349
973
    } \
350
973
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_1_0
Line
Count
Source
339
957
  { \
340
957
    Mips_add_cs_detail_0( \
341
957
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
957
      opNum); \
343
957
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
957
    if (MCOperand_isImm(MO)) { \
345
957
      uint64_t Imm = MCOperand_getImm(MO); \
346
957
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
957
      printUInt64(O, Imm); \
348
957
      return; \
349
957
    } \
350
957
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_3_0
Line
Count
Source
339
2.67k
  { \
340
2.67k
    Mips_add_cs_detail_0( \
341
2.67k
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
2.67k
      opNum); \
343
2.67k
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
2.67k
    if (MCOperand_isImm(MO)) { \
345
2.67k
      uint64_t Imm = MCOperand_getImm(MO); \
346
2.67k
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
2.67k
      printUInt64(O, Imm); \
348
2.67k
      return; \
349
2.67k
    } \
350
2.67k
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_0_0
Line
Count
Source
339
273
  { \
340
273
    Mips_add_cs_detail_0( \
341
273
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
273
      opNum); \
343
273
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
273
    if (MCOperand_isImm(MO)) { \
345
273
      uint64_t Imm = MCOperand_getImm(MO); \
346
273
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
273
      printUInt64(O, Imm); \
348
273
      return; \
349
273
    } \
350
273
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
MipsInstPrinter.c:printUImm_6_0
Line
Count
Source
339
791
  { \
340
791
    Mips_add_cs_detail_0( \
341
791
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, 0)), \
342
791
      opNum); \
343
791
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
344
791
    if (MCOperand_isImm(MO)) { \
345
791
      uint64_t Imm = MCOperand_getImm(MO); \
346
791
      Imm &= (((uint64_t)1) << Bits) - 1; \
347
791
      printUInt64(O, Imm); \
348
791
      return; \
349
791
    } \
350
791
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
351
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
352
0
  }
353
354
#define DEFINE_printUImm_2(Bits, Offset) \
355
  static void CONCAT(printUImm, CONCAT(Bits, Offset))( \
356
    MCInst * MI, int opNum, SStream *O) \
357
604
  { \
358
604
    Mips_add_cs_detail_0( \
359
604
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, Offset)), \
360
604
      opNum); \
361
604
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
362
604
    if (MCOperand_isImm(MO)) { \
363
604
      uint64_t Imm = MCOperand_getImm(MO); \
364
604
      Imm -= Offset; \
365
604
      Imm &= (1 << Bits) - 1; \
366
604
      Imm += Offset; \
367
604
      printUInt64(O, Imm); \
368
604
      return; \
369
604
    } \
370
604
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
371
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
372
0
  }
MipsInstPrinter.c:printUImm_2_1
Line
Count
Source
357
439
  { \
358
439
    Mips_add_cs_detail_0( \
359
439
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, Offset)), \
360
439
      opNum); \
361
439
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
362
439
    if (MCOperand_isImm(MO)) { \
363
439
      uint64_t Imm = MCOperand_getImm(MO); \
364
439
      Imm -= Offset; \
365
439
      Imm &= (1 << Bits) - 1; \
366
439
      Imm += Offset; \
367
439
      printUInt64(O, Imm); \
368
439
      return; \
369
439
    } \
370
439
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
371
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
372
0
  }
Unexecuted instantiation: MipsInstPrinter.c:printUImm_5_32
MipsInstPrinter.c:printUImm_5_1
Line
Count
Source
357
165
  { \
358
165
    Mips_add_cs_detail_0( \
359
165
      MI, CONCAT(Mips_OP_GROUP_UImm, CONCAT(Bits, Offset)), \
360
165
      opNum); \
361
165
    MCOperand *MO = MCInst_getOperand(MI, (opNum)); \
362
165
    if (MCOperand_isImm(MO)) { \
363
165
      uint64_t Imm = MCOperand_getImm(MO); \
364
165
      Imm -= Offset; \
365
165
      Imm &= (1 << Bits) - 1; \
366
165
      Imm += Offset; \
367
165
      printUInt64(O, Imm); \
368
165
      return; \
369
165
    } \
370
165
    MCOperand *Op = MCInst_getOperand(MI, (opNum)); \
371
0
    printRegName(MI, O, MCOperand_getReg(Op)); \
372
0
  }
Unexecuted instantiation: MipsInstPrinter.c:printUImm_6_1
Unexecuted instantiation: MipsInstPrinter.c:printUImm_5_33
Unexecuted instantiation: MipsInstPrinter.c:printUImm_6_2
373
374
DEFINE_printUImm(0);
375
DEFINE_printUImm(1);
376
DEFINE_printUImm(10);
377
DEFINE_printUImm(12);
378
DEFINE_printUImm(16);
379
DEFINE_printUImm(2);
380
DEFINE_printUImm(20);
381
DEFINE_printUImm(26);
382
DEFINE_printUImm(3);
383
DEFINE_printUImm(32);
384
DEFINE_printUImm(4);
385
DEFINE_printUImm(5);
386
DEFINE_printUImm(6);
387
DEFINE_printUImm(7);
388
DEFINE_printUImm(8);
389
DEFINE_printUImm_2(2, 1);
390
DEFINE_printUImm_2(5, 1);
391
DEFINE_printUImm_2(5, 32);
392
DEFINE_printUImm_2(5, 33);
393
DEFINE_printUImm_2(6, 1);
394
DEFINE_printUImm_2(6, 2);
395
396
static void printMemOperand(MCInst *MI, int opNum, SStream *O)
397
33.3k
{
398
  // Load/Store memory operands -- imm($reg)
399
  // If PIC target the target is loaded as the
400
  // pattern lw $25,%call16($28)
401
402
  // opNum can be invalid if instruction had reglist as operand.
403
  // MemOperand is always last operand of instruction (base + offset).
404
33.3k
  switch (MCInst_getOpcode(MI)) {
405
31.4k
  default:
406
31.4k
    break;
407
31.4k
  case Mips_SWM32_MM:
408
568
  case Mips_LWM32_MM:
409
936
  case Mips_SWM16_MM:
410
1.17k
  case Mips_SWM16_MMR6:
411
1.63k
  case Mips_LWM16_MM:
412
1.89k
  case Mips_LWM16_MMR6:
413
1.89k
    opNum = MCInst_getNumOperands(MI) - 2;
414
1.89k
    break;
415
33.3k
  }
416
417
33.3k
  set_mem_access(MI, true);
418
  // Index register is encoded as immediate value
419
  // in case of nanoMIPS indexed instructions
420
33.3k
  switch (MCInst_getOpcode(MI)) {
421
  // No offset needed for paired LL/SC
422
0
  case Mips_LLWP_NM:
423
0
  case Mips_SCWP_NM:
424
0
    break;
425
0
  case Mips_LWX_NM:
426
0
  case Mips_LWXS_NM:
427
0
  case Mips_LWXS16_NM:
428
0
  case Mips_LBX_NM:
429
0
  case Mips_LBUX_NM:
430
0
  case Mips_LHX_NM:
431
0
  case Mips_LHUX_NM:
432
0
  case Mips_LHXS_NM:
433
0
  case Mips_LHUXS_NM:
434
0
  case Mips_SWX_NM:
435
0
  case Mips_SWXS_NM:
436
0
  case Mips_SBX_NM:
437
0
  case Mips_SHX_NM:
438
0
  case Mips_SHXS_NM:
439
0
    if (!MCOperand_isReg(MCInst_getOperand(MI, (opNum + 1)))) {
440
0
      Mips_add_cs_detail_0(MI, Mips_OP_GROUP_MemOperand,
441
0
               (opNum + 1));
442
0
      printRegName(MI, O,
443
0
             MCOperand_getImm(MCInst_getOperand(
444
0
               MI, (opNum + 1))));
445
0
      break;
446
0
    }
447
    // Fall through
448
33.3k
  default:
449
33.3k
    printOperand((MCInst *)MI, opNum + 1, O);
450
33.3k
    break;
451
33.3k
  }
452
33.3k
  SStream_concat0(O, "(");
453
33.3k
  printOperand((MCInst *)MI, opNum, O);
454
33.3k
  SStream_concat0(O, ")");
455
33.3k
  set_mem_access(MI, false);
456
33.3k
}
457
458
static void printMemOperandEA(MCInst *MI, int opNum, SStream *O)
459
0
{
460
  // when using stack locations for not load/store instructions
461
  // print the same way as all normal 3 operand instructions.
462
0
  printOperand((MCInst *)MI, opNum, O);
463
0
  SStream_concat0(O, ", ");
464
0
  printOperand((MCInst *)MI, opNum + 1, O);
465
0
}
466
467
static void printFCCOperand(MCInst *MI, int opNum, SStream *O)
468
0
{
469
0
  MCOperand *MO = MCInst_getOperand(MI, (opNum));
470
0
  SStream_concat0(O,
471
0
      MipsFCCToString((Mips_CondCode)MCOperand_getImm(MO)));
472
0
}
473
474
static bool printAlias(const char *Str, const MCInst *MI, uint64_t Address,
475
           unsigned OpNo, SStream *OS, bool IsBranch)
476
2.11k
{
477
2.11k
  SStream_concat(OS, "%s%s", "\t", Str);
478
2.11k
  SStream_concat0(OS, "\t");
479
2.11k
  if (IsBranch)
480
1.15k
    printBranchOperand((MCInst *)MI, Address, OpNo, OS);
481
969
  else
482
969
    printOperand((MCInst *)MI, OpNo, OS);
483
2.11k
  return true;
484
2.11k
}
485
486
static bool printAlias2(const char *Str, const MCInst *MI, uint64_t Address,
487
      unsigned OpNo0, unsigned OpNo1, SStream *OS,
488
      bool IsBranch)
489
1.52k
{
490
1.52k
  printAlias(Str, MI, Address, OpNo0, OS, IsBranch);
491
1.52k
  SStream_concat0(OS, ", ");
492
1.52k
  if (IsBranch)
493
823
    printBranchOperand((MCInst *)MI, Address, OpNo1, OS);
494
702
  else
495
702
    printOperand((MCInst *)MI, OpNo1, OS);
496
1.52k
  return true;
497
1.52k
}
498
499
static bool printAlias3(const char *Str, const MCInst *MI, uint64_t Address,
500
      unsigned OpNo0, unsigned OpNo1, unsigned OpNo2,
501
      SStream *OS)
502
0
{
503
0
  printAlias(Str, MI, Address, OpNo0, OS, false);
504
0
  SStream_concat0(OS, ", ");
505
0
  printOperand((MCInst *)MI, OpNo1, OS);
506
0
  SStream_concat0(OS, ", ");
507
0
  printOperand((MCInst *)MI, OpNo2, OS);
508
0
  return true;
509
0
}
510
511
static bool printAlias4(const MCInst *MI, uint64_t Address, SStream *OS)
512
116k
{
513
116k
  switch (MCInst_getOpcode(MI)) {
514
1.82k
  case Mips_BEQ:
515
2.20k
  case Mips_BEQ_MM:
516
    // beq $zero, $zero, $L2 => b $L2
517
    // beq $r0, $zero, $L2 => beqz $r0, $L2
518
2.20k
    return (isReg(MI, 0, Mips_ZERO) && isReg(MI, 1, Mips_ZERO) &&
519
327
      printAlias("b", MI, Address, 2, OS, true)) ||
520
1.88k
           (isReg(MI, 1, Mips_ZERO) &&
521
353
      printAlias2("beqz", MI, Address, 0, 2, OS, true));
522
0
  case Mips_BEQ64:
523
    // beq $r0, $zero, $L2 => beqz $r0, $L2
524
0
    return isReg(MI, 1, Mips_ZERO_64) &&
525
0
           printAlias2("beqz", MI, Address, 0, 2, OS, true);
526
955
  case Mips_BNE:
527
1.22k
  case Mips_BNE_MM:
528
    // bne $r0, $zero, $L2 => bnez $r0, $L2
529
1.22k
    return isReg(MI, 1, Mips_ZERO) &&
530
470
           printAlias2("bnez", MI, Address, 0, 2, OS, true);
531
0
  case Mips_BNE64:
532
    // bne $r0, $zero, $L2 => bnez $r0, $L2
533
0
    return isReg(MI, 1, Mips_ZERO_64) &&
534
0
           printAlias2("bnez", MI, Address, 0, 2, OS, true);
535
454
  case Mips_BGEZAL:
536
    // bgezal $zero, $L1 => bal $L1
537
454
    return isReg(MI, 0, Mips_ZERO) &&
538
0
           printAlias("bal", MI, Address, 1, OS, true);
539
27
  case Mips_BC1T:
540
    // bc1t $fcc0, $L1 => bc1t $L1
541
27
    return isReg(MI, 0, Mips_FCC0) &&
542
0
           printAlias("bc1t", MI, Address, 1, OS, true);
543
10
  case Mips_BC1F:
544
    // bc1f $fcc0, $L1 => bc1f $L1
545
10
    return isReg(MI, 0, Mips_FCC0) &&
546
0
           printAlias("bc1f", MI, Address, 1, OS, true);
547
339
  case Mips_JALR:
548
    // jalr $zero, $r1 => jr $r1
549
    // jalr $ra, $r1 => jalr $r1
550
339
    return (isReg(MI, 0, Mips_ZERO) &&
551
92
      printAlias("jr", MI, Address, 1, OS, false)) ||
552
247
           (isReg(MI, 0, Mips_RA) &&
553
175
      printAlias("jalr", MI, Address, 1, OS, false));
554
0
  case Mips_JALR64:
555
    // jalr $zero, $r1 => jr $r1
556
    // jalr $ra, $r1 => jalr $r1
557
0
    return (isReg(MI, 0, Mips_ZERO_64) &&
558
0
      printAlias("jr", MI, Address, 1, OS, false)) ||
559
0
           (isReg(MI, 0, Mips_RA_64) &&
560
0
      printAlias("jalr", MI, Address, 1, OS, false));
561
57
  case Mips_NOR:
562
116
  case Mips_NOR_MM:
563
155
  case Mips_NOR_MMR6:
564
    // nor $r0, $r1, $zero => not $r0, $r1
565
155
    return isReg(MI, 2, Mips_ZERO) &&
566
58
           printAlias2("not", MI, Address, 0, 1, OS, false);
567
0
  case Mips_NOR64:
568
    // nor $r0, $r1, $zero => not $r0, $r1
569
0
    return isReg(MI, 2, Mips_ZERO_64) &&
570
0
           printAlias2("not", MI, Address, 0, 1, OS, false);
571
46
  case Mips_OR:
572
1.10k
  case Mips_ADDu:
573
    // or $r0, $r1, $zero => move $r0, $r1
574
    // addu $r0, $r1, $zero => move $r0, $r1
575
1.10k
    return isReg(MI, 2, Mips_ZERO) &&
576
644
           printAlias2("move", MI, Address, 0, 1, OS, false);
577
0
  case Mips_LI48_NM:
578
0
  case Mips_LI16_NM:
579
    // li[16/48] $r0, imm => li $r0, imm
580
0
    return printAlias2("li", MI, Address, 0, 1, OS, false);
581
0
  case Mips_ADDIU_NM:
582
0
  case Mips_ADDIUNEG_NM:
583
0
    if (isReg(MI, 1, Mips_ZERO_NM))
584
0
      return printAlias2("li", MI, Address, 0, 2, OS, false);
585
0
    else
586
0
      return printAlias3("addiu", MI, Address, 0, 1, 2, OS);
587
0
  case Mips_ADDIU48_NM:
588
0
  case Mips_ADDIURS5_NM:
589
0
  case Mips_ADDIUR1SP_NM:
590
0
  case Mips_ADDIUR2_NM:
591
0
  case Mips_ADDIUGPB_NM:
592
0
  case Mips_ADDIUGPW_NM:
593
0
    return printAlias3("addiu", MI, Address, 0, 1, 2, OS);
594
0
  case Mips_ANDI16_NM:
595
0
  case Mips_ANDI_NM:
596
    // andi[16/32] $r0, $r1, imm => andi $r0, $r1, imm
597
0
    return printAlias3("andi", MI, Address, 0, 1, 2, OS);
598
110k
  default:
599
110k
    return false;
600
116k
  }
601
116k
}
602
603
static void printRegisterList(MCInst *MI, int opNum, SStream *O)
604
1.89k
{
605
  // - 2 because register List is always first operand of instruction and it is
606
  // always followed by memory operand (base + offset).
607
1.89k
  Mips_add_cs_detail_0(MI, Mips_OP_GROUP_RegisterList, opNum);
608
7.54k
  for (int i = opNum, e = MCInst_getNumOperands(MI) - 2; i != e; ++i) {
609
5.64k
    if (i != opNum)
610
3.74k
      SStream_concat0(O, ", ");
611
5.64k
    printRegName(MI, O,
612
5.64k
           MCOperand_getReg(MCInst_getOperand(MI, (i))));
613
5.64k
  }
614
1.89k
}
615
616
static void printNanoMipsRegisterList(MCInst *MI, int OpNum, SStream *O)
617
0
{
618
0
  Mips_add_cs_detail_0(MI, Mips_OP_GROUP_NanoMipsRegisterList, OpNum);
619
0
  for (unsigned I = OpNum; I < MCInst_getNumOperands(MI); I++) {
620
0
    SStream_concat0(O, ", ");
621
0
    printRegName(MI, O,
622
0
           MCOperand_getReg(MCInst_getOperand(MI, (I))));
623
0
  }
624
0
}
625
626
static void printHi20(MCInst *MI, int OpNum, SStream *O)
627
0
{
628
0
  MCOperand *MO = MCInst_getOperand(MI, (OpNum));
629
0
  if (MCOperand_isImm(MO)) {
630
0
    Mips_add_cs_detail_0(MI, Mips_OP_GROUP_Hi20, OpNum);
631
0
    SStream_concat0(O, "%hi(");
632
0
    printUInt64(O, MCOperand_getImm(MO));
633
0
    SStream_concat0(O, ")");
634
0
  } else
635
0
    printOperand(MI, OpNum, O);
636
0
}
637
638
static void printHi20PCRel(MCInst *MI, uint64_t Address, int OpNum, SStream *O)
639
0
{
640
0
  MCOperand *MO = MCInst_getOperand(MI, (OpNum));
641
0
  if (MCOperand_isImm(MO)) {
642
0
    Mips_add_cs_detail_0(MI, Mips_OP_GROUP_Hi20PCRel, OpNum);
643
0
    SStream_concat0(O, "%pcrel_hi(");
644
0
    printUInt64(O, MCOperand_getImm(MO) + Address);
645
0
    SStream_concat0(O, ")");
646
0
  } else
647
0
    printOperand(MI, OpNum, O);
648
0
}
649
650
static void printPCRel(MCInst *MI, uint64_t Address, int OpNum, SStream *O)
651
0
{
652
0
  MCOperand *MO = MCInst_getOperand(MI, (OpNum));
653
0
  if (MCOperand_isImm(MO)) {
654
0
    Mips_add_cs_detail_0(MI, Mips_OP_GROUP_PCRel, OpNum);
655
0
    printUInt64(O, MCOperand_getImm(MO) + Address);
656
0
  } else
657
0
    printOperand(MI, OpNum, O);
658
0
}
659
660
const char *Mips_LLVM_getRegisterName(unsigned RegNo, bool noRegName)
661
251k
{
662
251k
  if (!RegNo || RegNo >= MIPS_REG_ENDING) {
663
0
    return NULL;
664
0
  }
665
251k
  if (noRegName) {
666
0
    return getRegisterName(RegNo);
667
0
  }
668
251k
  switch (RegNo) {
669
25.0k
  case MIPS_REG_AT:
670
25.5k
  case MIPS_REG_AT_64:
671
25.5k
    return "at";
672
6.52k
  case MIPS_REG_A0:
673
7.00k
  case MIPS_REG_A0_64:
674
7.00k
    return "a0";
675
6.35k
  case MIPS_REG_A1:
676
6.72k
  case MIPS_REG_A1_64:
677
6.72k
    return "a1";
678
7.25k
  case MIPS_REG_A2:
679
7.51k
  case MIPS_REG_A2_64:
680
7.51k
    return "a2";
681
6.17k
  case MIPS_REG_A3:
682
6.49k
  case MIPS_REG_A3_64:
683
6.49k
    return "a3";
684
2.54k
  case MIPS_REG_K0:
685
2.91k
  case MIPS_REG_K0_64:
686
2.91k
    return "k0";
687
2.44k
  case MIPS_REG_K1:
688
3.24k
  case MIPS_REG_K1_64:
689
3.24k
    return "k1";
690
14.3k
  case MIPS_REG_S0:
691
14.8k
  case MIPS_REG_S0_64:
692
14.8k
    return "s0";
693
3.97k
  case MIPS_REG_S1:
694
4.38k
  case MIPS_REG_S1_64:
695
4.38k
    return "s1";
696
3.39k
  case MIPS_REG_S2:
697
3.52k
  case MIPS_REG_S2_64:
698
3.52k
    return "s2";
699
3.41k
  case MIPS_REG_S3:
700
3.95k
  case MIPS_REG_S3_64:
701
3.95k
    return "s3";
702
3.55k
  case MIPS_REG_S4:
703
3.96k
  case MIPS_REG_S4_64:
704
3.96k
    return "s4";
705
2.23k
  case MIPS_REG_S5:
706
2.57k
  case MIPS_REG_S5_64:
707
2.57k
    return "s5";
708
1.96k
  case MIPS_REG_S6:
709
2.38k
  case MIPS_REG_S6_64:
710
2.38k
    return "s6";
711
2.82k
  case MIPS_REG_S7:
712
3.23k
  case MIPS_REG_S7_64:
713
3.23k
    return "s7";
714
4.25k
  case MIPS_REG_T0:
715
4.37k
  case MIPS_REG_T0_64:
716
4.37k
    return "t0";
717
1.75k
  case MIPS_REG_T1:
718
1.91k
  case MIPS_REG_T1_64:
719
1.91k
    return "t1";
720
2.51k
  case MIPS_REG_T2:
721
2.83k
  case MIPS_REG_T2_64:
722
2.83k
    return "t2";
723
1.28k
  case MIPS_REG_T3:
724
1.73k
  case MIPS_REG_T3_64:
725
1.73k
    return "t3";
726
2.46k
  case MIPS_REG_T4:
727
2.73k
  case MIPS_REG_T4_64:
728
2.73k
    return "t4";
729
2.67k
  case MIPS_REG_T5:
730
2.84k
  case MIPS_REG_T5_64:
731
2.84k
    return "t5";
732
2.47k
  case MIPS_REG_T6:
733
3.15k
  case MIPS_REG_T6_64:
734
3.15k
    return "t6";
735
3.25k
  case MIPS_REG_T7:
736
3.62k
  case MIPS_REG_T7_64:
737
3.62k
    return "t7";
738
3.94k
  case MIPS_REG_T8:
739
4.43k
  case MIPS_REG_T8_64:
740
4.43k
    return "t8";
741
2.06k
  case MIPS_REG_T9:
742
2.52k
  case MIPS_REG_T9_64:
743
2.52k
    return "t9";
744
10.9k
  case MIPS_REG_V0:
745
11.3k
  case MIPS_REG_V0_64:
746
11.3k
    return "v0";
747
5.84k
  case MIPS_REG_V1:
748
6.48k
  case MIPS_REG_V1_64:
749
6.48k
    return "v1";
750
104k
  default:
751
104k
    return getRegisterName(RegNo);
752
251k
  }
753
251k
}