Coverage Report

Created: 2025-11-16 06:38

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/capstonenext/arch/Sparc/SparcInstPrinter.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
//===-- SparcInstPrinter.cpp - Convert Sparc 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 Sparc 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 "../../MCInstPrinter.h"
33
#include "../../Mapping.h"
34
#include "SparcInstPrinter.h"
35
#include "SparcLinkage.h"
36
#include "SparcMCTargetDesc.h"
37
#include "SparcMapping.h"
38
#include "SparcDisassemblerExtension.h"
39
40
#define CONCAT(a, b) CONCAT_(a, b)
41
#define CONCAT_(a, b) a##_##b
42
43
#define DEBUG_TYPE "asm-printer"
44
45
static void printCustomAliasOperand(MCInst *MI, uint64_t Address,
46
            unsigned OpIdx, unsigned PrintMethodIdx,
47
            SStream *OS);
48
static void printOperand(MCInst *MI, int opNum, SStream *O);
49
50
#define GET_INSTRUCTION_NAME
51
#define PRINT_ALIAS_INSTR
52
#include "SparcGenAsmWriter.inc"
53
54
static void printRegName(SStream *OS, MCRegister Reg)
55
75.3k
{
56
75.3k
  SStream_concat1(OS, '%');
57
75.3k
  SStream_concat0(OS, getRegisterName(Reg, Sparc_NoRegAltName));
58
75.3k
}
59
60
static void printRegNameAlt(SStream *OS, MCRegister Reg, unsigned AltIdx)
61
47.9k
{
62
47.9k
  SStream_concat1(OS, '%');
63
47.9k
  SStream_concat0(OS, getRegisterName(Reg, AltIdx));
64
47.9k
}
65
66
static void printInst(MCInst *MI, uint64_t Address, SStream *O)
67
41.2k
{
68
41.2k
  bool isAlias = false;
69
41.2k
  bool useAliasDetails = map_use_alias_details(MI);
70
41.2k
  map_set_fill_detail_ops(MI, useAliasDetails);
71
72
41.2k
  if (!printAliasInstr(MI, Address, O) && !printSparcAliasInstr(MI, O)) {
73
36.1k
    MCInst_setIsAlias(MI, false);
74
36.1k
  } else {
75
5.06k
    isAlias = true;
76
5.06k
    MCInst_setIsAlias(MI, isAlias);
77
5.06k
    if (useAliasDetails) {
78
5.06k
      return;
79
5.06k
    }
80
5.06k
  }
81
82
36.1k
  if (!isAlias || !useAliasDetails) {
83
36.1k
    map_set_fill_detail_ops(MI, !(isAlias && useAliasDetails));
84
36.1k
    if (isAlias)
85
0
      SStream_Close(O);
86
36.1k
    printInstruction(MI, Address, O);
87
36.1k
    if (isAlias)
88
0
      SStream_Open(O);
89
36.1k
  }
90
36.1k
}
91
92
bool printSparcAliasInstr(MCInst *MI, SStream *O)
93
37.3k
{
94
37.3k
  switch (MCInst_getOpcode(MI)) {
95
35.6k
  default:
96
35.6k
    return false;
97
41
  case Sparc_JMPLrr:
98
1.27k
  case Sparc_JMPLri: {
99
1.27k
    if (MCInst_getNumOperands(MI) != 3)
100
0
      return false;
101
1.27k
    if (!MCOperand_isReg(MCInst_getOperand(MI, (0))))
102
0
      return false;
103
1.27k
    switch (MCOperand_getReg(MCInst_getOperand(MI, (0)))) {
104
77
    default:
105
77
      return false;
106
1.15k
    case Sparc_G0: // jmp $addr | ret | retl
107
1.15k
      if (MCOperand_isImm(MCInst_getOperand(MI, (2))) &&
108
1.11k
          MCOperand_getImm(MCInst_getOperand(MI, (2))) == 8) {
109
891
        switch (MCOperand_getReg(
110
891
          MCInst_getOperand(MI, (1)))) {
111
332
        default:
112
332
          break;
113
332
        case Sparc_I7:
114
44
          SStream_concat0(O, "\tret");
115
44
          return true;
116
515
        case Sparc_O7:
117
515
          SStream_concat0(O, "\tretl");
118
515
          return true;
119
891
        }
120
891
      }
121
599
      SStream_concat0(O, "\tjmp ");
122
599
      printMemOperand(MI, 1, O);
123
599
      return true;
124
35
    case Sparc_O7: // call $addr
125
35
      SStream_concat0(O, "\tcall ");
126
35
      printMemOperand(MI, 1, O);
127
35
      return true;
128
1.27k
    }
129
1.27k
  }
130
80
  case Sparc_V9FCMPS:
131
161
  case Sparc_V9FCMPD:
132
229
  case Sparc_V9FCMPQ:
133
287
  case Sparc_V9FCMPES:
134
376
  case Sparc_V9FCMPED:
135
458
  case Sparc_V9FCMPEQ: {
136
458
    if (Sparc_getFeatureBits(MI->csh->mode, Sparc_FeatureV9) ||
137
175
        (MCInst_getNumOperands(MI) != 3) ||
138
175
        (!MCOperand_isReg(MCInst_getOperand(MI, (0)))) ||
139
175
        (MCOperand_getReg(MCInst_getOperand(MI, (0))) !=
140
175
         Sparc_FCC0))
141
458
      return false;
142
    // if V8, skip printing %fcc0.
143
0
    switch (MCInst_getOpcode(MI)) {
144
0
    default:
145
0
    case Sparc_V9FCMPS:
146
0
      SStream_concat0(O, "\tfcmps ");
147
0
      break;
148
0
    case Sparc_V9FCMPD:
149
0
      SStream_concat0(O, "\tfcmpd ");
150
0
      break;
151
0
    case Sparc_V9FCMPQ:
152
0
      SStream_concat0(O, "\tfcmpq ");
153
0
      break;
154
0
    case Sparc_V9FCMPES:
155
0
      SStream_concat0(O, "\tfcmpes ");
156
0
      break;
157
0
    case Sparc_V9FCMPED:
158
0
      SStream_concat0(O, "\tfcmped ");
159
0
      break;
160
0
    case Sparc_V9FCMPEQ:
161
0
      SStream_concat0(O, "\tfcmpeq ");
162
0
      break;
163
0
    }
164
0
    printOperand(MI, 1, O);
165
0
    SStream_concat0(O, ", ");
166
0
    printOperand(MI, 2, O);
167
0
    return true;
168
0
  }
169
37.3k
  }
170
37.3k
}
171
172
static void printOperand(MCInst *MI, int opNum, SStream *O)
173
79.4k
{
174
79.4k
  Sparc_add_cs_detail_0(MI, Sparc_OP_GROUP_Operand, opNum);
175
79.4k
  MCOperand *MO = MCInst_getOperand(MI, (opNum));
176
177
79.4k
  if (MCOperand_isReg(MO)) {
178
54.5k
    unsigned Reg = MCOperand_getReg(MO);
179
54.5k
    if (Sparc_getFeatureBits(MI->csh->mode, Sparc_FeatureV9))
180
47.9k
      printRegNameAlt(O, Reg, Sparc_RegNamesStateReg);
181
6.67k
    else
182
6.67k
      printRegName(O, Reg);
183
54.5k
    return;
184
54.5k
  }
185
186
24.8k
  if (MCOperand_isImm(MO)) {
187
24.8k
    switch (MCInst_getOpcode(MI)) {
188
24.5k
    default:
189
24.5k
      printInt32(O, (int)MCOperand_getImm(MO));
190
24.5k
      return;
191
192
115
    case Sparc_TICCri: // Fall through
193
115
    case Sparc_TICCrr: // Fall through
194
184
    case Sparc_TRAPri: // Fall through
195
184
    case Sparc_TRAPrr: // Fall through
196
310
    case Sparc_TXCCri: // Fall through
197
310
    case Sparc_TXCCrr: // Fall through
198
      // Only seven-bit values up to 127.
199
310
      printInt8(O, ((int)MCOperand_getImm(MO) & 0x7f));
200
310
      return;
201
24.8k
    }
202
24.8k
  }
203
204
0
  CS_ASSERT(MCOperand_isExpr(MO) &&
205
0
      "Unknown operand kind in printOperand");
206
0
}
207
208
void printMemOperand(MCInst *MI, int opNum, SStream *O)
209
10.7k
{
210
10.7k
  Sparc_add_cs_detail_0(MI, Sparc_OP_GROUP_MemOperand, opNum);
211
10.7k
  MCOperand *Op1 = MCInst_getOperand(MI, (opNum));
212
10.7k
  MCOperand *Op2 = MCInst_getOperand(MI, (opNum + 1));
213
214
10.7k
  bool PrintedFirstOperand = false;
215
10.7k
  if (MCOperand_isReg(Op1) && MCOperand_getReg(Op1) != Sparc_G0) {
216
9.69k
    printOperand(MI, opNum, O);
217
9.69k
    PrintedFirstOperand = true;
218
9.69k
  }
219
220
  // Skip the second operand iff it adds nothing (literal 0 or %g0) and we've
221
  // already printed the first one
222
10.7k
  const bool SkipSecondOperand =
223
10.7k
    PrintedFirstOperand &&
224
9.69k
    ((MCOperand_isReg(Op2) && MCOperand_getReg(Op2) == Sparc_G0) ||
225
8.30k
     (MCOperand_isImm(Op2) && MCOperand_getImm(Op2) == 0));
226
227
10.7k
  if (!SkipSecondOperand) {
228
9.21k
    if (PrintedFirstOperand)
229
8.19k
      SStream_concat0(O, "+");
230
231
9.21k
    printOperand(MI, opNum + 1, O);
232
9.21k
  }
233
10.7k
}
234
235
void printCCOperand(MCInst *MI, int opNum, SStream *O)
236
8.50k
{
237
8.50k
  Sparc_add_cs_detail_0(MI, Sparc_OP_GROUP_CCOperand, opNum);
238
8.50k
  int CC = (int)MCOperand_getImm(MCInst_getOperand(MI, (opNum)));
239
8.50k
  switch (MCInst_getOpcode(MI)) {
240
1.61k
  default:
241
1.61k
    break;
242
1.61k
  case Sparc_FBCOND:
243
1.00k
  case Sparc_FBCONDA:
244
1.22k
  case Sparc_FBCOND_V9:
245
1.42k
  case Sparc_FBCONDA_V9:
246
1.72k
  case Sparc_BPFCC:
247
2.08k
  case Sparc_BPFCCA:
248
2.08k
  case Sparc_BPFCCNT:
249
2.08k
  case Sparc_BPFCCANT:
250
2.62k
  case Sparc_MOVFCCrr:
251
2.62k
  case Sparc_V9MOVFCCrr:
252
3.16k
  case Sparc_MOVFCCri:
253
3.16k
  case Sparc_V9MOVFCCri:
254
3.20k
  case Sparc_FMOVS_FCC:
255
3.20k
  case Sparc_V9FMOVS_FCC:
256
3.55k
  case Sparc_FMOVD_FCC:
257
3.55k
  case Sparc_V9FMOVD_FCC:
258
4.03k
  case Sparc_FMOVQ_FCC:
259
4.03k
  case Sparc_V9FMOVQ_FCC:
260
    // Make sure CC is a fp conditional flag.
261
4.03k
    CC = (CC < SPARC_CC_FCC_BEGIN) ? (CC + SPARC_CC_FCC_BEGIN) : CC;
262
4.03k
    break;
263
666
  case Sparc_CBCOND:
264
1.11k
  case Sparc_CBCONDA:
265
    // Make sure CC is a cp conditional flag.
266
1.11k
    CC = (CC < SPARC_CC_CPCC_BEGIN) ? (CC + SPARC_CC_CPCC_BEGIN) :
267
1.11k
              CC;
268
1.11k
    break;
269
283
  case Sparc_BPR:
270
714
  case Sparc_BPRA:
271
1.23k
  case Sparc_BPRNT:
272
1.31k
  case Sparc_BPRANT:
273
1.38k
  case Sparc_MOVRri:
274
1.46k
  case Sparc_MOVRrr:
275
1.54k
  case Sparc_FMOVRS:
276
1.61k
  case Sparc_FMOVRD:
277
1.74k
  case Sparc_FMOVRQ:
278
    // Make sure CC is a register conditional flag.
279
1.74k
    CC = (CC < SPARC_CC_REG_BEGIN) ? (CC + SPARC_CC_REG_BEGIN) : CC;
280
1.74k
    break;
281
8.50k
  }
282
8.50k
  SStream_concat0(O, SPARCCondCodeToString((sparc_cc)CC));
283
8.50k
}
284
285
bool printGetPCX(MCInst *MI, unsigned opNum, SStream *O)
286
0
{
287
0
  printf("FIXME: Implement SparcInstPrinter::printGetPCX.");
288
0
  return true;
289
0
}
290
291
void printMembarTag(MCInst *MI, int opNum, SStream *O)
292
1.04k
{
293
1.04k
  Sparc_add_cs_detail_0(MI, Sparc_OP_GROUP_MembarTag, opNum);
294
1.04k
  static const char *const TagNames[] = { "#LoadLoad",  "#StoreLoad",
295
1.04k
            "#LoadStore", "#StoreStore",
296
1.04k
            "#Lookaside", "#MemIssue",
297
1.04k
            "#Sync" };
298
299
1.04k
  unsigned Imm = MCOperand_getImm(MCInst_getOperand(MI, (opNum)));
300
301
1.04k
  if (Imm > 127) {
302
108
    printUInt32(O, Imm);
303
108
    return;
304
108
  }
305
306
1.04k
  bool First = true;
307
53.2k
  for (unsigned i = 0; i < sizeof(TagNames); i++) {
308
52.3k
    if (Imm & (1ull << i)) {
309
3.70k
      SStream_concat(O, "%s", (First ? "" : " | "));
310
3.70k
      SStream_concat0(O, TagNames[i]);
311
3.70k
      First = false;
312
3.70k
    }
313
52.3k
  }
314
934
}
315
316
#define GET_ASITAG_IMPL
317
#include "SparcGenSystemOperands.inc"
318
319
void printASITag(MCInst *MI, int opNum, SStream *O)
320
3.47k
{
321
3.47k
  Sparc_add_cs_detail_0(MI, Sparc_OP_GROUP_ASITag, opNum);
322
3.47k
  unsigned Imm = MCOperand_getImm(MCInst_getOperand(MI, (opNum)));
323
3.47k
  const Sparc_ASITag_ASITag *ASITag =
324
3.47k
    Sparc_ASITag_lookupASITagByEncoding(Imm);
325
3.47k
  if (Sparc_getFeatureBits(MI->csh->mode, Sparc_FeatureV9) && ASITag) {
326
245
    SStream_concat1(O, '#');
327
245
    SStream_concat0(O, ASITag->Name);
328
245
  } else
329
3.22k
    printUInt32(O, Imm);
330
3.47k
}
331
332
void Sparc_LLVM_printInst(MCInst *MI, uint64_t Address, const char *Annot,
333
        SStream *O)
334
41.2k
{
335
41.2k
  printInst(MI, Address, O);
336
41.2k
}
337
338
const char *Sparc_LLVM_getRegisterName(unsigned RegNo, unsigned AltIdx)
339
16.8k
{
340
16.8k
  return getRegisterName(RegNo, AltIdx);
341
16.8k
}