//===-- M68kMCCodeEmitter.cpp - Convert M68k code emitter -------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// /// /// \file /// This file contains defintions for M68k code emitter. /// //===----------------------------------------------------------------------===// #include "MCTargetDesc/M68kMCCodeEmitter.h" #include "MCTargetDesc/M68kBaseInfo.h" #include "MCTargetDesc/M68kFixupKinds.h" #include "MCTargetDesc/M68kMCTargetDesc.h" #include "llvm/MC/MCCodeEmitter.h" #include "llvm/MC/MCContext.h" #include "llvm/MC/MCExpr.h" #include "llvm/MC/MCInst.h" #include "llvm/MC/MCInstrInfo.h" #include "llvm/MC/MCRegisterInfo.h" #include "llvm/MC/MCSubtargetInfo.h" #include "llvm/MC/MCSymbol.h" #include "llvm/Support/Debug.h" #include "llvm/Support/EndianStream.h" #include "llvm/Support/raw_ostream.h" #include using namespace llvm; #define DEBUG_TYPE "m68k-mccodeemitter" namespace { class M68kMCCodeEmitter : public MCCodeEmitter { M68kMCCodeEmitter(const M68kMCCodeEmitter &) = delete; void operator=(const M68kMCCodeEmitter &) = delete; const MCInstrInfo &MCII; MCContext &Ctx; void getBinaryCodeForInstr(const MCInst &MI, SmallVectorImpl &Fixups, APInt &Inst, APInt &Scratch, const MCSubtargetInfo &STI) const; void getMachineOpValue(const MCInst &MI, const MCOperand &Op, unsigned InsertPos, APInt &Value, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const; template void encodeRelocImm(const MCInst &MI, unsigned OpIdx, unsigned InsertPos, APInt &Value, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const; template void encodePCRelImm(const MCInst &MI, unsigned OpIdx, unsigned InsertPos, APInt &Value, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const; public: M68kMCCodeEmitter(const MCInstrInfo &mcii, MCContext &ctx) : MCII(mcii), Ctx(ctx) {} ~M68kMCCodeEmitter() override {} void encodeInstruction(const MCInst &MI, raw_ostream &OS, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const override; }; } // end anonymous namespace #include "M68kGenMCCodeEmitter.inc" // Select the proper unsigned integer type from a bit size. template struct select_uint_t { using type = typename std::conditional< Size == 8, uint8_t, typename std::conditional< Size == 16, uint16_t, typename std::conditional::type>::type>::type; }; // On a LE host: // MSB LSB MSB LSB // | 0x12 0x34 | 0xAB 0xCD | -> | 0xAB 0xCD | 0x12 0x34 | // (On a BE host nothing changes) template static value_t swapWord(value_t Val) { const unsigned NumWords = sizeof(Val) / 2; if (NumWords <= 1) return Val; Val = support::endian::byte_swap(Val, support::big); value_t NewVal = 0; for (unsigned i = 0U; i != NumWords; ++i) { uint16_t Part = (Val >> (i * 16)) & 0xFFFF; Part = support::endian::byte_swap(Part, support::big); NewVal |= (Part << (i * 16)); } return NewVal; } // Figure out which byte we're at in big endian mode. template static unsigned getBytePosition(unsigned BitPos) { if (Size % 16) { return static_cast(BitPos / 8 + ((BitPos & 0b1111) < 8 ? 1 : -1)); } else { assert(!(BitPos & 0b1111) && "Not aligned to word boundary?"); return BitPos / 8; } } // We need special handlings for relocatable & pc-relative operands that are // larger than a word. // A M68k instruction is aligned by word (16 bits). That means, 32-bit // (& 64-bit) immediate values are separated into hi & lo words and placed // at lower & higher addresses, respectively. For immediate values that can // be easily expressed in TG, we explicitly rotate the word ordering like // this: // ``` // (ascend (slice "$imm", 31, 16), (slice "$imm", 15, 0)) // ``` // For operands that call into encoder functions, we need to use the `swapWord` // function to assure the correct word ordering on LE host. Note that // M68kMCCodeEmitter does massage _byte_ ordering of the final encoded // instruction but it assumes everything aligns on word boundaries. So things // will go wrong if we don't take care of the _word_ ordering here. template void M68kMCCodeEmitter::encodeRelocImm(const MCInst &MI, unsigned OpIdx, unsigned InsertPos, APInt &Value, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const { using value_t = typename select_uint_t::type; const MCOperand &MCO = MI.getOperand(OpIdx); if (MCO.isImm()) { Value |= swapWord(static_cast(MCO.getImm())); } else if (MCO.isExpr()) { const MCExpr *Expr = MCO.getExpr(); // Absolute address int64_t Addr; if (Expr->evaluateAsAbsolute(Addr)) { Value |= swapWord(static_cast(Addr)); return; } // Relocatable address unsigned InsertByte = getBytePosition(InsertPos); Fixups.push_back(MCFixup::create(InsertByte, Expr, getFixupForSize(Size, /*IsPCRel=*/false), MI.getLoc())); } } template void M68kMCCodeEmitter::encodePCRelImm(const MCInst &MI, unsigned OpIdx, unsigned InsertPos, APInt &Value, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const { const MCOperand &MCO = MI.getOperand(OpIdx); if (MCO.isImm()) { using value_t = typename select_uint_t::type; Value |= swapWord(static_cast(MCO.getImm())); } else if (MCO.isExpr()) { const MCExpr *Expr = MCO.getExpr(); unsigned InsertByte = getBytePosition(InsertPos); // Special handlings for sizes smaller than a word. if (Size < 16) { int LabelOffset = 0; if (InsertPos < 16) // If the patch point is at the first word, PC is pointing at the // next word. LabelOffset = InsertByte - 2; else if (InsertByte % 2) // Otherwise the PC is pointing at the first byte of this word. // So we need to consider the offset between PC and the fixup byte. LabelOffset = 1; if (LabelOffset) Expr = MCBinaryExpr::createAdd( Expr, MCConstantExpr::create(LabelOffset, Ctx), Ctx); } Fixups.push_back(MCFixup::create(InsertByte, Expr, getFixupForSize(Size, /*IsPCRel=*/true), MI.getLoc())); } } void M68kMCCodeEmitter::getMachineOpValue(const MCInst &MI, const MCOperand &Op, unsigned InsertPos, APInt &Value, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const { // Register if (Op.isReg()) { unsigned RegNum = Op.getReg(); const auto *RI = Ctx.getRegisterInfo(); Value |= RI->getEncodingValue(RegNum); // Setup the D/A bit if (M68kII::isAddressRegister(RegNum)) Value |= 0b1000; } else if (Op.isImm()) { // Immediate Value |= static_cast(Op.getImm()); } else if (Op.isExpr()) { // Absolute address int64_t Addr; if (!Op.getExpr()->evaluateAsAbsolute(Addr)) report_fatal_error("Unsupported asm expression. Only absolute address " "can be placed here."); Value |= static_cast(Addr); } else { llvm_unreachable("Unsupported operand type"); } } void M68kMCCodeEmitter::encodeInstruction(const MCInst &MI, raw_ostream &OS, SmallVectorImpl &Fixups, const MCSubtargetInfo &STI) const { unsigned Opcode = MI.getOpcode(); LLVM_DEBUG(dbgs() << "EncodeInstruction: " << MCII.getName(Opcode) << "(" << Opcode << ")\n"); // Try using the new method first. APInt EncodedInst(16, 0U); APInt Scratch(16, 0U); getBinaryCodeForInstr(MI, Fixups, EncodedInst, Scratch, STI); ArrayRef Data(EncodedInst.getRawData(), EncodedInst.getNumWords()); int64_t InstSize = EncodedInst.getBitWidth(); for (uint64_t Word : Data) { for (int i = 0; i < 4 && InstSize > 0; ++i, InstSize -= 16) { support::endian::write(OS, static_cast(Word), support::big); Word >>= 16; } } } MCCodeEmitter *llvm::createM68kMCCodeEmitter(const MCInstrInfo &MCII, MCContext &Ctx) { return new M68kMCCodeEmitter(MCII, Ctx); }