LLVM 23.0.0git
LoongArchMCTargetDesc.cpp
Go to the documentation of this file.
1//===-- LoongArchMCTargetDesc.cpp - LoongArch Target Descriptions ---------===//
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 provides LoongArch specific target descriptions.
10//
11//===----------------------------------------------------------------------===//
12
16#include "LoongArchMCAsmInfo.h"
19#include "llvm/MC/MCAsmInfo.h"
21#include "llvm/MC/MCDwarf.h"
23#include "llvm/MC/MCInstrInfo.h"
29#include <bitset>
30
31#define GET_INSTRINFO_MC_DESC
32#define ENABLE_INSTR_PREDICATE_VERIFIER
33#include "LoongArchGenInstrInfo.inc"
34
35#define GET_REGINFO_MC_DESC
36#include "LoongArchGenRegisterInfo.inc"
37
38#define GET_SUBTARGETINFO_MC_DESC
39#include "LoongArchGenSubtargetInfo.inc"
40
41using namespace llvm;
42
45 InitLoongArchMCRegisterInfo(X, LoongArch::R1);
46 return X;
47}
48
50 MCInstrInfo *X = new MCInstrInfo();
51 InitLoongArchMCInstrInfo(X);
52 return X;
53}
54
55static MCSubtargetInfo *
57 if (CPU.empty() || CPU == "generic")
58 CPU = TT.isArch64Bit() ? "generic-la64" : "generic-la32";
59 return createLoongArchMCSubtargetInfoImpl(TT, CPU, /*TuneCPU*/ CPU, FS);
60}
61
63 const Triple &TT,
64 const MCTargetOptions &Options) {
65 MCAsmInfo *MAI = new LoongArchMCAsmInfo(TT);
66
67 // Initial state of the frame pointer is sp(r3).
68 unsigned SP = MRI.getDwarfRegNum(LoongArch::R3, true);
69 MCCFIInstruction Inst = MCCFIInstruction::cfiDefCfa(nullptr, SP, 0);
70 MAI->addInitialFrameState(Inst);
71
72 return MAI;
73}
74
76 unsigned SyntaxVariant,
77 const MCAsmInfo &MAI,
78 const MCInstrInfo &MII,
79 const MCRegisterInfo &MRI) {
80 return new LoongArchInstPrinter(MAI, MII, MRI);
81}
82
83static MCTargetStreamer *
89
90static MCTargetStreamer *
95
96namespace {
97
98class LoongArchMCInstrAnalysis : public MCInstrAnalysis {
99 int64_t GPRState[31] = {};
100 std::bitset<31> GPRValidMask;
101
102 static bool isGPR(MCRegister Reg) {
103 return Reg >= LoongArch::R0 && Reg <= LoongArch::R31;
104 }
105
106 static unsigned getRegIndex(MCRegister Reg) {
107 assert(isGPR(Reg) && Reg != LoongArch::R0 && "Invalid GPR reg");
108 return Reg - LoongArch::R1;
109 }
110
111 void setGPRState(MCRegister Reg, std::optional<int64_t> Value) {
112 if (Reg == LoongArch::R0)
113 return;
114
115 auto Index = getRegIndex(Reg);
116
117 if (Value) {
118 GPRState[Index] = *Value;
119 GPRValidMask.set(Index);
120 } else {
121 GPRValidMask.reset(Index);
122 }
123 }
124
125 std::optional<int64_t> getGPRState(MCRegister Reg) const {
126 if (Reg == LoongArch::R0)
127 return 0;
128
129 auto Index = getRegIndex(Reg);
130
131 if (GPRValidMask.test(Index))
132 return GPRState[Index];
133 return std::nullopt;
134 }
135
136public:
137 explicit LoongArchMCInstrAnalysis(const MCInstrInfo *Info)
138 : MCInstrAnalysis(Info) {}
139
140 void resetState() override { GPRValidMask.reset(); }
141
142 void updateState(const MCInst &Inst, const MCSubtargetInfo *STI,
143 uint64_t Addr) override {
144 // Terminators mark the end of a basic block which means the sequentially
145 // next instruction will be the first of another basic block and the current
146 // state will typically not be valid anymore. For calls, we assume all
147 // registers may be clobbered by the callee (TODO: should we take the
148 // calling convention into account?).
149 if (isTerminator(Inst) || isCall(Inst)) {
150 resetState();
151 return;
152 }
153
154 switch (Inst.getOpcode()) {
155 default: {
156 // Clear the state of all defined registers for instructions that we don't
157 // explicitly support.
158 auto NumDefs = Info->get(Inst.getOpcode()).getNumDefs();
159 for (unsigned I = 0; I < NumDefs; ++I) {
160 auto DefReg = Inst.getOperand(I).getReg();
161 if (isGPR(DefReg))
162 setGPRState(DefReg, std::nullopt);
163 }
164 break;
165 }
166 case LoongArch::PCADDU18I:
167 setGPRState(
168 Inst.getOperand(0).getReg(),
169 Addr + SignExtend64<38>(
170 static_cast<uint64_t>(Inst.getOperand(1).getImm()) << 18));
171 break;
172 }
173 }
174
175 bool evaluateBranch(const MCInst &Inst, uint64_t Addr, uint64_t Size,
176 uint64_t &Target) const override {
177 unsigned NumOps = Inst.getNumOperands();
178 if ((isBranch(Inst) && !isIndirectBranch(Inst)) ||
179 Inst.getOpcode() == LoongArch::BL) {
180 Target = Addr + Inst.getOperand(NumOps - 1).getImm();
181 return true;
182 }
183
184 if (Inst.getOpcode() == LoongArch::JIRL) {
185 if (auto TargetRegState = getGPRState(Inst.getOperand(1).getReg())) {
186 Target = *TargetRegState + Inst.getOperand(2).getImm();
187 return true;
188 }
189 return false;
190 }
191
192 return false;
193 }
194
195 bool isTerminator(const MCInst &Inst) const override {
197 return true;
198
199 switch (Inst.getOpcode()) {
200 default:
201 return false;
202 case LoongArch::JIRL:
203 return Inst.getOperand(0).getReg() == LoongArch::R0;
204 }
205 }
206
207 bool isCall(const MCInst &Inst) const override {
208 if (MCInstrAnalysis::isCall(Inst))
209 return true;
210
211 switch (Inst.getOpcode()) {
212 default:
213 return false;
214 case LoongArch::JIRL:
215 return Inst.getOperand(0).getReg() != LoongArch::R0;
216 }
217 }
218
219 bool isReturn(const MCInst &Inst) const override {
221 return true;
222
223 switch (Inst.getOpcode()) {
224 default:
225 return false;
226 case LoongArch::JIRL:
227 return Inst.getOperand(0).getReg() == LoongArch::R0 &&
228 Inst.getOperand(1).getReg() == LoongArch::R1;
229 }
230 }
231
232 bool isBranch(const MCInst &Inst) const override {
234 return true;
235
236 switch (Inst.getOpcode()) {
237 default:
238 return false;
239 case LoongArch::JIRL:
240 return Inst.getOperand(0).getReg() == LoongArch::R0 &&
241 Inst.getOperand(1).getReg() != LoongArch::R1;
242 }
243 }
244
245 bool isUnconditionalBranch(const MCInst &Inst) const override {
247 return true;
248
249 switch (Inst.getOpcode()) {
250 default:
251 return false;
252 case LoongArch::JIRL:
253 return Inst.getOperand(0).getReg() == LoongArch::R0 &&
254 Inst.getOperand(1).getReg() != LoongArch::R1;
255 }
256 }
257
258 bool isIndirectBranch(const MCInst &Inst) const override {
260 return true;
261
262 switch (Inst.getOpcode()) {
263 default:
264 return false;
265 case LoongArch::JIRL:
266 return Inst.getOperand(0).getReg() == LoongArch::R0 &&
267 Inst.getOperand(1).getReg() != LoongArch::R1;
268 }
269 }
270};
271
272} // end namespace
273
275 return new LoongArchMCInstrAnalysis(Info);
276}
277
278namespace {
280 std::unique_ptr<MCAsmBackend> &&MAB,
281 std::unique_ptr<MCObjectWriter> &&MOW,
282 std::unique_ptr<MCCodeEmitter> &&MCE) {
283 return createLoongArchELFStreamer(Context, std::move(MAB), std::move(MOW),
284 std::move(MCE));
285}
286} // end namespace
287
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static bool isUnconditionalBranch(Instruction *Term)
Definition ADCE.cpp:215
#define X(NUM, ENUM, NAME)
Definition ELF.h:849
#define LLVM_ABI
Definition Compiler.h:213
#define LLVM_EXTERNAL_VISIBILITY
Definition Compiler.h:132
const size_t AbstractManglingParser< Derived, Alloc >::NumOps
static LVOptions Options
Definition LVOptions.cpp:25
static MCSubtargetInfo * createLoongArchMCSubtargetInfo(const Triple &TT, StringRef CPU, StringRef FS)
static MCTargetStreamer * createLoongArchObjectTargetStreamer(MCStreamer &S, const MCSubtargetInfo &STI)
static MCInstPrinter * createLoongArchMCInstPrinter(const Triple &T, unsigned SyntaxVariant, const MCAsmInfo &MAI, const MCInstrInfo &MII, const MCRegisterInfo &MRI)
static MCRegisterInfo * createLoongArchMCRegisterInfo(const Triple &TT)
static MCTargetStreamer * createLoongArchAsmTargetStreamer(MCStreamer &S, formatted_raw_ostream &OS, MCInstPrinter *InstPrint)
static MCAsmInfo * createLoongArchMCAsmInfo(const MCRegisterInfo &MRI, const Triple &TT, const MCTargetOptions &Options)
static MCInstrAnalysis * createLoongArchInstrAnalysis(const MCInstrInfo *Info)
static MCInstrInfo * createLoongArchMCInstrInfo()
LLVM_ABI LLVM_EXTERNAL_VISIBILITY void LLVMInitializeLoongArchTargetMC()
#define I(x, y, z)
Definition MD5.cpp:57
Register Reg
#define T
static bool isBranch(unsigned Opcode)
This class is intended to be used as a base class for asm properties and features specific to the tar...
Definition MCAsmInfo.h:64
void addInitialFrameState(const MCCFIInstruction &Inst)
Definition MCAsmInfo.cpp:53
static MCCFIInstruction cfiDefCfa(MCSymbol *L, unsigned Register, int64_t Offset, SMLoc Loc={})
.cfi_def_cfa defines a rule for computing CFA as: take address from Register and add Offset to it.
Definition MCDwarf.h:576
Context object for machine code objects.
Definition MCContext.h:83
This is an instance of a target assembly language printer that converts an MCInst to valid target ass...
unsigned getNumOperands() const
Definition MCInst.h:212
unsigned getOpcode() const
Definition MCInst.h:202
const MCOperand & getOperand(unsigned i) const
Definition MCInst.h:210
virtual bool isCall(const MCInst &Inst) const
virtual bool isBranch(const MCInst &Inst) const
virtual bool isUnconditionalBranch(const MCInst &Inst) const
virtual bool isTerminator(const MCInst &Inst) const
virtual bool isReturn(const MCInst &Inst) const
virtual bool isIndirectBranch(const MCInst &Inst) const
Interface to description of machine instruction set.
Definition MCInstrInfo.h:27
int64_t getImm() const
Definition MCInst.h:84
MCRegister getReg() const
Returns the register number.
Definition MCInst.h:73
MCRegisterInfo base class - We assume that the target defines a static array of MCRegisterDesc object...
virtual int64_t getDwarfRegNum(MCRegister Reg, bool isEH) const
Map a target register to an equivalent dwarf register number.
Streaming machine code generation interface.
Definition MCStreamer.h:221
Generic base class for all target subtargets.
const Triple & getTargetTriple() const
Target specific streamer interface.
Definition MCStreamer.h:94
StringRef - Represent a constant reference to a string, i.e.
Definition StringRef.h:55
Target - Wrapper for Target specific information.
Triple - Helper class for working with autoconf configuration names.
Definition Triple.h:47
bool isOSBinFormatELF() const
Tests whether the OS uses the ELF binary format.
Definition Triple.h:803
formatted_raw_ostream - A raw_ostream that wraps another one and keeps track of line and column posit...
This is an optimization pass for GlobalISel generic memory operations.
Definition Types.h:26
Target & getTheLoongArch64Target()
FunctionAddr VTableAddr Value
Definition InstrProf.h:137
MCCodeEmitter * createLoongArchMCCodeEmitter(const MCInstrInfo &MCII, MCContext &Ctx)
MCAsmBackend * createLoongArchAsmBackend(const Target &T, const MCSubtargetInfo &STI, const MCRegisterInfo &MRI, const MCTargetOptions &Options)
Target & getTheLoongArch32Target()
constexpr int64_t SignExtend64(uint64_t x)
Sign-extend the number in the bottom B bits of X to a 64-bit integer.
Definition MathExtras.h:572
MCELFStreamer * createLoongArchELFStreamer(MCContext &C, std::unique_ptr< MCAsmBackend > MAB, std::unique_ptr< MCObjectWriter > MOW, std::unique_ptr< MCCodeEmitter > MCE)
static void RegisterMCRegInfo(Target &T, Target::MCRegInfoCtorFnTy Fn)
RegisterMCRegInfo - Register a MCRegisterInfo implementation for the given target.
static void RegisterMCAsmBackend(Target &T, Target::MCAsmBackendCtorTy Fn)
RegisterMCAsmBackend - Register a MCAsmBackend implementation for the given target.
static void RegisterMCCodeEmitter(Target &T, Target::MCCodeEmitterCtorTy Fn)
RegisterMCCodeEmitter - Register a MCCodeEmitter implementation for the given target.
static void RegisterMCAsmInfo(Target &T, Target::MCAsmInfoCtorFnTy Fn)
RegisterMCAsmInfo - Register a MCAsmInfo implementation for the given target.
static void RegisterMCSubtargetInfo(Target &T, Target::MCSubtargetInfoCtorFnTy Fn)
RegisterMCSubtargetInfo - Register a MCSubtargetInfo implementation for the given target.
static void RegisterObjectTargetStreamer(Target &T, Target::ObjectTargetStreamerCtorTy Fn)
static void RegisterMCInstrAnalysis(Target &T, Target::MCInstrAnalysisCtorFnTy Fn)
RegisterMCInstrAnalysis - Register a MCInstrAnalysis implementation for the given target.
static void RegisterELFStreamer(Target &T, Target::ELFStreamerCtorTy Fn)
static void RegisterMCInstPrinter(Target &T, Target::MCInstPrinterCtorTy Fn)
RegisterMCInstPrinter - Register a MCInstPrinter implementation for the given target.
static void RegisterMCInstrInfo(Target &T, Target::MCInstrInfoCtorFnTy Fn)
RegisterMCInstrInfo - Register a MCInstrInfo implementation for the given target.
static void RegisterAsmTargetStreamer(Target &T, Target::AsmTargetStreamerCtorTy Fn)