Skip to content

Commit 702dac6

Browse files
author
George Rimar
committedApr 12, 2017
[DWARF] - Refactoring of DWARFContextInMemory implementation.
This change is basically relative to D31136, where I initially wanted to implement some relocations handling optimization which shows it can give significant boost. Though even without any caching algorithm looks code can have some cleanup at first. Refactoring separates out code for taking symbol address, used in relocations computation. Differential revision: https://reviews.llvm.org/D31747 llvm-svn: 300039
1 parent 6832fae commit 702dac6

File tree

1 file changed

+71
-64
lines changed

1 file changed

+71
-64
lines changed
 

‎llvm/lib/DebugInfo/DWARF/DWARFContext.cpp

+71-64
Original file line numberDiff line numberDiff line change
@@ -616,6 +616,66 @@ DWARFContext::getInliningInfoForAddress(uint64_t Address,
616616
return InliningInfo;
617617
}
618618

619+
static Error createError(const Twine &Reason, llvm::Error E) {
620+
return make_error<StringError>(Reason + toString(std::move(E)),
621+
inconvertibleErrorCode());
622+
}
623+
624+
/// Returns the address of symbol relocation used against. Used for futher
625+
/// relocations computation. Symbol's section load address is taken in account if
626+
/// LoadedObjectInfo interface is provided.
627+
static Expected<uint64_t> getSymbolAddress(const object::ObjectFile &Obj,
628+
const RelocationRef &Reloc,
629+
const LoadedObjectInfo *L) {
630+
uint64_t Ret = 0;
631+
object::section_iterator RSec = Obj.section_end();
632+
object::symbol_iterator Sym = Reloc.getSymbol();
633+
634+
// First calculate the address of the symbol or section as it appears
635+
// in the object file
636+
if (Sym != Obj.symbol_end()) {
637+
Expected<uint64_t> SymAddrOrErr = Sym->getAddress();
638+
if (!SymAddrOrErr)
639+
return createError("error: failed to compute symbol address: ",
640+
SymAddrOrErr.takeError());
641+
642+
// Also remember what section this symbol is in for later
643+
auto SectOrErr = Sym->getSection();
644+
if (!SectOrErr)
645+
return createError("error: failed to get symbol section: ",
646+
SectOrErr.takeError());
647+
648+
RSec = *SectOrErr;
649+
Ret = *SymAddrOrErr;
650+
} else if (auto *MObj = dyn_cast<MachOObjectFile>(&Obj)) {
651+
RSec = MObj->getRelocationSection(Reloc.getRawDataRefImpl());
652+
Ret = RSec->getAddress();
653+
}
654+
655+
// If we are given load addresses for the sections, we need to adjust:
656+
// SymAddr = (Address of Symbol Or Section in File) -
657+
// (Address of Section in File) +
658+
// (Load Address of Section)
659+
// RSec is now either the section being targeted or the section
660+
// containing the symbol being targeted. In either case,
661+
// we need to perform the same computation.
662+
if (L && RSec != Obj.section_end())
663+
if (uint64_t SectionLoadAddress = L->getSectionLoadAddress(*RSec))
664+
Ret += SectionLoadAddress - RSec->getAddress();
665+
return Ret;
666+
}
667+
668+
static bool isRelocScattered(const object::ObjectFile &Obj,
669+
const RelocationRef &Reloc) {
670+
if (!isa<MachOObjectFile>(&Obj))
671+
return false;
672+
// MachO also has relocations that point to sections and
673+
// scattered relocations.
674+
const MachOObjectFile *MachObj = cast<MachOObjectFile>(&Obj);
675+
auto RelocInfo = MachObj->getRelocation(Reloc.getRawDataRefImpl());
676+
return MachObj->isRelocationScattered(RelocInfo);
677+
}
678+
619679
DWARFContextInMemory::DWARFContextInMemory(const object::ObjectFile &Obj,
620680
const LoadedObjectInfo *L)
621681
: IsLittleEndian(Obj.isLittleEndian()),
@@ -721,81 +781,28 @@ DWARFContextInMemory::DWARFContextInMemory(const object::ObjectFile &Obj,
721781
if (Section.relocation_begin() != Section.relocation_end()) {
722782
uint64_t SectionSize = RelocatedSection->getSize();
723783
for (const RelocationRef &Reloc : Section.relocations()) {
724-
uint64_t Address = Reloc.getOffset();
725-
uint64_t Type = Reloc.getType();
726-
uint64_t SymAddr = 0;
727-
uint64_t SectionLoadAddress = 0;
728-
object::symbol_iterator Sym = Reloc.getSymbol();
729-
object::section_iterator RSec = Obj.section_end();
730-
731-
// First calculate the address of the symbol or section as it appears
732-
// in the objct file
733-
if (Sym != Obj.symbol_end()) {
734-
Expected<uint64_t> SymAddrOrErr = Sym->getAddress();
735-
if (!SymAddrOrErr) {
736-
std::string Buf;
737-
raw_string_ostream OS(Buf);
738-
logAllUnhandledErrors(SymAddrOrErr.takeError(), OS, "");
739-
OS.flush();
740-
errs() << "error: failed to compute symbol address: "
741-
<< Buf << '\n';
742-
continue;
743-
}
744-
SymAddr = *SymAddrOrErr;
745-
// Also remember what section this symbol is in for later
746-
auto SectOrErr = Sym->getSection();
747-
if (!SectOrErr) {
748-
std::string Buf;
749-
raw_string_ostream OS(Buf);
750-
logAllUnhandledErrors(SectOrErr.takeError(), OS, "");
751-
OS.flush();
752-
errs() << "error: failed to get symbol section: "
753-
<< Buf << '\n';
754-
continue;
755-
}
756-
RSec = *SectOrErr;
757-
} else if (auto *MObj = dyn_cast<MachOObjectFile>(&Obj)) {
758-
// MachO also has relocations that point to sections and
759-
// scattered relocations.
760-
auto RelocInfo = MObj->getRelocation(Reloc.getRawDataRefImpl());
761-
if (MObj->isRelocationScattered(RelocInfo)) {
762-
// FIXME: it's not clear how to correctly handle scattered
763-
// relocations.
764-
continue;
765-
} else {
766-
RSec = MObj->getRelocationSection(Reloc.getRawDataRefImpl());
767-
SymAddr = RSec->getAddress();
768-
}
769-
}
784+
// FIXME: it's not clear how to correctly handle scattered
785+
// relocations.
786+
if (isRelocScattered(Obj, Reloc))
787+
continue;
770788

771-
// If we are given load addresses for the sections, we need to adjust:
772-
// SymAddr = (Address of Symbol Or Section in File) -
773-
// (Address of Section in File) +
774-
// (Load Address of Section)
775-
if (L != nullptr && RSec != Obj.section_end()) {
776-
// RSec is now either the section being targeted or the section
777-
// containing the symbol being targeted. In either case,
778-
// we need to perform the same computation.
779-
StringRef SecName;
780-
RSec->getName(SecName);
781-
// dbgs() << "Name: '" << SecName
782-
// << "', RSec: " << RSec->getRawDataRefImpl()
783-
// << ", Section: " << Section.getRawDataRefImpl() << "\n";
784-
SectionLoadAddress = L->getSectionLoadAddress(*RSec);
785-
if (SectionLoadAddress != 0)
786-
SymAddr += SectionLoadAddress - RSec->getAddress();
789+
object::symbol_iterator RelSym = Reloc.getSymbol();
790+
Expected<uint64_t> SymAddrOrErr = getSymbolAddress(Obj, Reloc, L);
791+
if (!SymAddrOrErr) {
792+
errs() << toString(std::move(SymAddrOrErr.takeError())) << '\n';
793+
continue;
787794
}
788795

789796
object::RelocVisitor V(Obj);
790-
object::RelocToApply R(V.visit(Type, Reloc, SymAddr));
797+
object::RelocToApply R(V.visit(Reloc.getType(), Reloc, *SymAddrOrErr));
791798
if (V.error()) {
792799
SmallString<32> Name;
793800
Reloc.getTypeName(Name);
794801
errs() << "error: failed to compute relocation: "
795802
<< Name << "\n";
796803
continue;
797804
}
798-
805+
uint64_t Address = Reloc.getOffset();
799806
if (Address + R.Width > SectionSize) {
800807
errs() << "error: " << R.Width << "-byte relocation starting "
801808
<< Address << " bytes into section " << name << " which is "

0 commit comments

Comments
 (0)
Please sign in to comment.