Path: blob/main/contrib/llvm-project/llvm/lib/Target/LoongArch/MCTargetDesc/LoongArchMCTargetDesc.cpp
35294 views
//===-- LoongArchMCTargetDesc.cpp - LoongArch Target Descriptions ---------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//7//8// This file provides LoongArch specific target descriptions.9//10//===----------------------------------------------------------------------===//1112#include "LoongArchMCTargetDesc.h"13#include "LoongArchBaseInfo.h"14#include "LoongArchELFStreamer.h"15#include "LoongArchInstPrinter.h"16#include "LoongArchMCAsmInfo.h"17#include "TargetInfo/LoongArchTargetInfo.h"18#include "llvm/MC/MCAsmBackend.h"19#include "llvm/MC/MCAsmInfo.h"20#include "llvm/MC/MCCodeEmitter.h"21#include "llvm/MC/MCDwarf.h"22#include "llvm/MC/MCInstrAnalysis.h"23#include "llvm/MC/MCInstrInfo.h"24#include "llvm/MC/MCObjectWriter.h"25#include "llvm/MC/MCRegisterInfo.h"26#include "llvm/MC/MCSubtargetInfo.h"27#include "llvm/MC/TargetRegistry.h"28#include "llvm/Support/Compiler.h"2930#define GET_INSTRINFO_MC_DESC31#define ENABLE_INSTR_PREDICATE_VERIFIER32#include "LoongArchGenInstrInfo.inc"3334#define GET_REGINFO_MC_DESC35#include "LoongArchGenRegisterInfo.inc"3637#define GET_SUBTARGETINFO_MC_DESC38#include "LoongArchGenSubtargetInfo.inc"3940using namespace llvm;4142static MCRegisterInfo *createLoongArchMCRegisterInfo(const Triple &TT) {43MCRegisterInfo *X = new MCRegisterInfo();44InitLoongArchMCRegisterInfo(X, LoongArch::R1);45return X;46}4748static MCInstrInfo *createLoongArchMCInstrInfo() {49MCInstrInfo *X = new MCInstrInfo();50InitLoongArchMCInstrInfo(X);51return X;52}5354static MCSubtargetInfo *55createLoongArchMCSubtargetInfo(const Triple &TT, StringRef CPU, StringRef FS) {56if (CPU.empty() || CPU == "generic")57CPU = TT.isArch64Bit() ? "generic-la64" : "generic-la32";58return createLoongArchMCSubtargetInfoImpl(TT, CPU, /*TuneCPU*/ CPU, FS);59}6061static MCAsmInfo *createLoongArchMCAsmInfo(const MCRegisterInfo &MRI,62const Triple &TT,63const MCTargetOptions &Options) {64MCAsmInfo *MAI = new LoongArchMCAsmInfo(TT);6566// Initial state of the frame pointer is sp(r3).67MCRegister SP = MRI.getDwarfRegNum(LoongArch::R3, true);68MCCFIInstruction Inst = MCCFIInstruction::cfiDefCfa(nullptr, SP, 0);69MAI->addInitialFrameState(Inst);7071return MAI;72}7374static MCInstPrinter *createLoongArchMCInstPrinter(const Triple &T,75unsigned SyntaxVariant,76const MCAsmInfo &MAI,77const MCInstrInfo &MII,78const MCRegisterInfo &MRI) {79return new LoongArchInstPrinter(MAI, MII, MRI);80}8182static MCTargetStreamer *83createLoongArchObjectTargetStreamer(MCStreamer &S, const MCSubtargetInfo &STI) {84return STI.getTargetTriple().isOSBinFormatELF()85? new LoongArchTargetELFStreamer(S, STI)86: nullptr;87}8889namespace {9091class LoongArchMCInstrAnalysis : public MCInstrAnalysis {92public:93explicit LoongArchMCInstrAnalysis(const MCInstrInfo *Info)94: MCInstrAnalysis(Info) {}9596bool evaluateBranch(const MCInst &Inst, uint64_t Addr, uint64_t Size,97uint64_t &Target) const override {98unsigned NumOps = Inst.getNumOperands();99if ((isBranch(Inst) && !isIndirectBranch(Inst)) ||100Inst.getOpcode() == LoongArch::BL) {101Target = Addr + Inst.getOperand(NumOps - 1).getImm();102return true;103}104105return false;106}107108bool isTerminator(const MCInst &Inst) const override {109if (MCInstrAnalysis::isTerminator(Inst))110return true;111112switch (Inst.getOpcode()) {113default:114return false;115case LoongArch::JIRL:116return Inst.getOperand(0).getReg() == LoongArch::R0;117}118}119120bool isCall(const MCInst &Inst) const override {121if (MCInstrAnalysis::isCall(Inst))122return true;123124switch (Inst.getOpcode()) {125default:126return false;127case LoongArch::JIRL:128return Inst.getOperand(0).getReg() != LoongArch::R0;129}130}131132bool isReturn(const MCInst &Inst) const override {133if (MCInstrAnalysis::isReturn(Inst))134return true;135136switch (Inst.getOpcode()) {137default:138return false;139case LoongArch::JIRL:140return Inst.getOperand(0).getReg() == LoongArch::R0 &&141Inst.getOperand(1).getReg() == LoongArch::R1;142}143}144145bool isBranch(const MCInst &Inst) const override {146if (MCInstrAnalysis::isBranch(Inst))147return true;148149switch (Inst.getOpcode()) {150default:151return false;152case LoongArch::JIRL:153return Inst.getOperand(0).getReg() == LoongArch::R0 &&154Inst.getOperand(1).getReg() != LoongArch::R1;155}156}157158bool isUnconditionalBranch(const MCInst &Inst) const override {159if (MCInstrAnalysis::isUnconditionalBranch(Inst))160return true;161162switch (Inst.getOpcode()) {163default:164return false;165case LoongArch::JIRL:166return Inst.getOperand(0).getReg() == LoongArch::R0 &&167Inst.getOperand(1).getReg() != LoongArch::R1;168}169}170171bool isIndirectBranch(const MCInst &Inst) const override {172if (MCInstrAnalysis::isIndirectBranch(Inst))173return true;174175switch (Inst.getOpcode()) {176default:177return false;178case LoongArch::JIRL:179return Inst.getOperand(0).getReg() == LoongArch::R0 &&180Inst.getOperand(1).getReg() != LoongArch::R1;181}182}183};184185} // end namespace186187static MCInstrAnalysis *createLoongArchInstrAnalysis(const MCInstrInfo *Info) {188return new LoongArchMCInstrAnalysis(Info);189}190191namespace {192MCStreamer *createLoongArchELFStreamer(const Triple &T, MCContext &Context,193std::unique_ptr<MCAsmBackend> &&MAB,194std::unique_ptr<MCObjectWriter> &&MOW,195std::unique_ptr<MCCodeEmitter> &&MCE) {196return createLoongArchELFStreamer(Context, std::move(MAB), std::move(MOW),197std::move(MCE));198}199} // end namespace200201extern "C" LLVM_EXTERNAL_VISIBILITY void LLVMInitializeLoongArchTargetMC() {202for (Target *T : {&getTheLoongArch32Target(), &getTheLoongArch64Target()}) {203TargetRegistry::RegisterMCRegInfo(*T, createLoongArchMCRegisterInfo);204TargetRegistry::RegisterMCInstrInfo(*T, createLoongArchMCInstrInfo);205TargetRegistry::RegisterMCSubtargetInfo(*T, createLoongArchMCSubtargetInfo);206TargetRegistry::RegisterMCAsmInfo(*T, createLoongArchMCAsmInfo);207TargetRegistry::RegisterMCCodeEmitter(*T, createLoongArchMCCodeEmitter);208TargetRegistry::RegisterMCAsmBackend(*T, createLoongArchAsmBackend);209TargetRegistry::RegisterMCInstPrinter(*T, createLoongArchMCInstPrinter);210TargetRegistry::RegisterMCInstrAnalysis(*T, createLoongArchInstrAnalysis);211TargetRegistry::RegisterELFStreamer(*T, createLoongArchELFStreamer);212TargetRegistry::RegisterObjectTargetStreamer(213*T, createLoongArchObjectTargetStreamer);214}215}216217218