xref: /llvm-project/llvm/lib/Object/IRObjectFile.cpp (revision 3500f5e3bf1c5818913d5b6a675f3cae3daee808)
1 //===- IRObjectFile.cpp - IR object file implementation ---------*- C++ -*-===//
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 // Part of the IRObjectFile class implementation.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "llvm/Object/IRObjectFile.h"
15 #include "RecordStreamer.h"
16 #include "llvm/ADT/STLExtras.h"
17 #include "llvm/BinaryFormat/Magic.h"
18 #include "llvm/Bitcode/BitcodeReader.h"
19 #include "llvm/IR/GVMaterializer.h"
20 #include "llvm/IR/LLVMContext.h"
21 #include "llvm/IR/Mangler.h"
22 #include "llvm/IR/Module.h"
23 #include "llvm/MC/MCAsmInfo.h"
24 #include "llvm/MC/MCContext.h"
25 #include "llvm/MC/MCInstrInfo.h"
26 #include "llvm/MC/MCObjectFileInfo.h"
27 #include "llvm/MC/MCParser/MCAsmParser.h"
28 #include "llvm/MC/MCParser/MCTargetAsmParser.h"
29 #include "llvm/MC/MCRegisterInfo.h"
30 #include "llvm/MC/MCSubtargetInfo.h"
31 #include "llvm/Object/ObjectFile.h"
32 #include "llvm/Support/MemoryBuffer.h"
33 #include "llvm/Support/SourceMgr.h"
34 #include "llvm/Support/TargetRegistry.h"
35 #include "llvm/Support/raw_ostream.h"
36 using namespace llvm;
37 using namespace object;
38 
39 IRObjectFile::IRObjectFile(MemoryBufferRef Object,
40                            std::vector<std::unique_ptr<Module>> Mods)
41     : SymbolicFile(Binary::ID_IR, Object), Mods(std::move(Mods)) {
42   for (auto &M : this->Mods)
43     SymTab.addModule(M.get());
44 }
45 
46 IRObjectFile::~IRObjectFile() {}
47 
48 static ModuleSymbolTable::Symbol getSym(DataRefImpl &Symb) {
49   return *reinterpret_cast<ModuleSymbolTable::Symbol *>(Symb.p);
50 }
51 
52 void IRObjectFile::moveSymbolNext(DataRefImpl &Symb) const {
53   Symb.p += sizeof(ModuleSymbolTable::Symbol);
54 }
55 
56 std::error_code IRObjectFile::printSymbolName(raw_ostream &OS,
57                                               DataRefImpl Symb) const {
58   SymTab.printSymbolName(OS, getSym(Symb));
59   return std::error_code();
60 }
61 
62 uint32_t IRObjectFile::getSymbolFlags(DataRefImpl Symb) const {
63   return SymTab.getSymbolFlags(getSym(Symb));
64 }
65 
66 basic_symbol_iterator IRObjectFile::symbol_begin() const {
67   DataRefImpl Ret;
68   Ret.p = reinterpret_cast<uintptr_t>(SymTab.symbols().data());
69   return basic_symbol_iterator(BasicSymbolRef(Ret, this));
70 }
71 
72 basic_symbol_iterator IRObjectFile::symbol_end() const {
73   DataRefImpl Ret;
74   Ret.p = reinterpret_cast<uintptr_t>(SymTab.symbols().data() +
75                                       SymTab.symbols().size());
76   return basic_symbol_iterator(BasicSymbolRef(Ret, this));
77 }
78 
79 StringRef IRObjectFile::getTargetTriple() const {
80   // Each module must have the same target triple, so we arbitrarily access the
81   // first one.
82   return Mods[0]->getTargetTriple();
83 }
84 
85 Expected<MemoryBufferRef>
86 IRObjectFile::findBitcodeInObject(const ObjectFile &Obj) {
87   for (const SectionRef &Sec : Obj.sections()) {
88     if (Sec.isBitcode()) {
89       StringRef SecContents;
90       if (std::error_code EC = Sec.getContents(SecContents))
91         return errorCodeToError(EC);
92       return MemoryBufferRef(SecContents, Obj.getFileName());
93     }
94   }
95 
96   return errorCodeToError(object_error::bitcode_section_not_found);
97 }
98 
99 Expected<MemoryBufferRef>
100 IRObjectFile::findBitcodeInMemBuffer(MemoryBufferRef Object) {
101   file_magic Type = identify_magic(Object.getBuffer());
102   switch (Type) {
103   case file_magic::bitcode:
104     return Object;
105   case file_magic::elf_relocatable:
106   case file_magic::macho_object:
107   case file_magic::coff_object: {
108     Expected<std::unique_ptr<ObjectFile>> ObjFile =
109         ObjectFile::createObjectFile(Object, Type);
110     if (!ObjFile)
111       return ObjFile.takeError();
112     return findBitcodeInObject(*ObjFile->get());
113   }
114   default:
115     return errorCodeToError(object_error::invalid_file_type);
116   }
117 }
118 
119 Expected<std::unique_ptr<IRObjectFile>>
120 IRObjectFile::create(MemoryBufferRef Object, LLVMContext &Context) {
121   Expected<MemoryBufferRef> BCOrErr = findBitcodeInMemBuffer(Object);
122   if (!BCOrErr)
123     return BCOrErr.takeError();
124 
125   Expected<std::vector<BitcodeModule>> BMsOrErr =
126       getBitcodeModuleList(*BCOrErr);
127   if (!BMsOrErr)
128     return BMsOrErr.takeError();
129 
130   std::vector<std::unique_ptr<Module>> Mods;
131   for (auto BM : *BMsOrErr) {
132     Expected<std::unique_ptr<Module>> MOrErr =
133         BM.getLazyModule(Context, /*ShouldLazyLoadMetadata*/ true,
134                          /*IsImporting*/ false);
135     if (!MOrErr)
136       return MOrErr.takeError();
137 
138     Mods.push_back(std::move(*MOrErr));
139   }
140 
141   return std::unique_ptr<IRObjectFile>(
142       new IRObjectFile(*BCOrErr, std::move(Mods)));
143 }
144 
145 Expected<IRSymtabFile> object::readIRSymtab(MemoryBufferRef MBRef) {
146   IRSymtabFile F;
147   Expected<MemoryBufferRef> BCOrErr =
148       IRObjectFile::findBitcodeInMemBuffer(MBRef);
149   if (!BCOrErr)
150     return BCOrErr.takeError();
151 
152   Expected<BitcodeFileContents> BFCOrErr = getBitcodeFileContents(*BCOrErr);
153   if (!BFCOrErr)
154     return BFCOrErr.takeError();
155 
156   Expected<irsymtab::FileContents> FCOrErr = irsymtab::readBitcode(*BFCOrErr);
157   if (!FCOrErr)
158     return FCOrErr.takeError();
159 
160   F.Mods = std::move(BFCOrErr->Mods);
161   F.Symtab = std::move(FCOrErr->Symtab);
162   F.Strtab = std::move(FCOrErr->Strtab);
163   F.TheReader = std::move(FCOrErr->TheReader);
164   return std::move(F);
165 }
166