1 //===-- LanaiAsmBackend.cpp - Lanai Assembler Backend ---------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "LanaiFixupKinds.h"
11 #include "MCTargetDesc/LanaiMCTargetDesc.h"
12 #include "llvm/MC/MCAsmBackend.h"
13 #include "llvm/MC/MCAssembler.h"
14 #include "llvm/MC/MCDirectives.h"
15 #include "llvm/MC/MCELFObjectWriter.h"
16 #include "llvm/MC/MCFixupKindInfo.h"
17 #include "llvm/MC/MCObjectWriter.h"
18 #include "llvm/MC/MCSubtargetInfo.h"
19 #include "llvm/Support/ErrorHandling.h"
20 #include "llvm/Support/raw_ostream.h"
21 
22 using namespace llvm;
23 
24 // Prepare value for the target space
adjustFixupValue(unsigned Kind,uint64_t Value)25 static unsigned adjustFixupValue(unsigned Kind, uint64_t Value) {
26   switch (Kind) {
27   case FK_Data_1:
28   case FK_Data_2:
29   case FK_Data_4:
30   case FK_Data_8:
31     return Value;
32   case Lanai::FIXUP_LANAI_21:
33   case Lanai::FIXUP_LANAI_21_F:
34   case Lanai::FIXUP_LANAI_25:
35   case Lanai::FIXUP_LANAI_32:
36   case Lanai::FIXUP_LANAI_HI16:
37   case Lanai::FIXUP_LANAI_LO16:
38     return Value;
39   default:
40     llvm_unreachable("Unknown fixup kind!");
41   }
42 }
43 
44 namespace {
45 class LanaiAsmBackend : public MCAsmBackend {
46   Triple::OSType OSType;
47 
48 public:
LanaiAsmBackend(const Target & T,Triple::OSType OST)49   LanaiAsmBackend(const Target &T, Triple::OSType OST)
50       : MCAsmBackend(support::big), OSType(OST) {}
51 
52   void applyFixup(const MCAssembler &Asm, const MCFixup &Fixup,
53                   const MCValue &Target, MutableArrayRef<char> Data,
54                   uint64_t Value, bool IsResolved,
55                   const MCSubtargetInfo *STI) const override;
56 
57   std::unique_ptr<MCObjectTargetWriter>
58   createObjectTargetWriter() const override;
59 
60   // No instruction requires relaxation
fixupNeedsRelaxation(const MCFixup &,uint64_t,const MCRelaxableFragment *,const MCAsmLayout &) const61   bool fixupNeedsRelaxation(const MCFixup & /*Fixup*/, uint64_t /*Value*/,
62                             const MCRelaxableFragment * /*DF*/,
63                             const MCAsmLayout & /*Layout*/) const override {
64     return false;
65   }
66 
67   const MCFixupKindInfo &getFixupKindInfo(MCFixupKind Kind) const override;
68 
getNumFixupKinds() const69   unsigned getNumFixupKinds() const override {
70     return Lanai::NumTargetFixupKinds;
71   }
72 
mayNeedRelaxation(const MCInst &,const MCSubtargetInfo & STI) const73   bool mayNeedRelaxation(const MCInst & /*Inst*/,
74                          const MCSubtargetInfo &STI) const override {
75     return false;
76   }
77 
relaxInstruction(const MCInst &,const MCSubtargetInfo &,MCInst &) const78   void relaxInstruction(const MCInst & /*Inst*/,
79                         const MCSubtargetInfo & /*STI*/,
80                         MCInst & /*Res*/) const override {}
81 
82   bool writeNopData(raw_ostream &OS, uint64_t Count) const override;
83 };
84 
writeNopData(raw_ostream & OS,uint64_t Count) const85 bool LanaiAsmBackend::writeNopData(raw_ostream &OS, uint64_t Count) const {
86   if ((Count % 4) != 0)
87     return false;
88 
89   for (uint64_t i = 0; i < Count; i += 4)
90     OS.write("\x15\0\0\0", 4);
91 
92   return true;
93 }
94 
applyFixup(const MCAssembler & Asm,const MCFixup & Fixup,const MCValue & Target,MutableArrayRef<char> Data,uint64_t Value,bool,const MCSubtargetInfo *) const95 void LanaiAsmBackend::applyFixup(const MCAssembler &Asm, const MCFixup &Fixup,
96                                  const MCValue &Target,
97                                  MutableArrayRef<char> Data, uint64_t Value,
98                                  bool /*IsResolved*/,
99                                  const MCSubtargetInfo * /*STI*/) const {
100   MCFixupKind Kind = Fixup.getKind();
101   Value = adjustFixupValue(static_cast<unsigned>(Kind), Value);
102 
103   if (!Value)
104     return; // This value doesn't change the encoding
105 
106   // Where in the object and where the number of bytes that need
107   // fixing up
108   unsigned Offset = Fixup.getOffset();
109   unsigned NumBytes = (getFixupKindInfo(Kind).TargetSize + 7) / 8;
110   unsigned FullSize = 4;
111 
112   // Grab current value, if any, from bits.
113   uint64_t CurVal = 0;
114 
115   // Load instruction and apply value
116   for (unsigned i = 0; i != NumBytes; ++i) {
117     unsigned Idx = (FullSize - 1 - i);
118     CurVal |= static_cast<uint64_t>(static_cast<uint8_t>(Data[Offset + Idx]))
119               << (i * 8);
120   }
121 
122   uint64_t Mask =
123       (static_cast<uint64_t>(-1) >> (64 - getFixupKindInfo(Kind).TargetSize));
124   CurVal |= Value & Mask;
125 
126   // Write out the fixed up bytes back to the code/data bits.
127   for (unsigned i = 0; i != NumBytes; ++i) {
128     unsigned Idx = (FullSize - 1 - i);
129     Data[Offset + Idx] = static_cast<uint8_t>((CurVal >> (i * 8)) & 0xff);
130   }
131 }
132 
133 std::unique_ptr<MCObjectTargetWriter>
createObjectTargetWriter() const134 LanaiAsmBackend::createObjectTargetWriter() const {
135   return createLanaiELFObjectWriter(MCELFObjectTargetWriter::getOSABI(OSType));
136 }
137 
138 const MCFixupKindInfo &
getFixupKindInfo(MCFixupKind Kind) const139 LanaiAsmBackend::getFixupKindInfo(MCFixupKind Kind) const {
140   static const MCFixupKindInfo Infos[Lanai::NumTargetFixupKinds] = {
141       // This table *must* be in same the order of fixup_* kinds in
142       // LanaiFixupKinds.h.
143       // Note: The number of bits indicated here are assumed to be contiguous.
144       //   This does not hold true for LANAI_21 and LANAI_21_F which are applied
145       //   to bits 0x7cffff and 0x7cfffc, respectively. Since the 'bits' counts
146       //   here are used only for cosmetic purposes, we set the size to 16 bits
147       //   for these 21-bit relocation as llvm/lib/MC/MCAsmStreamer.cpp checks
148       //   no bits are set in the fixup range.
149       //
150       // name          offset bits flags
151       {"FIXUP_LANAI_NONE", 0, 32, 0},
152       {"FIXUP_LANAI_21", 16, 16 /*21*/, 0},
153       {"FIXUP_LANAI_21_F", 16, 16 /*21*/, 0},
154       {"FIXUP_LANAI_25", 7, 25, 0},
155       {"FIXUP_LANAI_32", 0, 32, 0},
156       {"FIXUP_LANAI_HI16", 16, 16, 0},
157       {"FIXUP_LANAI_LO16", 16, 16, 0}};
158 
159   if (Kind < FirstTargetFixupKind)
160     return MCAsmBackend::getFixupKindInfo(Kind);
161 
162   assert(unsigned(Kind - FirstTargetFixupKind) < getNumFixupKinds() &&
163          "Invalid kind!");
164   return Infos[Kind - FirstTargetFixupKind];
165 }
166 
167 } // namespace
168 
createLanaiAsmBackend(const Target & T,const MCSubtargetInfo & STI,const MCRegisterInfo &,const MCTargetOptions &)169 MCAsmBackend *llvm::createLanaiAsmBackend(const Target &T,
170                                           const MCSubtargetInfo &STI,
171                                           const MCRegisterInfo & /*MRI*/,
172                                           const MCTargetOptions & /*Options*/) {
173   const Triple &TT = STI.getTargetTriple();
174   if (!TT.isOSBinFormatELF())
175     llvm_unreachable("OS not supported");
176 
177   return new LanaiAsmBackend(T, TT.getOS());
178 }
179