IRObjectFile.cpp 4.9 KB

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