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(), OSType(OST) {}
51 
52   void applyFixup(const MCFixup &Fixup, char *Data, unsigned DataSize,
53                   uint64_t Value, bool IsPCRel) const override;
54 
55   MCObjectWriter *createObjectWriter(raw_pwrite_stream &OS) const override;
56 
57   // No instruction requires relaxation
fixupNeedsRelaxation(const MCFixup & Fixup,uint64_t Value,const MCRelaxableFragment * DF,const MCAsmLayout & Layout) const58   bool fixupNeedsRelaxation(const MCFixup &Fixup, uint64_t Value,
59                             const MCRelaxableFragment *DF,
60                             const MCAsmLayout &Layout) const override {
61     return false;
62   }
63 
64   const MCFixupKindInfo &getFixupKindInfo(MCFixupKind Kind) const override;
65 
getNumFixupKinds() const66   unsigned getNumFixupKinds() const override {
67     return Lanai::NumTargetFixupKinds;
68   }
69 
mayNeedRelaxation(const MCInst & Inst) const70   bool mayNeedRelaxation(const MCInst &Inst) const override { return false; }
71 
relaxInstruction(const MCInst & Inst,const MCSubtargetInfo & STI,MCInst & Res) const72   void relaxInstruction(const MCInst &Inst, const MCSubtargetInfo &STI,
73                         MCInst &Res) const override {}
74 
75   bool writeNopData(uint64_t Count, MCObjectWriter *OW) const override;
76 };
77 
writeNopData(uint64_t Count,MCObjectWriter * OW) const78 bool LanaiAsmBackend::writeNopData(uint64_t Count, MCObjectWriter *OW) const {
79   if ((Count % 4) != 0)
80     return false;
81 
82   for (uint64_t i = 0; i < Count; i += 4)
83     OW->write32(0x15000000);
84 
85   return true;
86 }
87 
applyFixup(const MCFixup & Fixup,char * Data,unsigned DataSize,uint64_t Value,bool IsPCRel) const88 void LanaiAsmBackend::applyFixup(const MCFixup &Fixup, char *Data,
89                                  unsigned DataSize, uint64_t Value,
90                                  bool IsPCRel) const {
91   MCFixupKind Kind = Fixup.getKind();
92   Value = adjustFixupValue(static_cast<unsigned>(Kind), Value);
93 
94   if (!Value)
95     return; // This value doesn't change the encoding
96 
97   // Where in the object and where the number of bytes that need
98   // fixing up
99   unsigned Offset = Fixup.getOffset();
100   unsigned NumBytes = (getFixupKindInfo(Kind).TargetSize + 7) / 8;
101   unsigned FullSize = 4;
102 
103   // Grab current value, if any, from bits.
104   uint64_t CurVal = 0;
105 
106   // Load instruction and apply value
107   for (unsigned i = 0; i != NumBytes; ++i) {
108     unsigned Idx = (FullSize - 1 - i);
109     CurVal |= static_cast<uint64_t>(static_cast<uint8_t>(Data[Offset + Idx]))
110               << (i * 8);
111   }
112 
113   uint64_t Mask =
114       (static_cast<uint64_t>(-1) >> (64 - getFixupKindInfo(Kind).TargetSize));
115   CurVal |= Value & Mask;
116 
117   // Write out the fixed up bytes back to the code/data bits.
118   for (unsigned i = 0; i != NumBytes; ++i) {
119     unsigned Idx = (FullSize - 1 - i);
120     Data[Offset + Idx] = static_cast<uint8_t>((CurVal >> (i * 8)) & 0xff);
121   }
122 }
123 
124 MCObjectWriter *
createObjectWriter(raw_pwrite_stream & OS) const125 LanaiAsmBackend::createObjectWriter(raw_pwrite_stream &OS) const {
126   return createLanaiELFObjectWriter(OS,
127                                     MCELFObjectTargetWriter::getOSABI(OSType));
128 }
129 
130 const MCFixupKindInfo &
getFixupKindInfo(MCFixupKind Kind) const131 LanaiAsmBackend::getFixupKindInfo(MCFixupKind Kind) const {
132   static const MCFixupKindInfo Infos[Lanai::NumTargetFixupKinds] = {
133       // This table *must* be in same the order of fixup_* kinds in
134       // LanaiFixupKinds.h.
135       // Note: The number of bits indicated here are assumed to be contiguous.
136       //   This does not hold true for LANAI_21 and LANAI_21_F which are applied
137       //   to bits 0x7cffff and 0x7cfffc, respectively. Since the 'bits' counts
138       //   here are used only for cosmetic purposes, we set the size to 16 bits
139       //   for these 21-bit relocation as llvm/lib/MC/MCAsmStreamer.cpp checks
140       //   no bits are set in the fixup range.
141       //
142       // name          offset bits flags
143       {"FIXUP_LANAI_NONE", 0, 32, 0},
144       {"FIXUP_LANAI_21", 16, 16 /*21*/, 0},
145       {"FIXUP_LANAI_21_F", 16, 16 /*21*/, 0},
146       {"FIXUP_LANAI_25", 7, 25, 0},
147       {"FIXUP_LANAI_32", 0, 32, 0},
148       {"FIXUP_LANAI_HI16", 16, 16, 0},
149       {"FIXUP_LANAI_LO16", 16, 16, 0}};
150 
151   if (Kind < FirstTargetFixupKind)
152     return MCAsmBackend::getFixupKindInfo(Kind);
153 
154   assert(unsigned(Kind - FirstTargetFixupKind) < getNumFixupKinds() &&
155          "Invalid kind!");
156   return Infos[Kind - FirstTargetFixupKind];
157 }
158 
159 } // namespace
160 
createLanaiAsmBackend(const Target & T,const MCRegisterInfo & MRI,const Triple & TheTriple,StringRef CPU)161 MCAsmBackend *llvm::createLanaiAsmBackend(const Target &T,
162                                           const MCRegisterInfo &MRI,
163                                           const Triple &TheTriple,
164                                           StringRef CPU) {
165   if (!TheTriple.isOSBinFormatELF())
166     llvm_unreachable("OS not supported");
167 
168   return new LanaiAsmBackend(T, TheTriple.getOS());
169 }
170