| File: | build/source/llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp |
| Warning: | line 199, column 34 Called C++ object pointer is null |
Press '?' to see keyboard shortcuts
Keyboard shortcuts:
| 1 | //===-- SparcMCCodeEmitter.cpp - Convert Sparc 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 SparcMCCodeEmitter class. | |||
| 10 | // | |||
| 11 | //===----------------------------------------------------------------------===// | |||
| 12 | ||||
| 13 | #include "MCTargetDesc/SparcFixupKinds.h" | |||
| 14 | #include "SparcMCExpr.h" | |||
| 15 | #include "SparcMCTargetDesc.h" | |||
| 16 | #include "llvm/ADT/SmallVector.h" | |||
| 17 | #include "llvm/ADT/Statistic.h" | |||
| 18 | #include "llvm/MC/MCAsmInfo.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/MCObjectFileInfo.h" | |||
| 26 | #include "llvm/MC/MCRegisterInfo.h" | |||
| 27 | #include "llvm/MC/MCSubtargetInfo.h" | |||
| 28 | #include "llvm/MC/MCSymbol.h" | |||
| 29 | #include "llvm/MC/SubtargetFeature.h" | |||
| 30 | #include "llvm/Support/Casting.h" | |||
| 31 | #include "llvm/Support/Endian.h" | |||
| 32 | #include "llvm/Support/EndianStream.h" | |||
| 33 | #include "llvm/Support/ErrorHandling.h" | |||
| 34 | #include "llvm/Support/raw_ostream.h" | |||
| 35 | #include <cassert> | |||
| 36 | #include <cstdint> | |||
| 37 | ||||
| 38 | using namespace llvm; | |||
| 39 | ||||
| 40 | #define DEBUG_TYPE"mccodeemitter" "mccodeemitter" | |||
| 41 | ||||
| 42 | STATISTIC(MCNumEmitted, "Number of MC instructions emitted")static llvm::Statistic MCNumEmitted = {"mccodeemitter", "MCNumEmitted" , "Number of MC instructions emitted"}; | |||
| 43 | ||||
| 44 | namespace { | |||
| 45 | ||||
| 46 | class SparcMCCodeEmitter : public MCCodeEmitter { | |||
| 47 | MCContext &Ctx; | |||
| 48 | ||||
| 49 | public: | |||
| 50 | SparcMCCodeEmitter(const MCInstrInfo &, MCContext &ctx) | |||
| 51 | : Ctx(ctx) {} | |||
| 52 | SparcMCCodeEmitter(const SparcMCCodeEmitter &) = delete; | |||
| 53 | SparcMCCodeEmitter &operator=(const SparcMCCodeEmitter &) = delete; | |||
| 54 | ~SparcMCCodeEmitter() override = default; | |||
| 55 | ||||
| 56 | void encodeInstruction(const MCInst &MI, raw_ostream &OS, | |||
| 57 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 58 | const MCSubtargetInfo &STI) const override; | |||
| 59 | ||||
| 60 | // getBinaryCodeForInstr - TableGen'erated function for getting the | |||
| 61 | // binary encoding for an instruction. | |||
| 62 | uint64_t getBinaryCodeForInstr(const MCInst &MI, | |||
| 63 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 64 | const MCSubtargetInfo &STI) const; | |||
| 65 | ||||
| 66 | /// getMachineOpValue - Return binary encoding of operand. If the machine | |||
| 67 | /// operand requires relocation, record the relocation and return zero. | |||
| 68 | unsigned getMachineOpValue(const MCInst &MI, const MCOperand &MO, | |||
| 69 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 70 | const MCSubtargetInfo &STI) const; | |||
| 71 | unsigned getCallTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 72 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 73 | const MCSubtargetInfo &STI) const; | |||
| 74 | unsigned getBranchTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 75 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 76 | const MCSubtargetInfo &STI) const; | |||
| 77 | unsigned getSImm13OpValue(const MCInst &MI, unsigned OpNo, | |||
| 78 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 79 | const MCSubtargetInfo &STI) const; | |||
| 80 | unsigned getBranchPredTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 81 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 82 | const MCSubtargetInfo &STI) const; | |||
| 83 | unsigned getBranchOnRegTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 84 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 85 | const MCSubtargetInfo &STI) const; | |||
| 86 | }; | |||
| 87 | ||||
| 88 | } // end anonymous namespace | |||
| 89 | ||||
| 90 | void SparcMCCodeEmitter::encodeInstruction(const MCInst &MI, raw_ostream &OS, | |||
| 91 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 92 | const MCSubtargetInfo &STI) const { | |||
| 93 | unsigned Bits = getBinaryCodeForInstr(MI, Fixups, STI); | |||
| 94 | support::endian::write(OS, Bits, | |||
| 95 | Ctx.getAsmInfo()->isLittleEndian() ? support::little | |||
| 96 | : support::big); | |||
| 97 | ||||
| 98 | // Some instructions have phantom operands that only contribute a fixup entry. | |||
| 99 | unsigned SymOpNo = 0; | |||
| 100 | switch (MI.getOpcode()) { | |||
| 101 | default: break; | |||
| 102 | case SP::TLS_CALL: SymOpNo = 1; break; | |||
| 103 | case SP::GDOP_LDrr: | |||
| 104 | case SP::GDOP_LDXrr: | |||
| 105 | case SP::TLS_ADDrr: | |||
| 106 | case SP::TLS_ADDXrr: | |||
| 107 | case SP::TLS_LDrr: | |||
| 108 | case SP::TLS_LDXrr: SymOpNo = 3; break; | |||
| 109 | } | |||
| 110 | if (SymOpNo != 0) { | |||
| 111 | const MCOperand &MO = MI.getOperand(SymOpNo); | |||
| 112 | uint64_t op = getMachineOpValue(MI, MO, Fixups, STI); | |||
| 113 | assert(op == 0 && "Unexpected operand value!")(static_cast <bool> (op == 0 && "Unexpected operand value!" ) ? void (0) : __assert_fail ("op == 0 && \"Unexpected operand value!\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 113, __extension__ __PRETTY_FUNCTION__)); | |||
| 114 | (void)op; // suppress warning. | |||
| 115 | } | |||
| 116 | ||||
| 117 | ++MCNumEmitted; // Keep track of the # of mi's emitted. | |||
| 118 | } | |||
| 119 | ||||
| 120 | unsigned SparcMCCodeEmitter:: | |||
| 121 | getMachineOpValue(const MCInst &MI, const MCOperand &MO, | |||
| 122 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 123 | const MCSubtargetInfo &STI) const { | |||
| 124 | if (MO.isReg()) | |||
| 125 | return Ctx.getRegisterInfo()->getEncodingValue(MO.getReg()); | |||
| 126 | ||||
| 127 | if (MO.isImm()) | |||
| 128 | return MO.getImm(); | |||
| 129 | ||||
| 130 | assert(MO.isExpr())(static_cast <bool> (MO.isExpr()) ? void (0) : __assert_fail ("MO.isExpr()", "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 130, __extension__ __PRETTY_FUNCTION__)); | |||
| 131 | const MCExpr *Expr = MO.getExpr(); | |||
| 132 | if (const SparcMCExpr *SExpr = dyn_cast<SparcMCExpr>(Expr)) { | |||
| 133 | MCFixupKind Kind = (MCFixupKind)SExpr->getFixupKind(); | |||
| 134 | Fixups.push_back(MCFixup::create(0, Expr, Kind)); | |||
| 135 | return 0; | |||
| 136 | } | |||
| 137 | ||||
| 138 | int64_t Res; | |||
| 139 | if (Expr->evaluateAsAbsolute(Res)) | |||
| 140 | return Res; | |||
| 141 | ||||
| 142 | llvm_unreachable("Unhandled expression!")::llvm::llvm_unreachable_internal("Unhandled expression!", "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 142); | |||
| 143 | return 0; | |||
| 144 | } | |||
| 145 | ||||
| 146 | unsigned | |||
| 147 | SparcMCCodeEmitter::getSImm13OpValue(const MCInst &MI, unsigned OpNo, | |||
| 148 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 149 | const MCSubtargetInfo &STI) const { | |||
| 150 | const MCOperand &MO = MI.getOperand(OpNo); | |||
| 151 | ||||
| 152 | if (MO.isImm()) | |||
| 153 | return MO.getImm(); | |||
| 154 | ||||
| 155 | assert(MO.isExpr() &&(static_cast <bool> (MO.isExpr() && "getSImm13OpValue expects only expressions or an immediate" ) ? void (0) : __assert_fail ("MO.isExpr() && \"getSImm13OpValue expects only expressions or an immediate\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 156, __extension__ __PRETTY_FUNCTION__)) | |||
| 156 | "getSImm13OpValue expects only expressions or an immediate")(static_cast <bool> (MO.isExpr() && "getSImm13OpValue expects only expressions or an immediate" ) ? void (0) : __assert_fail ("MO.isExpr() && \"getSImm13OpValue expects only expressions or an immediate\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 156, __extension__ __PRETTY_FUNCTION__)); | |||
| 157 | ||||
| 158 | const MCExpr *Expr = MO.getExpr(); | |||
| 159 | ||||
| 160 | // Constant value, no fixup is needed | |||
| 161 | if (const MCConstantExpr *CE = dyn_cast<MCConstantExpr>(Expr)) | |||
| 162 | return CE->getValue(); | |||
| 163 | ||||
| 164 | MCFixupKind Kind; | |||
| 165 | if (const SparcMCExpr *SExpr = dyn_cast<SparcMCExpr>(Expr)) { | |||
| 166 | Kind = MCFixupKind(SExpr->getFixupKind()); | |||
| 167 | } else { | |||
| 168 | bool IsPic = Ctx.getObjectFileInfo()->isPositionIndependent(); | |||
| 169 | Kind = IsPic ? MCFixupKind(Sparc::fixup_sparc_got13) | |||
| 170 | : MCFixupKind(Sparc::fixup_sparc_13); | |||
| 171 | } | |||
| 172 | ||||
| 173 | Fixups.push_back(MCFixup::create(0, Expr, Kind)); | |||
| 174 | return 0; | |||
| 175 | } | |||
| 176 | ||||
| 177 | unsigned SparcMCCodeEmitter:: | |||
| 178 | getCallTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 179 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 180 | const MCSubtargetInfo &STI) const { | |||
| 181 | const MCOperand &MO = MI.getOperand(OpNo); | |||
| 182 | const MCExpr *Expr = MO.getExpr(); | |||
| 183 | const SparcMCExpr *SExpr = dyn_cast<SparcMCExpr>(Expr); | |||
| ||||
| 184 | ||||
| 185 | if (MI.getOpcode() == SP::TLS_CALL) { | |||
| 186 | // No fixups for __tls_get_addr. Will emit for fixups for tls_symbol in | |||
| 187 | // encodeInstruction. | |||
| 188 | #ifndef NDEBUG | |||
| 189 | // Verify that the callee is actually __tls_get_addr. | |||
| 190 | assert(SExpr && SExpr->getSubExpr()->getKind() == MCExpr::SymbolRef &&(static_cast <bool> (SExpr && SExpr->getSubExpr ()->getKind() == MCExpr::SymbolRef && "Unexpected expression in TLS_CALL" ) ? void (0) : __assert_fail ("SExpr && SExpr->getSubExpr()->getKind() == MCExpr::SymbolRef && \"Unexpected expression in TLS_CALL\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 191, __extension__ __PRETTY_FUNCTION__)) | |||
| 191 | "Unexpected expression in TLS_CALL")(static_cast <bool> (SExpr && SExpr->getSubExpr ()->getKind() == MCExpr::SymbolRef && "Unexpected expression in TLS_CALL" ) ? void (0) : __assert_fail ("SExpr && SExpr->getSubExpr()->getKind() == MCExpr::SymbolRef && \"Unexpected expression in TLS_CALL\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 191, __extension__ __PRETTY_FUNCTION__)); | |||
| 192 | const MCSymbolRefExpr *SymExpr = cast<MCSymbolRefExpr>(SExpr->getSubExpr()); | |||
| 193 | assert(SymExpr->getSymbol().getName() == "__tls_get_addr" &&(static_cast <bool> (SymExpr->getSymbol().getName() == "__tls_get_addr" && "Unexpected function for TLS_CALL" ) ? void (0) : __assert_fail ("SymExpr->getSymbol().getName() == \"__tls_get_addr\" && \"Unexpected function for TLS_CALL\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 194, __extension__ __PRETTY_FUNCTION__)) | |||
| 194 | "Unexpected function for TLS_CALL")(static_cast <bool> (SymExpr->getSymbol().getName() == "__tls_get_addr" && "Unexpected function for TLS_CALL" ) ? void (0) : __assert_fail ("SymExpr->getSymbol().getName() == \"__tls_get_addr\" && \"Unexpected function for TLS_CALL\"" , "llvm/lib/Target/Sparc/MCTargetDesc/SparcMCCodeEmitter.cpp" , 194, __extension__ __PRETTY_FUNCTION__)); | |||
| 195 | #endif | |||
| 196 | return 0; | |||
| 197 | } | |||
| 198 | ||||
| 199 | MCFixupKind Kind = MCFixupKind(SExpr->getFixupKind()); | |||
| ||||
| 200 | Fixups.push_back(MCFixup::create(0, Expr, Kind)); | |||
| 201 | return 0; | |||
| 202 | } | |||
| 203 | ||||
| 204 | unsigned SparcMCCodeEmitter:: | |||
| 205 | getBranchTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 206 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 207 | const MCSubtargetInfo &STI) const { | |||
| 208 | const MCOperand &MO = MI.getOperand(OpNo); | |||
| 209 | if (MO.isReg() || MO.isImm()) | |||
| 210 | return getMachineOpValue(MI, MO, Fixups, STI); | |||
| 211 | ||||
| 212 | Fixups.push_back(MCFixup::create(0, MO.getExpr(), | |||
| 213 | (MCFixupKind)Sparc::fixup_sparc_br22)); | |||
| 214 | return 0; | |||
| 215 | } | |||
| 216 | ||||
| 217 | unsigned SparcMCCodeEmitter:: | |||
| 218 | getBranchPredTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 219 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 220 | const MCSubtargetInfo &STI) const { | |||
| 221 | const MCOperand &MO = MI.getOperand(OpNo); | |||
| 222 | if (MO.isReg() || MO.isImm()) | |||
| 223 | return getMachineOpValue(MI, MO, Fixups, STI); | |||
| 224 | ||||
| 225 | Fixups.push_back(MCFixup::create(0, MO.getExpr(), | |||
| 226 | (MCFixupKind)Sparc::fixup_sparc_br19)); | |||
| 227 | return 0; | |||
| 228 | } | |||
| 229 | ||||
| 230 | unsigned SparcMCCodeEmitter:: | |||
| 231 | getBranchOnRegTargetOpValue(const MCInst &MI, unsigned OpNo, | |||
| 232 | SmallVectorImpl<MCFixup> &Fixups, | |||
| 233 | const MCSubtargetInfo &STI) const { | |||
| 234 | const MCOperand &MO = MI.getOperand(OpNo); | |||
| 235 | if (MO.isReg() || MO.isImm()) | |||
| 236 | return getMachineOpValue(MI, MO, Fixups, STI); | |||
| 237 | ||||
| 238 | Fixups.push_back( | |||
| 239 | MCFixup::create(0, MO.getExpr(), (MCFixupKind)Sparc::fixup_sparc_br16)); | |||
| 240 | ||||
| 241 | return 0; | |||
| 242 | } | |||
| 243 | ||||
| 244 | #include "SparcGenMCCodeEmitter.inc" | |||
| 245 | ||||
| 246 | MCCodeEmitter *llvm::createSparcMCCodeEmitter(const MCInstrInfo &MCII, | |||
| 247 | MCContext &Ctx) { | |||
| 248 | return new SparcMCCodeEmitter(MCII, Ctx); | |||
| 249 | } |