diff --git a/llvm/lib/ExecutionEngine/JITLink/ELF_riscv.cpp b/llvm/lib/ExecutionEngine/JITLink/ELF_riscv.cpp --- a/llvm/lib/ExecutionEngine/JITLink/ELF_riscv.cpp +++ b/llvm/lib/ExecutionEngine/JITLink/ELF_riscv.cpp @@ -286,93 +286,63 @@ } Error addRelocations() override { + LLVM_DEBUG(dbgs() << "Processing relocations:\n"); using Base = ELFLinkGraphBuilder; - LLVM_DEBUG(dbgs() << "Adding relocations\n"); - - // TODO a partern is forming of iterate some sections but only give me - // ones I am interested, I should abstract that concept some where - for (auto &SecRef : Base::Sections) { - if (SecRef.sh_type != ELF::SHT_RELA && SecRef.sh_type != ELF::SHT_REL) - continue; - auto RelSectName = Base::Obj.getSectionName(SecRef); - if (!RelSectName) - return RelSectName.takeError(); - - LLVM_DEBUG({ - dbgs() << "Adding relocations from section " << *RelSectName << "\n"; - }); - - auto UpdateSection = Base::Obj.getSection(SecRef.sh_info); - if (!UpdateSection) - return UpdateSection.takeError(); - - auto UpdateSectionName = Base::Obj.getSectionName(**UpdateSection); - if (!UpdateSectionName) - return UpdateSectionName.takeError(); - // Don't process relocations for debug sections. - if (Base::isDwarfSection(*UpdateSectionName)) { - LLVM_DEBUG({ - dbgs() << " Target is dwarf section " << *UpdateSectionName - << ". Skipping.\n"; - }); - continue; - } else - LLVM_DEBUG({ - dbgs() << " For target section " << *UpdateSectionName << "\n"; - }); - - auto *JITSection = Base::G->findSectionByName(*UpdateSectionName); - if (!JITSection) - return make_error( - "Refencing a section that wasn't added to graph" + - *UpdateSectionName, - llvm::inconvertibleErrorCode()); - - auto Relocations = Base::Obj.relas(SecRef); - if (!Relocations) - return Relocations.takeError(); - - for (const auto &Rela : *Relocations) { - auto Type = Rela.getType(false); - - LLVM_DEBUG({ - dbgs() << "Relocation Type: " << Type << "\n" - << "Name: " << Base::Obj.getRelocationTypeName(Type) << "\n"; - }); - - auto SymbolIndex = Rela.getSymbol(false); - auto Symbol = Base::Obj.getRelocationSymbol(Rela, Base::SymTabSec); - if (!Symbol) - return Symbol.takeError(); - - auto BlockToFix = *(JITSection->blocks().begin()); - auto *TargetSymbol = Base::getGraphSymbol(SymbolIndex); - - if (!TargetSymbol) { - return make_error( - "Could not find symbol at given index, did you add it to " - "JITSymbolTable? index: " + - std::to_string(SymbolIndex) + ", shndx: " + - std::to_string((*Symbol)->st_shndx) + " Size of table: " + - std::to_string(Base::GraphSymbols.size()), - llvm::inconvertibleErrorCode()); - } - int64_t Addend = Rela.r_addend; - JITTargetAddress FixupAddress = - (*UpdateSection)->sh_addr + Rela.r_offset; - - LLVM_DEBUG({ - dbgs() << "Processing relocation at " - << format("0x%016" PRIx64, FixupAddress) << "\n"; - }); - auto Kind = getRelocationKind(Type); - if (!Kind) - return Kind.takeError(); - - BlockToFix->addEdge(*Kind, FixupAddress - BlockToFix->getAddress(), - *TargetSymbol, Addend); - } + + for (const auto &RelSect : Base::Sections) { + auto Process = [&](const auto &Rel, const auto &Target, Section &GS) { + // Sanity check the section we read relocation entries from. + if (RelSect.sh_type == ELF::SHT_REL) + return make_error( + "No SHT_REL in valid x64 ELF object files", + inconvertibleErrorCode()); + + return addSingleRelocation(Rel, Target, GS); + }; + + if (Error Err = Base::forEachRelocation(RelSect, std::move(Process))) + return Err; } + + return Error::success(); + } + + Error addSingleRelocation(const typename ELFT::Rela &Rel, + const typename ELFT::Shdr &FixupSect, + Section &GraphSection) { + using Base = ELFLinkGraphBuilder; + + uint32_t SymbolIndex = Rel.getSymbol(false); + auto ObjSymbol = Base::Obj.getRelocationSymbol(Rel, Base::SymTabSec); + if (!ObjSymbol) + return ObjSymbol.takeError(); + + Symbol *GraphSymbol = Base::getGraphSymbol(SymbolIndex); + if (!GraphSymbol) + return make_error( + formatv("Could not find symbol at given index, did you add it to " + "JITSymbolTable? index: {0}, shndx: {1} Size of table: {2}", + SymbolIndex, (*ObjSymbol)->st_shndx, + Base::GraphSymbols.size()), + inconvertibleErrorCode()); + + uint32_t Type = Rel.getType(false); + Expected Kind = getRelocationKind(Type); + if (!Kind) + return Kind.takeError(); + + int64_t Addend = Rel.r_addend; + Block *BlockToFix = *(GraphSection.blocks().begin()); + JITTargetAddress FixupAddress = FixupSect.sh_addr + Rel.r_offset; + Edge::OffsetT Offset = FixupAddress - BlockToFix->getAddress(); + Edge GE(*Kind, Offset, *GraphSymbol, Addend); + LLVM_DEBUG({ + dbgs() << " "; + printEdge(dbgs(), *BlockToFix, GE, riscv::getEdgeKindName(*Kind)); + dbgs() << "\n"; + }); + + BlockToFix->addEdge(std::move(GE)); return Error::success(); }