[test] Pre-commit llvm.experimental.memset.pattern tests prior to MemoryLocation...
[llvm-project.git] / llvm / lib / Target / MSP430 / MCTargetDesc / MSP430MCCodeEmitter.cpp
blob35a7da4fbe5a22487617b2c03b561696fe1843aa
1 //===-- MSP430MCCodeEmitter.cpp - Convert MSP430 code to machine code -----===//
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 // This file implements the MSP430MCCodeEmitter class.
11 //===----------------------------------------------------------------------===//
13 #include "MSP430.h"
14 #include "MCTargetDesc/MSP430MCTargetDesc.h"
15 #include "MCTargetDesc/MSP430FixupKinds.h"
17 #include "llvm/ADT/APFloat.h"
18 #include "llvm/ADT/SmallVector.h"
19 #include "llvm/MC/MCCodeEmitter.h"
20 #include "llvm/MC/MCContext.h"
21 #include "llvm/MC/MCExpr.h"
22 #include "llvm/MC/MCFixup.h"
23 #include "llvm/MC/MCInst.h"
24 #include "llvm/MC/MCInstrInfo.h"
25 #include "llvm/MC/MCRegisterInfo.h"
26 #include "llvm/MC/MCSubtargetInfo.h"
27 #include "llvm/Support/EndianStream.h"
29 #define DEBUG_TYPE "mccodeemitter"
31 namespace llvm {
33 class MSP430MCCodeEmitter : public MCCodeEmitter {
34 MCContext &Ctx;
35 MCInstrInfo const &MCII;
37 // Offset keeps track of current word number being emitted
38 // inside a particular instruction.
39 mutable unsigned Offset;
41 /// TableGen'erated function for getting the binary encoding for an
42 /// instruction.
43 uint64_t getBinaryCodeForInstr(const MCInst &MI,
44 SmallVectorImpl<MCFixup> &Fixups,
45 const MCSubtargetInfo &STI) const;
47 /// Returns the binary encoding of operands.
48 ///
49 /// If an operand requires relocation, the relocation is recorded
50 /// and zero is returned.
51 unsigned getMachineOpValue(const MCInst &MI, const MCOperand &MO,
52 SmallVectorImpl<MCFixup> &Fixups,
53 const MCSubtargetInfo &STI) const;
55 unsigned getMemOpValue(const MCInst &MI, unsigned Op,
56 SmallVectorImpl<MCFixup> &Fixups,
57 const MCSubtargetInfo &STI) const;
59 unsigned getPCRelImmOpValue(const MCInst &MI, unsigned Op,
60 SmallVectorImpl<MCFixup> &Fixups,
61 const MCSubtargetInfo &STI) const;
63 unsigned getCGImmOpValue(const MCInst &MI, unsigned Op,
64 SmallVectorImpl<MCFixup> &Fixups,
65 const MCSubtargetInfo &STI) const;
67 unsigned getCCOpValue(const MCInst &MI, unsigned Op,
68 SmallVectorImpl<MCFixup> &Fixups,
69 const MCSubtargetInfo &STI) const;
71 public:
72 MSP430MCCodeEmitter(MCContext &ctx, MCInstrInfo const &MCII)
73 : Ctx(ctx), MCII(MCII) {}
75 void encodeInstruction(const MCInst &MI, SmallVectorImpl<char> &CB,
76 SmallVectorImpl<MCFixup> &Fixups,
77 const MCSubtargetInfo &STI) const override;
80 void MSP430MCCodeEmitter::encodeInstruction(const MCInst &MI,
81 SmallVectorImpl<char> &CB,
82 SmallVectorImpl<MCFixup> &Fixups,
83 const MCSubtargetInfo &STI) const {
84 const MCInstrDesc &Desc = MCII.get(MI.getOpcode());
85 // Get byte count of instruction.
86 unsigned Size = Desc.getSize();
88 // Initialize fixup offset
89 Offset = 2;
91 uint64_t BinaryOpCode = getBinaryCodeForInstr(MI, Fixups, STI);
92 size_t WordCount = Size / 2;
94 while (WordCount--) {
95 support::endian::write(CB, (uint16_t)BinaryOpCode,
96 llvm::endianness::little);
97 BinaryOpCode >>= 16;
101 unsigned MSP430MCCodeEmitter::getMachineOpValue(const MCInst &MI,
102 const MCOperand &MO,
103 SmallVectorImpl<MCFixup> &Fixups,
104 const MCSubtargetInfo &STI) const {
105 if (MO.isReg())
106 return Ctx.getRegisterInfo()->getEncodingValue(MO.getReg());
108 if (MO.isImm()) {
109 Offset += 2;
110 return MO.getImm();
113 assert(MO.isExpr() && "Expected expr operand");
114 Fixups.push_back(MCFixup::create(Offset, MO.getExpr(),
115 static_cast<MCFixupKind>(MSP430::fixup_16_byte), MI.getLoc()));
116 Offset += 2;
117 return 0;
120 unsigned MSP430MCCodeEmitter::getMemOpValue(const MCInst &MI, unsigned Op,
121 SmallVectorImpl<MCFixup> &Fixups,
122 const MCSubtargetInfo &STI) const {
123 const MCOperand &MO1 = MI.getOperand(Op);
124 assert(MO1.isReg() && "Register operand expected");
125 unsigned Reg = Ctx.getRegisterInfo()->getEncodingValue(MO1.getReg());
127 const MCOperand &MO2 = MI.getOperand(Op + 1);
128 if (MO2.isImm()) {
129 Offset += 2;
130 return ((unsigned)MO2.getImm() << 4) | Reg;
133 assert(MO2.isExpr() && "Expr operand expected");
134 MSP430::Fixups FixupKind;
135 switch (Reg) {
136 case 0:
137 FixupKind = MSP430::fixup_16_pcrel_byte;
138 break;
139 case 2:
140 FixupKind = MSP430::fixup_16_byte;
141 break;
142 default:
143 FixupKind = MSP430::fixup_16_byte;
144 break;
146 Fixups.push_back(MCFixup::create(Offset, MO2.getExpr(),
147 static_cast<MCFixupKind>(FixupKind), MI.getLoc()));
148 Offset += 2;
149 return Reg;
152 unsigned MSP430MCCodeEmitter::getPCRelImmOpValue(const MCInst &MI, unsigned Op,
153 SmallVectorImpl<MCFixup> &Fixups,
154 const MCSubtargetInfo &STI) const {
155 const MCOperand &MO = MI.getOperand(Op);
156 if (MO.isImm())
157 return MO.getImm();
159 assert(MO.isExpr() && "Expr operand expected");
160 Fixups.push_back(MCFixup::create(0, MO.getExpr(),
161 static_cast<MCFixupKind>(MSP430::fixup_10_pcrel), MI.getLoc()));
162 return 0;
165 unsigned MSP430MCCodeEmitter::getCGImmOpValue(const MCInst &MI, unsigned Op,
166 SmallVectorImpl<MCFixup> &Fixups,
167 const MCSubtargetInfo &STI) const {
168 const MCOperand &MO = MI.getOperand(Op);
169 assert(MO.isImm() && "Expr operand expected");
171 int64_t Imm = MO.getImm();
172 switch (Imm) {
173 default:
174 llvm_unreachable("Invalid immediate value");
175 case 4: return 0x22;
176 case 8: return 0x32;
177 case 0: return 0x03;
178 case 1: return 0x13;
179 case 2: return 0x23;
180 case -1: return 0x33;
184 unsigned MSP430MCCodeEmitter::getCCOpValue(const MCInst &MI, unsigned Op,
185 SmallVectorImpl<MCFixup> &Fixups,
186 const MCSubtargetInfo &STI) const {
187 const MCOperand &MO = MI.getOperand(Op);
188 assert(MO.isImm() && "Immediate operand expected");
189 switch (MO.getImm()) {
190 case MSP430CC::COND_NE: return 0;
191 case MSP430CC::COND_E: return 1;
192 case MSP430CC::COND_LO: return 2;
193 case MSP430CC::COND_HS: return 3;
194 case MSP430CC::COND_N: return 4;
195 case MSP430CC::COND_GE: return 5;
196 case MSP430CC::COND_L: return 6;
197 default:
198 llvm_unreachable("Unknown condition code");
202 MCCodeEmitter *createMSP430MCCodeEmitter(const MCInstrInfo &MCII,
203 MCContext &Ctx) {
204 return new MSP430MCCodeEmitter(Ctx, MCII);
207 #include "MSP430GenMCCodeEmitter.inc"
209 } // end of namespace llvm