[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / lib / Target / M68k / MCTargetDesc / M68kInstPrinter.cpp
blobe5f5909b5d79965ea08a37aa6f4babf09fb91d25
1 //===-- M68kInstPrinter.cpp - Convert M68k MCInst to asm ----*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 ///
9 /// \file
10 /// This file contains definitions for an M68k MCInst printer.
11 ///
12 //===----------------------------------------------------------------------===//
14 // TODO Conform with all supported Motorola ASM syntax
15 // Motorola's assembly has several syntax variants, especially on
16 // addressing modes.
17 // For example, you can write pc indirect w/ displacement as
18 // `x(%pc)`, where `x` is the displacement imm, or `(x,%pc)`.
19 // Currently we're picking the variant that is different from
20 // GCC, albeit being recognizable by GNU AS.
21 // Not sure what is the impact now (e.g. some syntax might
22 // not be recognized by some old consoles' toolchains, in which
23 // case we can not use our integrated assembler), but either way,
24 // it will be great to support all of the variants in the future.
26 #include "M68kInstPrinter.h"
27 #include "M68kBaseInfo.h"
29 #include "llvm/ADT/StringExtras.h"
30 #include "llvm/MC/MCExpr.h"
31 #include "llvm/MC/MCInst.h"
32 #include "llvm/MC/MCInstrInfo.h"
33 #include "llvm/MC/MCSymbol.h"
34 #include "llvm/Support/ErrorHandling.h"
35 #include "llvm/Support/raw_ostream.h"
37 using namespace llvm;
39 #define DEBUG_TYPE "asm-printer"
41 #define PRINT_ALIAS_INSTR
42 #include "M68kGenAsmWriter.inc"
44 void M68kInstPrinter::printRegName(raw_ostream &OS, unsigned RegNo) const {
45 OS << "%" << getRegisterName(RegNo);
48 void M68kInstPrinter::printInst(const MCInst *MI, uint64_t Address,
49 StringRef Annot, const MCSubtargetInfo &STI,
50 raw_ostream &O) {
51 if (!printAliasInstr(MI, Address, O))
52 printInstruction(MI, Address, O);
54 printAnnotation(O, Annot);
57 void M68kInstPrinter::printOperand(const MCInst *MI, unsigned OpNo,
58 raw_ostream &O) {
59 const MCOperand &MO = MI->getOperand(OpNo);
60 if (MO.isReg()) {
61 printRegName(O, MO.getReg());
62 return;
65 if (MO.isImm()) {
66 printImmediate(MI, OpNo, O);
67 return;
70 assert(MO.isExpr() && "Unknown operand kind in printOperand");
71 MO.getExpr()->print(O, &MAI);
74 void M68kInstPrinter::printImmediate(const MCInst *MI, unsigned opNum,
75 raw_ostream &O) {
76 const MCOperand &MO = MI->getOperand(opNum);
77 if (MO.isImm())
78 O << '#' << MO.getImm();
79 else if (MO.isExpr()) {
80 O << '#';
81 MO.getExpr()->print(O, &MAI);
82 } else
83 llvm_unreachable("Unknown immediate kind");
86 void M68kInstPrinter::printMoveMask(const MCInst *MI, unsigned opNum,
87 raw_ostream &O) {
88 unsigned Mask = MI->getOperand(opNum).getImm();
89 assert((Mask & 0xFFFF) == Mask && "Mask is always 16 bits");
91 // A move mask is splitted into two parts:
92 // bits 0 ~ 7 correspond to D0 ~ D7 regs
93 // bits 8 ~ 15 correspond to A0 ~ A7 regs
95 // In the assembly syntax, we want to use a dash to replace
96 // a continuous range of registers. For example, if the bit
97 // mask is 0b101110, we want to print "D1-D3,D5" instead of
98 // "D1,D2,D3,D4,D5".
100 // However, we don't want a dash to cross between data registers
101 // and address registers (i.e. there shouldn't be a dash crossing
102 // bit 7 and 8) since that is not really intuitive. So we simply
103 // print the data register part (bit 0~7) and address register part
104 // separately.
105 uint8_t HalfMask;
106 unsigned Reg;
107 for (int s = 0; s < 16; s += 8) {
108 HalfMask = (Mask >> s) & 0xFF;
109 // Print separation comma only if
110 // both data & register parts have bit(s) set
111 if (s != 0 && (Mask & 0xFF) && HalfMask)
112 O << ',';
114 for (int i = 0; HalfMask; ++i) {
115 if ((HalfMask >> i) & 0b1) {
116 HalfMask ^= 0b1 << i;
117 Reg = M68kII::getMaskedSpillRegister(i + s);
118 printRegName(O, Reg);
120 int j = i;
121 while ((HalfMask >> (j + 1)) & 0b1)
122 HalfMask ^= 0b1 << ++j;
124 if (j != i) {
125 O << '-';
126 Reg = M68kII::getMaskedSpillRegister(j + s);
127 printRegName(O, Reg);
130 i = j;
132 if (HalfMask)
133 O << ',';
139 void M68kInstPrinter::printDisp(const MCInst *MI, unsigned opNum,
140 raw_ostream &O) {
141 const MCOperand &Op = MI->getOperand(opNum);
142 if (Op.isImm()) {
143 O << Op.getImm();
144 return;
146 assert(Op.isExpr() && "Unknown operand kind in printOperand");
147 Op.getExpr()->print(O, &MAI);
150 void M68kInstPrinter::printARIMem(const MCInst *MI, unsigned opNum,
151 raw_ostream &O) {
152 O << '(';
153 printOperand(MI, opNum, O);
154 O << ')';
157 void M68kInstPrinter::printARIPIMem(const MCInst *MI, unsigned opNum,
158 raw_ostream &O) {
159 O << "(";
160 printOperand(MI, opNum, O);
161 O << ")+";
164 void M68kInstPrinter::printARIPDMem(const MCInst *MI, unsigned opNum,
165 raw_ostream &O) {
166 O << "-(";
167 printOperand(MI, opNum, O);
168 O << ")";
171 void M68kInstPrinter::printARIDMem(const MCInst *MI, unsigned opNum,
172 raw_ostream &O) {
173 O << '(';
174 printDisp(MI, opNum + M68k::MemDisp, O);
175 O << ',';
176 printOperand(MI, opNum + M68k::MemBase, O);
177 O << ')';
180 void M68kInstPrinter::printARIIMem(const MCInst *MI, unsigned opNum,
181 raw_ostream &O) {
182 O << '(';
183 printDisp(MI, opNum + M68k::MemDisp, O);
184 O << ',';
185 printOperand(MI, opNum + M68k::MemBase, O);
186 O << ',';
187 printOperand(MI, opNum + M68k::MemIndex, O);
188 O << ')';
191 // NOTE forcing (W,L) size available since M68020 only
192 void M68kInstPrinter::printAbsMem(const MCInst *MI, unsigned opNum,
193 raw_ostream &O) {
194 const MCOperand &MO = MI->getOperand(opNum);
196 if (MO.isExpr()) {
197 MO.getExpr()->print(O, &MAI);
198 return;
201 assert(MO.isImm() && "absolute memory addressing needs an immediate");
202 O << format("$%0" PRIx64, (uint64_t)MO.getImm());
205 void M68kInstPrinter::printPCDMem(const MCInst *MI, uint64_t Address,
206 unsigned opNum, raw_ostream &O) {
207 O << '(';
208 printDisp(MI, opNum + M68k::PCRelDisp, O);
209 O << ",%pc)";
212 void M68kInstPrinter::printPCIMem(const MCInst *MI, uint64_t Address,
213 unsigned opNum, raw_ostream &O) {
214 O << '(';
215 printDisp(MI, opNum + M68k::PCRelDisp, O);
216 O << ",%pc,";
217 printOperand(MI, opNum + M68k::PCRelIndex, O);
218 O << ')';