14#include <unordered_map>
40#include "llvm/ADT/IntervalMap.h"
41#include "llvm/ADT/PointerUnion.h"
42#include "llvm/ADT/StringRef.h"
43#include "llvm/BinaryFormat/ELF.h"
44#include "llvm/Object/Decompressor.h"
45#include "llvm/Support/ARMBuildAttributes.h"
46#include "llvm/Support/CRC.h"
47#include "llvm/Support/FormatVariadic.h"
48#include "llvm/Support/MathExtras.h"
49#include "llvm/Support/MemoryBuffer.h"
50#include "llvm/Support/MipsABIFlags.h"
51#include "llvm/Support/RISCVAttributes.h"
52#include "llvm/TargetParser/RISCVISAInfo.h"
53#include "llvm/TargetParser/SubtargetFeature.h"
55#define CASE_AND_STREAM(s, def, width) \
57 s->Printf("%-*s", width, #def); \
63using namespace llvm::ELF;
110 ELFRelocation(
unsigned type);
116 static unsigned RelocType32(
const ELFRelocation &rel);
118 static unsigned RelocType64(
const ELFRelocation &rel);
120 static unsigned RelocSymbol32(
const ELFRelocation &rel);
122 static unsigned RelocSymbol64(
const ELFRelocation &rel);
124 static elf_addr RelocOffset32(
const ELFRelocation &rel);
126 static elf_addr RelocOffset64(
const ELFRelocation &rel);
128 static elf_sxword RelocAddend32(
const ELFRelocation &rel);
130 static elf_sxword RelocAddend64(
const ELFRelocation &rel);
132 bool IsRela() {
return (llvm::isa<ELFRela *>(reloc)); }
135 typedef llvm::PointerUnion<ELFRel *, ELFRela *> RelocUnion;
145 assert(lhs_module_parent && rhs_module_parent);
148 if (lhs->GetFileAddress() != rhs->GetFileAddress())
149 lhs_module_parent->ReportWarning(
150 "mismatch addresses for section {0} when "
151 "merging with {1}, expected: {2:x}, "
153 lhs->GetTypeAsCString(),
154 rhs_module_parent->GetFileSpec().GetPathAsConstString().GetCString(),
155 lhs->GetFileAddress(), rhs->GetFileAddress());
160 return rhs->GetFileSize() > lhs->GetFileSize() ? rhs : lhs;
164ELFRelocation::ELFRelocation(
unsigned type) {
165 if (type == DT_REL || type == SHT_REL)
166 reloc =
new ELFRel();
167 else if (type == DT_RELA || type == SHT_RELA)
168 reloc =
new ELFRela();
170 assert(
false &&
"unexpected relocation type");
171 reloc =
static_cast<ELFRel *
>(
nullptr);
175ELFRelocation::~ELFRelocation() {
176 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(reloc))
179 delete llvm::cast<ELFRela *>(reloc);
182bool ELFRelocation::Parse(
const lldb_private::DataExtractor &data,
184 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(reloc))
185 return elfrel->Parse(data, offset);
187 return llvm::cast<ELFRela *>(reloc)->Parse(data, offset);
190unsigned ELFRelocation::RelocType32(
const ELFRelocation &rel) {
191 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(rel.reloc))
197unsigned ELFRelocation::RelocType64(
const ELFRelocation &rel) {
198 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(rel.reloc))
204unsigned ELFRelocation::RelocSymbol32(
const ELFRelocation &rel) {
205 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(rel.reloc))
211unsigned ELFRelocation::RelocSymbol64(
const ELFRelocation &rel) {
212 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(rel.reloc))
218elf_addr ELFRelocation::RelocOffset32(
const ELFRelocation &rel) {
219 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(rel.reloc))
220 return elfrel->r_offset;
222 return llvm::cast<ELFRela *>(rel.reloc)->r_offset;
225elf_addr ELFRelocation::RelocOffset64(
const ELFRelocation &rel) {
226 if (
auto *elfrel = llvm::dyn_cast<ELFRel *>(rel.reloc))
227 return elfrel->r_offset;
229 return llvm::cast<ELFRela *>(rel.reloc)->r_offset;
232elf_sxword ELFRelocation::RelocAddend32(
const ELFRelocation &rel) {
233 if (llvm::isa<ELFRel *>(rel.reloc))
236 return llvm::cast<ELFRela *>(rel.reloc)->r_addend;
239elf_sxword ELFRelocation::RelocAddend64(
const ELFRelocation &rel) {
240 if (llvm::isa<ELFRel *>(rel.reloc))
243 return llvm::cast<ELFRela *>(rel.reloc)->r_addend;
263 if (strncmp(buf,
"CORE", 4) == 0) {
271 if (cstr ==
nullptr) {
273 LLDB_LOGF(log,
"Failed to parse note name lacking nul terminator");
282 const uint32_t mips_arch = header.
e_flags & llvm::ELF::EF_MIPS_ARCH;
285 uint32_t fileclass = header.
e_ident[EI_CLASS];
290 if (header.
e_type == ET_CORE) {
292 case llvm::ELF::ELFCLASS32:
295 case llvm::ELF::ELFCLASS64:
304 case llvm::ELF::EF_MIPS_ARCH_1:
305 case llvm::ELF::EF_MIPS_ARCH_2:
306 case llvm::ELF::EF_MIPS_ARCH_32:
309 case llvm::ELF::EF_MIPS_ARCH_32R2:
312 case llvm::ELF::EF_MIPS_ARCH_32R6:
315 case llvm::ELF::EF_MIPS_ARCH_3:
316 case llvm::ELF::EF_MIPS_ARCH_4:
317 case llvm::ELF::EF_MIPS_ARCH_5:
318 case llvm::ELF::EF_MIPS_ARCH_64:
321 case llvm::ELF::EF_MIPS_ARCH_64R2:
324 case llvm::ELF::EF_MIPS_ARCH_64R6:
335 uint32_t fileclass = header.
e_ident[EI_CLASS];
337 case llvm::ELF::ELFCLASS32:
339 case llvm::ELF::ELFCLASS64:
348 if (
endian == ELFDATA2LSB)
355 uint32_t fileclass = header.
e_ident[EI_CLASS];
357 case llvm::ELF::ELFCLASS32:
359 case llvm::ELF::ELFCLASS64:
367 if (header.
e_machine == llvm::ELF::EM_MIPS)
369 else if (header.
e_machine == llvm::ELF::EM_PPC64)
371 else if (header.
e_machine == llvm::ELF::EM_RISCV)
373 else if (header.
e_machine == llvm::ELF::EM_LOONGARCH)
401 bool mapped_writable =
false;
402 if (!extractor_sp || !extractor_sp->HasData()) {
406 extractor_sp = std::make_shared<DataExtractor>(buffer_sp);
408 mapped_writable =
true;
411 assert(extractor_sp && extractor_sp->HasData());
413 DataBufferSP data_sp = extractor_sp->GetSharedDataBuffer();
415 if (data_sp->GetByteSize() <= (llvm::ELF::EI_NIDENT + data_offset))
418 const uint8_t *magic = data_sp->GetBytes() + data_offset;
423 if (data_sp->GetByteSize() < length) {
428 mapped_writable =
true;
429 magic = data_sp->GetBytes();
430 extractor_sp->SetData(data_sp);
434 if (!mapped_writable) {
435 data_sp = std::make_shared<DataBufferHeap>(data_sp->GetBytes(),
436 data_sp->GetByteSize());
438 magic = data_sp->GetBytes();
439 extractor_sp->SetData(data_sp);
443 if (address_size == 4 || address_size == 8) {
444 extractor_sp->SetAddressByteSize(address_size);
446 module_sp, extractor_sp, data_offset, file, file_offset, length));
447 ArchSpec spec = objfile_up->GetArchitecture();
448 if (spec && objfile_up->SetModulesArchitecture(spec))
449 return objfile_up.release();
458 if (!data_sp || data_sp->GetByteSize() < (llvm::ELF::EI_NIDENT))
460 const uint8_t *magic = data_sp->GetBytes();
469 if (!hdr.
Parse(data, &offset))
478 if (end_phdrs > data_sp->GetByteSize())
479 data_sp =
ReadMemory(process_sp, header_addr, end_phdrs);
481 std::unique_ptr<ObjectFileELF> objfile_up(
482 new ObjectFileELF(module_sp, data_sp, process_sp, header_addr));
483 ArchSpec spec = objfile_up->GetArchitecture();
484 if (spec && objfile_up->SetModulesArchitecture(spec))
485 return objfile_up.release();
494 data_sp->GetByteSize() > (llvm::ELF::EI_NIDENT + data_offset)) {
495 const uint8_t *magic = data_sp->GetBytes() + data_offset;
502 return llvm::crc32(init,
509 uint32_t core_notes_crc = 0;
512 if (H.p_type == llvm::ELF::PT_NOTE) {
513 const elf_off ph_offset = H.p_offset;
514 const size_t ph_size = H.p_filesz;
517 if (segment_data.
SetData(object_data, ph_offset, ph_size) != ph_size) {
523 core_notes_crc =
calc_crc32(core_notes_crc, segment_data);
527 return core_notes_crc;
531#define _MAKE_OSABI_CASE(x) \
534 switch (osabi_byte) {
556 return "<unknown-osabi>";
558#undef _MAKE_OSABI_CASE
568 llvm::Triple::OSType &ostype) {
569 switch (osabi_byte) {
571 ostype = llvm::Triple::OSType::AIX;
573 case ELFOSABI_FREEBSD:
574 ostype = llvm::Triple::OSType::FreeBSD;
577 ostype = llvm::Triple::OSType::Linux;
579 case ELFOSABI_NETBSD:
580 ostype = llvm::Triple::OSType::NetBSD;
582 case ELFOSABI_OPENBSD:
583 ostype = llvm::Triple::OSType::OpenBSD;
585 case ELFOSABI_SOLARIS:
586 ostype = llvm::Triple::OSType::Solaris;
589 ostype = llvm::Triple::OSType::UnknownOS;
591 return ostype != llvm::Triple::OSType::UnknownOS;
599 if (!extractor_sp || !extractor_sp->HasData())
602 extractor_sp->GetByteSize())) {
605 if (header.
Parse(*extractor_sp, &header_offset)) {
621 llvm::Triple::OSType ostype;
622 llvm::Triple::VendorType vendor;
623 llvm::Triple::OSType spec_ostype =
626 LLDB_LOGF(log,
"ObjectFileELF::%s file '%s' module OSABI: %s",
627 __FUNCTION__, file.
GetPath().c_str(),
632 assert(vendor == llvm::Triple::UnknownVendor);
638 assert(spec_ostype == ostype);
639 if (spec_ostype != llvm::Triple::OSType::UnknownOS) {
641 "ObjectFileELF::%s file '%s' set ELF module OS type "
642 "from ELF header OSABI.",
643 __FUNCTION__, file.
GetPath().c_str());
650 if (extractor_sp->GetByteSize() < length)
652 extractor_sp->SetData(data_sp);
660 header.
Parse(*extractor_sp, &header_offset);
663 uint32_t gnu_debuglink_crc = 0;
664 std::string gnu_debuglink_file;
669 gnu_debuglink_file, gnu_debuglink_crc,
675 "ObjectFileELF::%s file '%s' module set to triple: %s "
677 __FUNCTION__, file.
GetPath().c_str(),
678 spec_triple.getTriple().c_str(),
682 uint32_t core_notes_crc = 0;
684 if (!gnu_debuglink_crc) {
688 (length - file_offset) / 1024);
694 if (header.
e_type == llvm::ELF::ET_CORE) {
701 gnu_debuglink_crc =
calc_crc32(0, *extractor_sp);
704 using u32le = llvm::support::ulittle32_t;
705 if (gnu_debuglink_crc) {
707 u32le data(gnu_debuglink_crc);
708 uuid =
UUID(&data,
sizeof(data));
709 }
else if (core_notes_crc) {
714 uuid =
UUID(data,
sizeof(data));
734 :
ObjectFile(module_sp, file, file_offset, length, extractor_sp,
744 :
ObjectFile(module_sp, process_sp, header_addr,
752 bool value_is_offset) {
755 size_t num_loaded_sections = 0;
758 if (!value_is_offset) {
765 const size_t num_sections = section_list->
GetSize();
768 for (sect_idx = 0; sect_idx < num_sections; ++sect_idx) {
784 if (section_sp->IsThreadSpecific())
786 if (section_sp->Test(SHF_ALLOC) ||
798 load_addr &= 0xFFFFFFFF;
801 ++num_loaded_sections;
804 return num_loaded_sections > 0;
811 if (
m_header.e_ident[EI_DATA] == ELFDATA2MSB)
813 if (
m_header.e_ident[EI_DATA] == ELFDATA2LSB)
830 if (symtab_objfile !=
nullptr && symtab_objfile !=
this)
871 if (H.p_type == llvm::ELF::PT_NOTE) {
888 using u32le = llvm::support::ulittle32_t;
890 uint32_t core_notes_crc = 0;
898 if (core_notes_crc) {
927 uint32_t num_specs = 0;
929 for (
unsigned i = 0; i < num_modules; ++i) {
948 if (symbol.
d_tag != DT_DEBUG && symbol.
d_tag != DT_MIPS_RLD_MAP &&
949 symbol.
d_tag != DT_MIPS_RLD_MAP_REL)
959 if (symbol.
d_tag == DT_DEBUG)
964 if ((symbol.
d_tag == DT_MIPS_RLD_MAP ||
965 symbol.
d_tag == DT_MIPS_RLD_MAP_REL) &&
972 if (symbol.
d_tag == DT_MIPS_RLD_MAP) {
979 if (symbol.
d_tag == DT_MIPS_RLD_MAP_REL) {
988 addr_t debug_ptr_address =
1046 if (entry.symbol.d_tag != DT_NEEDED)
1048 if (!entry.name.empty()) {
1063 if (!program_headers.empty())
1064 return program_headers.size();
1070 program_headers.resize(header.
e_phnum);
1071 if (program_headers.size() != header.
e_phnum)
1077 if (data.
SetData(object_data, ph_offset, ph_size) != ph_size)
1082 for (idx = 0, offset = 0; idx < header.
e_phnum; ++idx) {
1083 if (!program_headers[idx].Parse(data, &offset))
1087 if (idx < program_headers.size())
1088 program_headers.resize(idx);
1090 return program_headers.size();
1111 if (!
note.Parse(data, &offset)) {
1116 LLDB_LOGF(log,
"ObjectFileELF::%s parsing note name='%s', type=%" PRIu32,
1117 __FUNCTION__,
note.n_name.c_str(),
note.n_type);
1124 uint32_t version_info;
1125 if (data.
GetU32(&offset, &version_info, 1) ==
nullptr) {
1132 const uint32_t version_major = version_info / 100000;
1133 const uint32_t version_minor = (version_info / 1000) % 100;
1136 snprintf(os_name,
sizeof(os_name),
"freebsd%" PRIu32
".%" PRIu32,
1137 version_major, version_minor);
1140 arch_spec.
GetTriple().setOSName(os_name);
1141 arch_spec.
GetTriple().setVendor(llvm::Triple::VendorType::UnknownVendor);
1144 "ObjectFileELF::%s detected FreeBSD %" PRIu32
".%" PRIu32
1146 __FUNCTION__, version_major, version_minor,
1147 static_cast<uint32_t
>(version_info % 1000));
1151 switch (
note.n_type) {
1155 uint32_t version_info[4];
1156 if (data.
GetU32(&offset, &version_info[0],
note.n_descsz / 4) ==
1164 switch (version_info[0]) {
1166 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Linux);
1168 llvm::Triple::VendorType::UnknownVendor);
1170 "ObjectFileELF::%s detected Linux, min version %" PRIu32
1171 ".%" PRIu32
".%" PRIu32,
1172 __FUNCTION__, version_info[1], version_info[2],
1179 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::UnknownOS);
1181 llvm::Triple::VendorType::UnknownVendor);
1183 "ObjectFileELF::%s detected Hurd (unsupported), min "
1184 "version %" PRIu32
".%" PRIu32
".%" PRIu32,
1185 __FUNCTION__, version_info[1], version_info[2],
1189 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Solaris);
1191 llvm::Triple::VendorType::UnknownVendor);
1193 "ObjectFileELF::%s detected Solaris, min version %" PRIu32
1194 ".%" PRIu32
".%" PRIu32,
1195 __FUNCTION__, version_info[1], version_info[2],
1200 "ObjectFileELF::%s unrecognized OS in note, id %" PRIu32
1201 ", min version %" PRIu32
".%" PRIu32
".%" PRIu32,
1202 __FUNCTION__, version_info[0], version_info[1],
1203 version_info[2], version_info[3]);
1215 if (
note.n_descsz >= 4) {
1216 if (
const uint8_t *buf = data.
PeekData(offset,
note.n_descsz)) {
1221 "failed to read GNU_BUILD_ID note payload");
1228 if (arch_spec.
IsMIPS() &&
1229 arch_spec.
GetTriple().getOS() == llvm::Triple::OSType::UnknownOS)
1231 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Linux);
1239 uint32_t version_info;
1240 if (data.
GetU32(&offset, &version_info, 1) ==
nullptr) {
1252 const uint32_t version_major = version_info / 100000000;
1253 const uint32_t version_minor = (version_info % 100000000) / 1000000;
1254 const uint32_t version_patch = (version_info % 10000) / 100;
1257 llvm::formatv(
"netbsd{0}.{1}.{2}", version_major, version_minor,
1258 version_patch).str());
1259 arch_spec.
GetTriple().setVendor(llvm::Triple::VendorType::UnknownVendor);
1265 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::NetBSD);
1266 arch_spec.
GetTriple().setVendor(llvm::Triple::VendorType::UnknownVendor);
1271 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::OpenBSD);
1272 arch_spec.
GetTriple().setVendor(llvm::Triple::VendorType::UnknownVendor);
1274 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Linux);
1276 llvm::Triple::EnvironmentType::Android);
1280 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Linux);
1304 if (
note.n_type == NT_FILE) {
1308 offset += count * 3 *
1310 for (
size_t i = 0; i < count; ++i) {
1312 if (cstr ==
nullptr) {
1314 "ObjectFileELF::%s trying to read "
1315 "at an offset after the end "
1316 "(GetCStr returned nullptr)",
1320 llvm::StringRef path(cstr);
1321 if (path.contains(
"/lib/x86_64-linux-gnu") || path.contains(
"/lib/i386-linux-gnu")) {
1322 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Linux);
1326 if (arch_spec.
IsMIPS() &&
1327 arch_spec.
GetTriple().getOS() == llvm::Triple::OSType::UnknownOS)
1330 arch_spec.
GetTriple().setOS(llvm::Triple::OSType::Linux);
1336 offset = note_offset +
note.GetByteSize();
1346 uint8_t FormatVersion = data.
GetU8(&Offset);
1347 if (FormatVersion != llvm::ELFAttrs::Format_Version)
1350 Offset = Offset +
sizeof(uint32_t);
1351 llvm::StringRef VendorName = data.
GetCStr(&Offset);
1353 if (VendorName !=
"aeabi")
1356 if (arch_spec.
GetTriple().getEnvironment() ==
1357 llvm::Triple::UnknownEnvironment)
1358 arch_spec.
GetTriple().setEnvironment(llvm::Triple::EABI);
1360 while (Offset < length) {
1361 uint8_t Tag = data.
GetU8(&Offset);
1362 uint32_t Size = data.
GetU32(&Offset);
1364 if (Tag != llvm::ARMBuildAttrs::File || Size == 0)
1367 while (Offset < length) {
1373 else if (Tag % 2 == 0)
1380 case llvm::ARMBuildAttrs::CPU_raw_name:
1381 case llvm::ARMBuildAttrs::CPU_name:
1386 case llvm::ARMBuildAttrs::ABI_VFP_args: {
1389 if (VFPArgs == llvm::ARMBuildAttrs::BaseAAPCS) {
1390 if (arch_spec.
GetTriple().getEnvironment() ==
1391 llvm::Triple::UnknownEnvironment ||
1392 arch_spec.
GetTriple().getEnvironment() == llvm::Triple::EABIHF)
1393 arch_spec.
GetTriple().setEnvironment(llvm::Triple::EABI);
1396 }
else if (VFPArgs == llvm::ARMBuildAttrs::HardFPAAPCS) {
1397 if (arch_spec.
GetTriple().getEnvironment() ==
1398 llvm::Triple::UnknownEnvironment ||
1399 arch_spec.
GetTriple().getEnvironment() == llvm::Triple::EABI)
1400 arch_spec.
GetTriple().setEnvironment(llvm::Triple::EABIHF);
1412static std::optional<lldb::offset_t>
1414 uint32_t length, llvm::StringRef name) {
1415 uint32_t section_length = 0;
1416 llvm::StringRef section_name;
1418 offset += section_length;
1422 auto tmp_offset = offset;
1423 section_length = data.
GetU32(&tmp_offset);
1424 section_name = data.
GetCStr(&tmp_offset);
1425 }
while (section_name != name && offset + section_length < length);
1427 if (section_name == name)
1430 return std::nullopt;
1433static std::optional<lldb::offset_t>
1438 auto parent_section_length = data.
GetU32(&offset);
1440 auto parent_section_end_offset = offset + parent_section_length;
1442 uint32_t section_length = 0;
1443 unsigned section_tag = 0;
1445 offset += section_length;
1448 auto tmp_offset = offset;
1450 section_length = data.
GetU32(&tmp_offset);
1451 }
while (section_tag != tag &&
1452 offset + section_length < parent_section_end_offset);
1454 if (section_tag == tag)
1457 return std::nullopt;
1460static std::optional<std::variant<uint64_t, llvm::StringRef>>
1466 auto parent_section_length = data.
GetU32(&offset);
1467 auto parent_section_end_offset = offset + parent_section_length;
1469 std::variant<uint64_t, llvm::StringRef> result;
1470 unsigned attribute_tag = 0;
1476 if (attribute_tag % 2)
1477 result = data.
GetCStr(&offset);
1480 }
while (attribute_tag != tag && offset < parent_section_end_offset);
1482 if (attribute_tag == tag)
1485 return std::nullopt;
1535 uint8_t format_version = data.
GetU8(&offset);
1536 if (format_version != llvm::ELFAttrs::Format_Version)
1539 auto subsection_or_opt =
1541 if (!subsection_or_opt) {
1543 "ObjectFileELF::%s Ill-formed .riscv.attributes section: "
1544 "mandatory 'riscv' sub-section was not preserved",
1550 data, *subsection_or_opt, llvm::ELFAttrs::File);
1551 if (!subsubsection_or_opt)
1555 llvm::RISCVAttrs::ARCH);
1559 auto normalized_isa_info = llvm::RISCVISAInfo::parseNormalizedArchString(
1560 std::get<llvm::StringRef>(*value_or_opt));
1561 if (llvm::errorToBool(normalized_isa_info.takeError()))
1564 llvm::SubtargetFeatures features;
1565 features.addFeaturesVector((*normalized_isa_info)->toFeatures());
1571 auto isa_info = llvm::RISCVISAInfo::parseArchString(
1572 std::get<llvm::StringRef>(*value_or_opt),
1574 if (
auto error = isa_info.takeError()) {
1576 ss <<
"the .riscv.attributes section contains an invalid RISC-V arch "
1578 << llvm::toString(std::move(
error))
1579 <<
"\n\tThis could result in misleading disassembler output\n";
1589 std::string &gnu_debuglink_file,
1590 uint32_t &gnu_debuglink_crc,
1593 if (!section_headers.empty())
1594 return section_headers.size();
1598 if (arch_spec.
GetTriple().getOS() == llvm::Triple::OSType::UnknownOS) {
1599 llvm::Triple::OSType ostype;
1600 llvm::Triple::OSType spec_ostype;
1612 spec_ostype = arch_spec.
GetTriple().getOS();
1613 assert(spec_ostype == ostype);
1617 if (arch_spec.
GetMachine() == llvm::Triple::mips ||
1618 arch_spec.
GetMachine() == llvm::Triple::mipsel ||
1619 arch_spec.
GetMachine() == llvm::Triple::mips64 ||
1620 arch_spec.
GetMachine() == llvm::Triple::mips64el) {
1621 switch (header.
e_flags & llvm::ELF::EF_MIPS_ARCH_ASE) {
1622 case llvm::ELF::EF_MIPS_MICROMIPS:
1625 case llvm::ELF::EF_MIPS_ARCH_ASE_M16:
1628 case llvm::ELF::EF_MIPS_ARCH_ASE_MDMX:
1636 if (arch_spec.
GetMachine() == llvm::Triple::arm ||
1637 arch_spec.
GetMachine() == llvm::Triple::thumb) {
1638 if (header.
e_flags & llvm::ELF::EF_ARM_SOFT_FLOAT)
1640 else if (header.
e_flags & llvm::ELF::EF_ARM_VFP_FLOAT)
1644 if (arch_spec.
GetMachine() == llvm::Triple::riscv32 ||
1645 arch_spec.
GetMachine() == llvm::Triple::riscv64) {
1646 uint32_t flags = arch_spec.
GetFlags();
1648 if (header.
e_flags & llvm::ELF::EF_RISCV_RVC)
1650 if (header.
e_flags & llvm::ELF::EF_RISCV_RVE)
1653 if ((header.
e_flags & llvm::ELF::EF_RISCV_FLOAT_ABI_SINGLE) ==
1654 llvm::ELF::EF_RISCV_FLOAT_ABI_SINGLE)
1656 else if ((header.
e_flags & llvm::ELF::EF_RISCV_FLOAT_ABI_DOUBLE) ==
1657 llvm::ELF::EF_RISCV_FLOAT_ABI_DOUBLE)
1659 else if ((header.
e_flags & llvm::ELF::EF_RISCV_FLOAT_ABI_QUAD) ==
1660 llvm::ELF::EF_RISCV_FLOAT_ABI_QUAD)
1666 if (arch_spec.
GetMachine() == llvm::Triple::loongarch32 ||
1667 arch_spec.
GetMachine() == llvm::Triple::loongarch64) {
1668 uint32_t flags = arch_spec.
GetFlags();
1669 switch (header.
e_flags & llvm::ELF::EF_LOONGARCH_ABI_MODIFIER_MASK) {
1670 case llvm::ELF::EF_LOONGARCH_ABI_SINGLE_FLOAT:
1673 case llvm::ELF::EF_LOONGARCH_ABI_DOUBLE_FLOAT:
1676 case llvm::ELF::EF_LOONGARCH_ABI_SOFT_FLOAT:
1689 section_headers.resize(header.
e_shnum);
1690 if (section_headers.size() != header.
e_shnum)
1696 if (sh_data.
SetData(object_data, sh_offset, sh_size) != sh_size)
1701 for (idx = 0, offset = 0; idx < header.
e_shnum; ++idx) {
1702 if (!section_headers[idx].Parse(sh_data, &offset))
1705 if (idx < section_headers.size())
1706 section_headers.resize(idx);
1708 const unsigned strtab_idx = header.
e_shstrndx;
1709 if (strtab_idx && strtab_idx < section_headers.size()) {
1711 const size_t byte_size = sheader.
sh_size;
1712 const Elf64_Off offset = sheader.
sh_offset;
1715 if (shstr_data.
SetData(object_data, offset, byte_size) == byte_size) {
1717 I != section_headers.end(); ++I) {
1718 static ConstString g_sect_name_gnu_debuglink(
".gnu_debuglink");
1720 const uint64_t section_size =
1724 I->section_name = name;
1726 if (arch_spec.
IsMIPS()) {
1727 uint32_t arch_flags = arch_spec.
GetFlags();
1729 if (sheader.
sh_type == SHT_MIPS_ABIFLAGS) {
1732 section_size) == section_size)) {
1735 arch_flags |= data.
GetU32(&offset);
1739 switch (data.
GetU8(&offset)) {
1740 case llvm::Mips::Val_GNU_MIPS_ABI_FP_ANY:
1743 case llvm::Mips::Val_GNU_MIPS_ABI_FP_DOUBLE:
1746 case llvm::Mips::Val_GNU_MIPS_ABI_FP_SINGLE:
1749 case llvm::Mips::Val_GNU_MIPS_ABI_FP_SOFT:
1752 case llvm::Mips::Val_GNU_MIPS_ABI_FP_OLD_64:
1755 case llvm::Mips::Val_GNU_MIPS_ABI_FP_XX:
1758 case llvm::Mips::Val_GNU_MIPS_ABI_FP_64:
1761 case llvm::Mips::Val_GNU_MIPS_ABI_FP_64A:
1768 switch (header.
e_flags & llvm::ELF::EF_MIPS_ABI) {
1769 case llvm::ELF::EF_MIPS_ABI_O32:
1772 case EF_MIPS_ABI_O64:
1775 case EF_MIPS_ABI_EABI32:
1778 case EF_MIPS_ABI_EABI64:
1783 if (header.
e_ident[EI_CLASS] == llvm::ELF::ELFCLASS64)
1785 else if (header.
e_flags & llvm::ELF::EF_MIPS_ABI2)
1792 if (arch_spec.
GetMachine() == llvm::Triple::arm ||
1793 arch_spec.
GetMachine() == llvm::Triple::thumb) {
1796 if (sheader.
sh_type == SHT_ARM_ATTRIBUTES && section_size != 0 &&
1797 data.
SetData(object_data, sheader.
sh_offset, section_size) == section_size)
1803 if (sheader.
sh_type == llvm::ELF::SHT_RISCV_ATTRIBUTES &&
1804 section_size != 0 &&
1810 if (name == g_sect_name_gnu_debuglink) {
1813 section_size) == section_size)) {
1815 gnu_debuglink_file = data.
GetCStr(&gnu_debuglink_offset);
1816 gnu_debuglink_offset = llvm::alignTo(gnu_debuglink_offset, 4);
1817 data.
GetU32(&gnu_debuglink_offset, &gnu_debuglink_crc, 1);
1822 bool is_note_header = (sheader.
sh_type == SHT_NOTE);
1826 static ConstString g_sect_name_android_ident(
".note.android.ident");
1827 if (!is_note_header && name == g_sect_name_android_ident)
1828 is_note_header =
true;
1830 if (is_note_header) {
1834 section_size) == section_size)) {
1837 LLDB_LOGF(log,
"ObjectFileELF::%s ELF note processing failed: %s",
1838 __FUNCTION__,
error.AsCString());
1845 if (arch_spec.
GetTriple().getVendor() == llvm::Triple::UnknownVendor)
1846 arch_spec.
GetTriple().setVendorName(llvm::StringRef());
1847 if (arch_spec.
GetTriple().getOS() == llvm::Triple::UnknownOS)
1848 arch_spec.
GetTriple().setOSName(llvm::StringRef());
1850 return section_headers.size();
1854 section_headers.clear();
1860 size_t pos = symbol_name.find(
'@');
1861 return symbol_name.substr(0, pos);
1892 if (Name.consume_front(
".debug_"))
1895 return llvm::StringSwitch<SectionType>(Name)
1934 Permissions Perm = Permissions(0);
1936 Perm |= ePermissionsReadable;
1938 Perm |= ePermissionsWritable;
1940 Perm |= ePermissionsExecutable;
1945 Permissions Perm = Permissions(0);
1947 Perm |= ePermissionsReadable;
1949 Perm |= ePermissionsWritable;
1951 Perm |= ePermissionsExecutable;
1959struct SectionAddressInfo {
1967class VMAddressProvider {
1969 llvm::IntervalMapHalfOpenInfo<addr_t>>;
1972 addr_t NextVMAddress = 0;
1973 VMMap::Allocator Alloc;
1974 VMMap Segments{Alloc};
1975 VMMap Sections{Alloc};
1976 lldb_private::Log *Log =
GetLog(LLDBLog::Modules);
1977 size_t SegmentCount = 0;
1978 std::string SegmentName;
1980 VMRange GetVMRange(
const ELFSectionHeader &H) {
1986 if ((ObjectType == ObjectFile::Type::eTypeObjectFile ||
1987 (ObjectType == ObjectFile::Type::eTypeDebugInfo && H.
sh_addr == 0)) &&
1988 Segments.empty() && (H.
sh_flags & SHF_ALLOC)) {
1990 llvm::alignTo(NextVMAddress, std::max<addr_t>(H.
sh_addralign, 1));
1991 Address = NextVMAddress;
1992 NextVMAddress += Size;
1994 return VMRange(Address, Size);
1999 : ObjectType(
Type), SegmentName(std::string(SegmentName)) {}
2001 std::string GetNextSegmentName()
const {
2002 return llvm::formatv(
"{0}[{1}]", SegmentName, SegmentCount).str();
2005 std::optional<VMRange> GetAddressInfo(
const ELFProgramHeader &H) {
2007 LLDB_LOG(Log,
"Ignoring zero-sized {0} segment. Corrupt object file?",
2009 return std::nullopt;
2013 LLDB_LOG(Log,
"Ignoring overlapping {0} segment. Corrupt object file?",
2015 return std::nullopt;
2020 std::optional<SectionAddressInfo> GetAddressInfo(
const ELFSectionHeader &H) {
2021 VMRange Range = GetVMRange(H);
2023 auto It = Segments.find(Range.GetRangeBase());
2024 if ((H.
sh_flags & SHF_ALLOC) && It.valid()) {
2026 if (It.start() <= Range.GetRangeBase()) {
2027 MaxSize = It.stop() - Range.GetRangeBase();
2030 MaxSize = It.start() - Range.GetRangeBase();
2031 if (Range.GetByteSize() > MaxSize) {
2032 LLDB_LOG(Log,
"Shortening section crossing segment boundaries. "
2033 "Corrupt object file?");
2034 Range.SetByteSize(MaxSize);
2037 if (Range.GetByteSize() > 0 &&
2038 Sections.overlaps(Range.GetRangeBase(), Range.GetRangeEnd())) {
2039 LLDB_LOG(Log,
"Ignoring overlapping section. Corrupt object file?");
2040 return std::nullopt;
2043 Range.Slide(-Segment->GetFileAddress());
2044 return SectionAddressInfo{Segment, Range};
2047 void AddSegment(
const VMRange &Range,
SectionSP Seg) {
2048 Segments.insert(Range.GetRangeBase(), Range.GetRangeEnd(), std::move(Seg));
2052 void AddSection(SectionAddressInfo Info,
SectionSP Sect) {
2053 if (Info.Range.GetByteSize() == 0)
2056 Info.Range.Slide(Info.Segment->GetFileAddress());
2057 Sections.insert(Info.Range.GetRangeBase(), Info.Range.GetRangeEnd(),
2071 addr_t vm_addr = section->GetFileAddress();
2073 offset_t byte_size = section->GetByteSize();
2074 bool thread_specific = section->IsThreadSpecific();
2075 uint32_t permissions = section->GetPermissions();
2076 uint32_t alignment = section->GetLog2Align();
2078 for (
auto sect : section_list) {
2079 if (sect->GetName() == name &&
2080 sect->IsThreadSpecific() == thread_specific &&
2081 sect->GetPermissions() == permissions &&
2082 sect->GetByteSize() == byte_size && sect->GetFileAddress() == vm_addr &&
2083 sect->GetLog2Align() == alignment) {
2101 VMAddressProvider regular_provider(
GetType(),
"PT_LOAD");
2102 VMAddressProvider tls_provider(
GetType(),
"PT_TLS");
2109 VMAddressProvider &provider =
2110 PHdr.
p_type == PT_TLS ? tls_provider : regular_provider;
2111 auto InfoOr = provider.GetAddressInfo(PHdr);
2115 uint32_t Log2Align = llvm::Log2_64(std::max<elf_xword>(PHdr.
p_align, 1));
2116 SectionSP Segment = std::make_shared<Section>(
2119 InfoOr->GetRangeBase(), InfoOr->GetByteSize(), PHdr.
p_offset,
2122 Segment->SetIsThreadSpecific(PHdr.
p_type == PT_TLS);
2125 provider.AddSegment(*InfoOr, std::move(Segment));
2137 const uint64_t file_size =
2140 VMAddressProvider &provider =
2141 header.
sh_flags & SHF_TLS ? tls_provider : regular_provider;
2142 auto InfoOr = provider.GetAddressInfo(header);
2158 InfoOr->Range.GetRangeBase(),
2159 InfoOr->Range.GetByteSize(),
2166 section_sp->SetIsThreadSpecific(header.
sh_flags & SHF_TLS);
2167 (InfoOr->Segment ? InfoOr->Segment->GetChildren() : *
m_sections_up)
2168 .AddSection(section_sp);
2169 provider.AddSection(std::move(*InfoOr), std::move(section_sp));
2174 unified_section_list =
2181 if (
auto gdd_objfile_section_list = gdd_obj_file->GetSectionList()) {
2183 gdd_objfile_section_list->FindSectionByType(
2187 if (module_section_sp)
2191 unified_section_list.
AddSection(symtab_section_sp);
2208 "no LZMA support found for reading .gnu_debugdata section");
2214 section->GetSectionData(data);
2215 llvm::SmallVector<uint8_t, 0> uncompressedData;
2219 "an error occurred while decompressing the section {0}: {1}",
2220 section->GetName(), llvm::toString(std::move(err)).c_str());
2226 new DataBufferHeap(uncompressedData.data(), uncompressedData.size()));
2227 DataExtractorSP extractor_sp = std::make_shared<DataExtractor>(gdd_data_buf);
2229 llvm::StringRef(
"gnu_debugdata"));
2231 GetModule(), extractor_sp, 0, &fspec, 0, gdd_data_buf->GetByteSize()));
2252 const char *dollar_pos = ::strchr(symbol_name,
'$');
2253 if (!dollar_pos || dollar_pos[1] ==
'\0')
2256 if (dollar_pos[2] ==
'\0' || dollar_pos[2] ==
'.')
2257 return dollar_pos[1];
2265 if (strcmp(symbol_name,
"$d") == 0) {
2268 if (strcmp(symbol_name,
"$x") == 0) {
2274#define STO_MIPS_ISA (3 << 6)
2275#define STO_MICROMIPS (2 << 6)
2276#define IS_MICROMIPS(ST_OTHER) (((ST_OTHER)&STO_MIPS_ISA) == STO_MICROMIPS)
2279std::pair<unsigned, ObjectFileELF::FileAddressToAddressClassMap>
2281 SectionList *section_list,
const size_t num_symbols,
2298 static ConstString rodata_section_name(
".rodata");
2299 static ConstString rodata1_section_name(
".rodata1");
2314 llvm::StringRef file_extension =
m_file.GetFileNameExtension();
2315 bool skip_oatdata_oatexec =
2316 file_extension ==
".oat" || file_extension ==
".odex";
2321 module_sp ? module_sp->GetSectionList() :
nullptr;
2328 std::unordered_map<lldb::SectionSP, lldb::SectionSP> section_map;
2331 for (i = 0; i < num_symbols; ++i) {
2332 if (!symbol.
Parse(symtab_data, &offset))
2343 if (llvm::StringRef(symbol_name).starts_with(
".L"))
2346 if (symbol.
getType() != STT_SECTION &&
2347 (symbol_name ==
nullptr || symbol_name[0] ==
'\0'))
2352 if (skip_oatdata_oatexec && (::strcmp(symbol_name,
"oatdata") == 0 ||
2353 ::strcmp(symbol_name,
"oatexec") == 0))
2358 Elf64_Half shndx = symbol.
st_shndx;
2421 if (symbol_section_sp) {
2422 ConstString sect_name = symbol_section_sp->GetName();
2423 if (sect_name == text_section_name || sect_name == init_section_name ||
2424 sect_name == fini_section_name || sect_name == ctors_section_name ||
2425 sect_name == dtors_section_name) {
2427 }
else if (sect_name == data_section_name ||
2428 sect_name == data2_section_name ||
2429 sect_name == rodata_section_name ||
2430 sect_name == rodata1_section_name ||
2431 sect_name == bss_section_name) {
2433 }
else if (symbol_section_sp->Get() & SHF_ALLOC)
2441 int64_t symbol_value_offset = 0;
2442 uint32_t additional_flags = 0;
2444 if (arch.
GetMachine() == llvm::Triple::arm) {
2448 switch (mapping_symbol) {
2457 address_class_map[symbol.
st_value] =
2469 }
else if (arch.
GetMachine() == llvm::Triple::aarch64) {
2473 switch (mapping_symbol) {
2487 }
else if (arch.
GetTriple().isRISCV()) {
2494 switch (mapping_symbol) {
2510 if (arch.
GetMachine() == llvm::Triple::arm) {
2518 symbol_value_offset = -1;
2519 address_class_map[symbol.
st_value ^ 1] =
2561 uint64_t symbol_value = symbol.
st_value + symbol_value_offset;
2563 if (symbol_section_sp &&
2565 symbol_value -= symbol_section_sp->GetFileAddress();
2567 if (symbol_section_sp && module_section_list &&
2568 module_section_list != section_list) {
2569 auto section_it = section_map.find(symbol_section_sp);
2570 if (section_it == section_map.end()) {
2571 section_it = section_map
2572 .emplace(symbol_section_sp,
2577 if (section_it->second)
2578 symbol_section_sp = section_it->second;
2581 bool is_global = symbol.
getBinding() == STB_GLOBAL;
2582 uint32_t flags = symbol.
st_other << 8 | symbol.
st_info | additional_flags;
2583 llvm::StringRef symbol_ref(symbol_name);
2587 size_t version_pos = symbol_ref.find(
'@');
2588 bool has_suffix = version_pos != llvm::StringRef::npos;
2589 llvm::StringRef symbol_bare = symbol_ref.substr(0, version_pos);
2595 llvm::StringRef suffix = symbol_ref.substr(version_pos);
2598 if (!mangled_name.empty())
2602 llvm::StringRef demangled_name = demangled.
GetStringRef();
2603 if (!demangled_name.empty())
2612 bool symbol_size_valid =
2615 bool is_trampoline =
false;
2624 if (trampoline_name.starts_with(
"__AArch64ADRPThunk_") ||
2625 trampoline_name.starts_with(
"__AArch64AbsLongThunk_")) {
2627 is_trampoline =
true;
2653 return {i, address_class_map};
2656std::pair<unsigned, ObjectFileELF::FileAddressToAddressClassMap>
2664 auto [num_symbols, address_class_map] =
2671 return {num_symbols, address_class_map};
2681 assert(symtab_hdr->
sh_type == SHT_SYMTAB ||
2682 symtab_hdr->
sh_type == SHT_DYNSYM);
2688 if (symtab && strtab) {
2698 return ParseSymbols(symbol_table, start_id, section_list, num_symbols,
2699 symtab_data, strtab_data);
2721 if (std::optional<DataExtractor> dynstr_data =
GetDynstrData()) {
2723 switch (entry.symbol.d_tag) {
2731 const char *name = dynstr_data->GetCStr(&cursor);
2733 entry.name = std::string(name);
2748 if (entry.symbol.d_tag == tag)
2749 return &entry.symbol;
2763 return symbol->
d_val;
2772static std::pair<uint64_t, uint64_t>
2788 if (plt_entsize <= 4) {
2798 plt_entsize = plt_hdr->
sh_size / (num_relocations + 1);
2803 return std::make_pair(plt_entsize, plt_offset);
2812 ELFRelocation rel(rel_type);
2816 uint64_t plt_offset, plt_entsize;
2817 std::tie(plt_entsize, plt_offset) =
2821 typedef unsigned (*reloc_info_fn)(
const ELFRelocation &rel);
2822 reloc_info_fn reloc_type;
2823 reloc_info_fn reloc_symbol;
2826 reloc_type = ELFRelocation::RelocType32;
2827 reloc_symbol = ELFRelocation::RelocSymbol32;
2829 reloc_type = ELFRelocation::RelocType64;
2830 reloc_symbol = ELFRelocation::RelocSymbol64;
2835 for (i = 0; i < num_relocations; ++i) {
2836 if (!rel.Parse(rel_data, &offset))
2839 if (reloc_type(rel) != slot_type)
2843 if (!symbol.
Parse(symtab_data, &symbol_offset))
2847 uint64_t plt_index = plt_offset + i * plt_entsize;
2874 assert(rel_hdr->
sh_type == SHT_RELA || rel_hdr->
sh_type == SHT_REL);
2888 if (!symtab_id || !plt_id)
2908 if (!plt_section_sp)
2937 rel_hdr, plt_hdr, sym_hdr, plt_section_sp,
2938 rel_data, symtab_data, strtab_data);
2951 llvm::cast<WritableDataBuffer>(data_buffer_sp.get());
2953 ELFRelocation::RelocOffset64(rel);
2954 uint64_t val_offset = value + ELFRelocation::RelocAddend64(rel);
2955 memcpy(dst, &val_offset,
sizeof(uint64_t));
2961 Section *rel_section,
bool is_signed) {
2966 value += ELFRelocation::RelocAddend32(rel);
2969 ((int64_t)value >
INT32_MAX || (int64_t)value < INT32_MIN))) {
2971 LLDB_LOGF(log,
"Failed to apply debug info relocations");
2974 uint32_t truncated_addr = (value & 0xFFFFFFFF);
2978 llvm::cast<WritableDataBuffer>(data_buffer_sp.get());
2980 ELFRelocation::RelocOffset32(rel);
2981 memcpy(dst, &truncated_addr,
sizeof(uint32_t));
2995 LLDB_LOGF(log,
"Debug info symbol invalid: %s", name);
2998 assert(llvm::isUInt<32>(value) &&
"Valid addresses are 32-bit");
3002 llvm::cast<WritableDataBuffer>(data_buffer_sp.get());
3004 ELFRelocation::RelocOffset32(rel);
3007 memcpy(&addend, dst,
sizeof(int32_t));
3010 if (addend < 0 &&
static_cast<uint32_t
>(-addend) > value) {
3011 LLDB_LOGF(log,
"Debug info relocation overflow: 0x%" PRIx64,
3012 static_cast<int64_t
>(value) + addend);
3015 if (!llvm::isUInt<32>(value + addend)) {
3016 LLDB_LOGF(log,
"Debug info relocation out of range: 0x%" PRIx64, value);
3019 uint32_t addr = value + addend;
3020 memcpy(dst, &addr,
sizeof(uint32_t));
3029 ELFRelocation rel(rel_hdr->
sh_type);
3032 typedef unsigned (*reloc_info_fn)(
const ELFRelocation &rel);
3033 reloc_info_fn reloc_type;
3034 reloc_info_fn reloc_symbol;
3037 reloc_type = ELFRelocation::RelocType32;
3038 reloc_symbol = ELFRelocation::RelocSymbol32;
3040 reloc_type = ELFRelocation::RelocType64;
3041 reloc_symbol = ELFRelocation::RelocSymbol64;
3044 for (
unsigned i = 0; i < num_relocations; ++i) {
3045 if (!rel.Parse(rel_data, &offset)) {
3046 GetModule()->ReportError(
".rel{0}[{1:d}] failed to parse relocation",
3050 const Symbol *symbol =
nullptr;
3054 case llvm::ELF::EM_ARM:
3055 switch (reloc_type(rel)) {
3060 GetModule()->ReportError(
"unsupported AArch32 relocation:"
3061 " .rel{0}[{1}], type {2}",
3062 rel_section->
GetName(), i, reloc_type(rel));
3065 assert(
false &&
"unexpected relocation type");
3068 case llvm::ELF::EM_386:
3069 switch (reloc_type(rel)) {
3074 rel_section->
GetFileOffset() + ELFRelocation::RelocOffset32(rel);
3078 llvm::cast<WritableDataBuffer>(data_buffer_sp.get());
3079 uint32_t *dst =
reinterpret_cast<uint32_t *
>(
3080 data_buffer->
GetBytes() + f_offset);
3084 value += ELFRelocation::RelocAddend32(rel);
3090 GetModule()->ReportError(
".rel{0}[{1}] unknown symbol id: {2:d}",
3097 GetModule()->ReportError(
"unsupported i386 relocation:"
3098 " .rel{0}[{1}], type {2}",
3099 rel_section->
GetName(), i, reloc_type(rel));
3102 assert(
false &&
"unexpected relocation type");
3107 GetModule()->ReportError(
"unsupported 32-bit ELF machine arch: {0}", hdr->
e_machine);
3112 case llvm::ELF::EM_AARCH64:
3113 switch (reloc_type(rel)) {
3114 case R_AARCH64_ABS64:
3117 case R_AARCH64_ABS32:
3121 assert(
false &&
"unexpected relocation type");
3124 case llvm::ELF::EM_LOONGARCH:
3125 switch (reloc_type(rel)) {
3133 assert(
false &&
"unexpected relocation type");
3136 case llvm::ELF::EM_X86_64:
3137 switch (reloc_type(rel)) {
3150 assert(
false &&
"unexpected relocation type");
3154 GetModule()->ReportError(
"unsupported 64-bit ELF machine arch: {0}", hdr->
e_machine);
3166 assert(rel_hdr->
sh_type == SHT_RELA || rel_hdr->
sh_type == SHT_REL);
3204 *rel_data_sp, *symtab_data_sp, *debug_data_sp, debug);
3215 Progress progress(
"Parsing symbol table",
3216 m_file.GetFilename().AsCString(
"<Unknown>"));
3221 ObjectFile *module_obj_file = module_sp->GetObjectFile();
3222 if (module_obj_file && module_obj_file !=
this)
3225 SectionList *section_list = module_sp->GetSectionList();
3229 uint64_t symbol_id = 0;
3239 auto [num_symbols, address_class_map] =
3242 symbol_id += num_symbols;
3257 auto [num_symbols, address_class_map] =
3259 symbol_id += num_symbols;
3263 uint32_t dynamic_num_symbols = 0;
3264 std::optional<DataExtractor> symtab_data =
3267 if (symtab_data && strtab_data) {
3268 auto [num_symbols_parsed, address_class_map] =
ParseSymbols(
3269 &lldb_symtab, symbol_id, section_list, dynamic_num_symbols,
3270 symtab_data.value(), strtab_data.value());
3271 symbol_id += num_symbols_parsed;
3292 if (reloc_section) {
3302 GetModule()->GetUnwindTable().GetEHFrameInfo()) {
3316 bool is_valid_entry_point =
3317 entry_point_addr.IsValid() && entry_point_addr.IsSectionOffset();
3318 addr_t entry_point_file_addr = entry_point_addr.GetFileAddress();
3320 entry_point_file_addr)) {
3325 SectionSP section_sp = entry_point_addr.GetSection();
3335 entry_point_addr.GetOffset(),
3344 if (arch.
GetMachine() == llvm::Triple::arm &&
3345 (entry_point_file_addr & 1)) {
3359 static const char *debug_prefix =
".debug";
3371 if (section_name ==
nullptr)
3375 if (strncmp(section_name, debug_prefix, strlen(debug_prefix)))
3379 std::string needle = std::string(
".rel") + section_name;
3380 std::string needlea = std::string(
".rela") + section_name;
3384 if (I->sh_type == SHT_RELA || I->sh_type == SHT_REL) {
3385 const char *hay_name = I->section_name.GetCString();
3386 if (hay_name ==
nullptr)
3388 if (needle == hay_name || needlea == hay_name) {
3409 std::vector<Symbol> new_symbols;
3412 uint64_t last_symbol_id =
3426 addr_t offset = file_addr - section_sp->GetFileAddress();
3427 uint64_t symbol_id = ++last_symbol_id;
3445 new_symbols.push_back(eh_symbol);
3451 for (
const Symbol &s : new_symbols)
3471 std::lock_guard<std::recursive_mutex> guard(module_sp->GetMutex());
3472 s->
Printf(
"%p: ",
static_cast<void *
>(
this));
3478 *s <<
", file = '" <<
m_file
3503 if (image_info_addr.
IsValid())
3504 s->
Printf(
"image_info_address = %#16.16" PRIx64
"\n",
3513 s->
Printf(
"e_ident[EI_MAG0 ] = 0x%2.2x\n", header.
e_ident[EI_MAG0]);
3514 s->
Printf(
"e_ident[EI_MAG1 ] = 0x%2.2x '%c'\n", header.
e_ident[EI_MAG1],
3516 s->
Printf(
"e_ident[EI_MAG2 ] = 0x%2.2x '%c'\n", header.
e_ident[EI_MAG2],
3518 s->
Printf(
"e_ident[EI_MAG3 ] = 0x%2.2x '%c'\n", header.
e_ident[EI_MAG3],
3521 s->
Printf(
"e_ident[EI_CLASS ] = 0x%2.2x\n", header.
e_ident[EI_CLASS]);
3522 s->
Printf(
"e_ident[EI_DATA ] = 0x%2.2x ", header.
e_ident[EI_DATA]);
3524 s->
Printf(
"\ne_ident[EI_VERSION] = 0x%2.2x\n", header.
e_ident[EI_VERSION]);
3525 s->
Printf(
"e_ident[EI_PAD ] = 0x%2.2x\n", header.
e_ident[EI_PAD]);
3572 unsigned char ei_data) {
3575 *s <<
"ELFDATANONE";
3578 *s <<
"ELFDATA2LSB - Little Endian";
3581 *s <<
"ELFDATA2MSB - Big Endian";
3594 s->
Printf(
" %8.8" PRIx64
" %8.8" PRIx64
" %8.8" PRIx64, ph.
p_offset,
3608 const int kStrWidth = 15;
3620 s->
Printf(
"0x%8.8x%*s", p_type, kStrWidth - 10,
"");
3629 *s << ((p_flags & PF_X) ?
"PF_X" :
" ")
3630 << (((p_flags & PF_X) && (p_flags & PF_W)) ?
'+' :
' ')
3631 << ((p_flags & PF_W) ?
"PF_W" :
" ")
3632 << (((p_flags & PF_W) && (p_flags & PF_R)) ?
'+' :
' ')
3633 << ((p_flags & PF_R) ?
"PF_R" :
" ");
3644 s->
PutCString(
"IDX p_type p_offset p_vaddr p_paddr "
3645 "p_filesz p_memsz p_flags p_align\n");
3646 s->
PutCString(
"==== --------------- -------- -------- -------- "
3647 "-------- -------- ------------------------- --------\n");
3650 s->
Format(
"[{0,2}] ", H.index());
3665 s->
Printf(
") %8.8" PRIx64
" %8.8" PRIx64
" %8.8" PRIx64, sh.
sh_addr,
3676 const int kStrWidth = 12;
3695 s->
Printf(
"0x%8.8x%*s", sh_type, kStrWidth - 10,
"");
3705 *s << ((sh_flags & SHF_WRITE) ?
"WRITE" :
" ")
3706 << (((sh_flags & SHF_WRITE) && (sh_flags & SHF_ALLOC)) ?
'+' :
' ')
3707 << ((sh_flags & SHF_ALLOC) ?
"ALLOC" :
" ")
3708 << (((sh_flags & SHF_ALLOC) && (sh_flags & SHF_EXECINSTR)) ?
'+' :
' ')
3709 << ((sh_flags & SHF_EXECINSTR) ?
"EXECINSTR" :
" ");
3721 "addr offset size link info addralgn "
3723 s->
PutCString(
"==== -------- ------------ -------------------------------- "
3724 "-------- -------- -------- -------- -------- -------- "
3725 "-------- ====================\n");
3730 s->
Printf(
"[%2u] ", idx);
3732 const char *section_name = I->section_name.AsCString(
"");
3734 *s <<
' ' << section_name <<
"\n";
3741 if (num_modules > 0) {
3743 for (
unsigned i = 0; i < num_modules; ++i) {
3751#define DYNAMIC_STRINGIFY_ENUM(tag, value) \
3755#define DYNAMIC_TAG(n, v)
3757 case llvm::ELF::EM_AARCH64:
3759#define AARCH64_DYNAMIC_TAG(name, value) DYNAMIC_STRINGIFY_ENUM(name, value)
3760#include "llvm/BinaryFormat/DynamicTags.def"
3761#undef AARCH64_DYNAMIC_TAG
3765 case llvm::ELF::EM_HEXAGON:
3767#define HEXAGON_DYNAMIC_TAG(name, value) DYNAMIC_STRINGIFY_ENUM(name, value)
3768#include "llvm/BinaryFormat/DynamicTags.def"
3769#undef HEXAGON_DYNAMIC_TAG
3773 case llvm::ELF::EM_MIPS:
3775#define MIPS_DYNAMIC_TAG(name, value) DYNAMIC_STRINGIFY_ENUM(name, value)
3776#include "llvm/BinaryFormat/DynamicTags.def"
3777#undef MIPS_DYNAMIC_TAG
3781 case llvm::ELF::EM_PPC:
3783#define PPC_DYNAMIC_TAG(name, value) DYNAMIC_STRINGIFY_ENUM(name, value)
3784#include "llvm/BinaryFormat/DynamicTags.def"
3785#undef PPC_DYNAMIC_TAG
3789 case llvm::ELF::EM_PPC64:
3791#define PPC64_DYNAMIC_TAG(name, value) DYNAMIC_STRINGIFY_ENUM(name, value)
3792#include "llvm/BinaryFormat/DynamicTags.def"
3793#undef PPC64_DYNAMIC_TAG
3797 case llvm::ELF::EM_RISCV:
3799#define RISCV_DYNAMIC_TAG(name, value) DYNAMIC_STRINGIFY_ENUM(name, value)
3800#include "llvm/BinaryFormat/DynamicTags.def"
3801#undef RISCV_DYNAMIC_TAG
3808#define AARCH64_DYNAMIC_TAG(name, value)
3809#define MIPS_DYNAMIC_TAG(name, value)
3810#define HEXAGON_DYNAMIC_TAG(name, value)
3811#define PPC_DYNAMIC_TAG(name, value)
3812#define PPC64_DYNAMIC_TAG(name, value)
3813#define RISCV_DYNAMIC_TAG(name, value)
3815#define DYNAMIC_TAG_MARKER(name, value)
3816#define DYNAMIC_TAG(name, value) \
3819#include "llvm/BinaryFormat/DynamicTags.def"
3821#undef AARCH64_DYNAMIC_TAG
3822#undef MIPS_DYNAMIC_TAG
3823#undef HEXAGON_DYNAMIC_TAG
3824#undef PPC_DYNAMIC_TAG
3825#undef PPC64_DYNAMIC_TAG
3826#undef RISCV_DYNAMIC_TAG
3827#undef DYNAMIC_TAG_MARKER
3828#undef DYNAMIC_STRINGIFY_ENUM
3830 return "<unknown:>0x" + llvm::utohexstr(
Type,
true);
3841 s->
PutCString(
"==== ---------------- ------------------\n");
3844 s->
Printf(
"[%2u] ", idx++);
3846 "%-16s 0x%16.16" PRIx64,
3848 entry.symbol.d_ptr);
3849 if (!entry.name.empty())
3850 s->
Printf(
" \"%s\"", entry.name.c_str());
3869 if (H.p_type != PT_NOTE || H.p_offset == 0 || H.p_filesz == 0)
3883 case llvm::ELF::ET_NONE:
3887 case llvm::ELF::ET_REL:
3891 case llvm::ELF::ET_EXEC:
3895 case llvm::ELF::ET_DYN:
3911 case llvm::ELF::ET_NONE:
3915 case llvm::ELF::ET_REL:
3919 case llvm::ELF::ET_EXEC:
3924 static ConstString loader_section_name(
".interp");
3927 if (loader_section) {
3935 llvm::StringRef loader_name(buffer, read_size - 1);
3936 llvm::StringRef freebsd_kernel_loader_name(
"/red/herring");
3937 if (loader_name == freebsd_kernel_loader_name)
3944 case llvm::ELF::ET_DYN:
3972 if (!section->
Test(SHF_COMPRESSED))
3979 return data.
CopyData(section_offset, dst_len, dst);
3989 if (result == 0 || !(section->
Get() & llvm::ELF::SHF_COMPRESSED))
3992 auto Decompressor = llvm::object::Decompressor::create(
3994 {reinterpret_cast<const char *>(section_data.GetDataStart()),
3995 size_t(section_data.GetByteSize())},
3997 if (!Decompressor) {
3999 "unable to initialize decompressor for section '{0}': {1}",
4001 llvm::toString(Decompressor.takeError()).c_str());
4002 section_data.
Clear();
4007 std::make_shared<DataBufferHeap>(Decompressor->getDecompressedSize(), 0);
4008 if (
auto error = Decompressor->decompress(
4009 {buffer_sp->GetBytes(), size_t(buffer_sp->GetByteSize())})) {
4010 GetModule()->ReportWarning(
"decompression of section '{0}' failed: {1}",
4012 llvm::toString(std::move(
error)).c_str());
4013 section_data.
Clear();
4017 section_data.
SetData(buffer_sp);
4018 return buffer_sp->GetByteSize();
4055std::vector<ObjectFile::LoadableData>
4059 std::vector<LoadableData> loadables;
4063 if (H.p_type != llvm::ELF::PT_LOAD)
4065 loadable.
Dest = should_use_paddr ? H.p_paddr : H.p_vaddr;
4068 if (H.p_filesz == 0)
4071 loadable.
Contents = llvm::ArrayRef<uint8_t>(segment_data.GetDataStart(),
4072 segment_data.GetByteSize());
4073 loadables.push_back(loadable);
4085std::optional<DataExtractor>
4102 return std::nullopt;
4108 return std::nullopt;
4118 assert(dynamic->GetObjectFile() ==
this);
4124 section_list->FindSectionByID(header->sh_link).get()) {
4143 if (strtab ==
nullptr || strsz ==
nullptr)
4144 return std::nullopt;
4155 if (H.p_type == llvm::ELF::PT_DYNAMIC) {
4170 assert(dynamic->GetObjectFile() ==
this);
4177 return std::nullopt;
4182 if (hash ==
nullptr)
4183 return std::nullopt;
4186 struct DtHashHeader {
4193 offset_t offset = offsetof(DtHashHeader, nchain);
4194 return data->
GetU32(&offset);
4197 return std::nullopt;
4202 if (gnu_hash ==
nullptr)
4203 return std::nullopt;
4207 struct DtGnuHashHeader {
4208 uint32_t nbuckets = 0;
4209 uint32_t symoffset = 0;
4210 uint32_t bloom_size = 0;
4211 uint32_t bloom_shift = 0;
4213 uint32_t num_symbols = 0;
4217 DtGnuHashHeader header;
4218 header.nbuckets = data->
GetU32(&offset);
4219 header.symoffset = data->
GetU32(&offset);
4220 header.bloom_size = data->
GetU32(&offset);
4221 header.bloom_shift = data->
GetU32(&offset);
4223 const addr_t buckets_offset =
4224 sizeof(DtGnuHashHeader) + addr_size * header.bloom_size;
4225 std::vector<uint32_t> buckets;
4229 for (uint32_t i = 0; i < header.nbuckets; ++i)
4230 buckets.push_back(bucket_data->GetU32(&offset));
4232 uint32_t last_symbol = 0;
4233 for (uint32_t bucket_value : buckets)
4234 last_symbol = std::max(bucket_value, last_symbol);
4235 if (last_symbol < header.symoffset) {
4236 num_symbols = header.symoffset;
4239 const addr_t chains_base_offset = buckets_offset + header.nbuckets * 4;
4243 chains_base_offset + (last_symbol - header.symoffset) * 4)) {
4245 uint32_t chain_entry = chain_entry_data->GetU32(&offset);
4248 if (chain_entry & 1)
4254 num_symbols = last_symbol;
4258 if (num_symbols > 0)
4261 return std::nullopt;
4264std::optional<DataExtractor>
4282 if (symtab ==
nullptr || syment ==
nullptr)
4283 return std::nullopt;
4286 num_symbols = *syms;
4288 num_symbols = *syms;
4290 return std::nullopt;
4291 if (num_symbols == 0)
4292 return std::nullopt;
static llvm::raw_ostream & error(Stream &strm)
static llvm::raw_ostream & note(Stream &strm)
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
#define LLDB_LOGF(log,...)
static void ApplyELF64ABS32Relocation(Symtab *symtab, ELFRelocation &rel, DataExtractor &debug_data, Section *rel_section, bool is_signed)
static const elf_word LLDB_NT_NETBSD_IDENT_DESCSZ
static const char *const LLDB_NT_OWNER_NETBSDCORE
static const elf_word LLDB_NT_FREEBSD_ABI_TAG
static std::string getDynamicTagAsString(uint16_t Arch, uint64_t Type)
static uint32_t riscvVariantFromElfFlags(const elf::ELFHeader &header)
static const elf_word LLDB_NT_GNU_ABI_OS_LINUX
static uint32_t ppc64VariantFromElfFlags(const elf::ELFHeader &header)
static bool GetOsFromOSABI(unsigned char osabi_byte, llvm::Triple::OSType &ostype)
#define _MAKE_OSABI_CASE(x)
static std::optional< lldb::offset_t > FindSubSectionOffsetByName(const DataExtractor &data, lldb::offset_t offset, uint32_t length, llvm::StringRef name)
static uint32_t subTypeFromElfHeader(const elf::ELFHeader &header)
static uint32_t calc_crc32(uint32_t init, const DataExtractor &data)
static char FindArmAarch64MappingSymbol(const char *symbol_name)
static const char *const LLDB_NT_OWNER_CORE
static const elf_word LLDB_NT_NETBSD_IDENT_TAG
static const elf_word LLDB_NT_GNU_ABI_OS_SOLARIS
static std::pair< uint64_t, uint64_t > GetPltEntrySizeAndOffset(const ELFSectionHeader *rel_hdr, const ELFSectionHeader *plt_hdr)
static SectionType GetSectionTypeFromName(llvm::StringRef Name)
static const elf_word LLDB_NT_FREEBSD_ABI_SIZE
static const elf_word LLDB_NT_GNU_ABI_TAG
static char FindRISCVMappingSymbol(const char *symbol_name)
static SectionSP FindMatchingSection(const SectionList §ion_list, SectionSP section)
static const char *const LLDB_NT_OWNER_GNU
static const elf_word LLDB_NT_NETBSD_PROCINFO
#define CASE_AND_STREAM(s, def, width)
static user_id_t SegmentID(size_t PHdrIndex)
static void ApplyELF32ABS32RelRelocation(Symtab *symtab, ELFRelocation &rel, DataExtractor &debug_data, Section *rel_section)
static std::optional< std::variant< uint64_t, llvm::StringRef > > GetAttributeValueByTag(const DataExtractor &data, lldb::offset_t offset, unsigned tag)
static const elf_word LLDB_NT_GNU_ABI_SIZE
static const char *const LLDB_NT_OWNER_OPENBSD
static const char *const LLDB_NT_OWNER_FREEBSD
static const char *const LLDB_NT_OWNER_LINUX
static const char * OSABIAsCString(unsigned char osabi_byte)
static Permissions GetPermissions(const ELFSectionHeader &H)
static const char *const LLDB_NT_OWNER_ANDROID
#define IS_MICROMIPS(ST_OTHER)
static const elf_word LLDB_NT_NETBSD_IDENT_NAMESZ
static uint32_t loongarchVariantFromElfFlags(const elf::ELFHeader &header)
static const elf_word LLDB_NT_GNU_ABI_OS_HURD
static uint32_t mipsVariantFromElfFlags(const elf::ELFHeader &header)
static const char *const LLDB_NT_OWNER_NETBSD
static unsigned ParsePLTRelocations(Symtab *symbol_table, user_id_t start_id, unsigned rel_type, const ELFHeader *hdr, const ELFSectionHeader *rel_hdr, const ELFSectionHeader *plt_hdr, const ELFSectionHeader *sym_hdr, const lldb::SectionSP &plt_section_sp, DataExtractor &rel_data, DataExtractor &symtab_data, DataExtractor &strtab_data)
static void ApplyELF64ABS64Relocation(Symtab *symtab, ELFRelocation &rel, DataExtractor &debug_data, Section *rel_section)
static const elf_word LLDB_NT_GNU_BUILD_ID_TAG
static std::optional< lldb::offset_t > FindSubSubSectionOffsetByTag(const DataExtractor &data, lldb::offset_t offset, unsigned tag)
#define LLDB_PLUGIN_DEFINE(PluginName)
static double elapsed(const StatsTimepoint &start, const StatsTimepoint &end)
#define LLDB_SCOPED_TIMERF(...)
Generic COFF object file reader.
static size_t GetSectionHeaderInfo(SectionHeaderColl §ion_headers, lldb_private::DataExtractor &object_data, const elf::ELFHeader &header, lldb_private::UUID &uuid, std::string &gnu_debuglink_file, uint32_t &gnu_debuglink_crc, lldb_private::ArchSpec &arch_spec)
Parses the elf section headers and returns the uuid, debug link name, crc, archspec.
std::vector< elf::ELFProgramHeader > ProgramHeaderColl
static void DumpELFHeader(lldb_private::Stream *s, const elf::ELFHeader &header)
unsigned ParseTrampolineSymbols(lldb_private::Symtab *symbol_table, lldb::user_id_t start_id, const ELFSectionHeaderInfo *rela_hdr, lldb::user_id_t section_id)
Scans the relocation entries and adds a set of artificial symbols to the given symbol table for each ...
lldb_private::ArchSpec m_arch_spec
The architecture detected from parsing elf file contents.
static void DumpELFSectionHeader_sh_type(lldb_private::Stream *s, elf::elf_word sh_type)
std::shared_ptr< ObjectFileELF > m_gnu_debug_data_object_file
Object file parsed from .gnu_debugdata section (.
SectionHeaderColl::iterator SectionHeaderCollIter
uint32_t m_gnu_debuglink_crc
unsigned RelocateDebugSections(const elf::ELFSectionHeader *rel_hdr, lldb::user_id_t rel_id, lldb_private::Symtab *thetab)
Relocates debug sections.
bool AnySegmentHasPhysicalAddress()
static void DumpELFProgramHeader(lldb_private::Stream *s, const elf::ELFProgramHeader &ph)
lldb_private::Address m_entry_point_address
Cached value of the entry point for this module.
size_t ReadSectionData(lldb_private::Section *section, lldb::offset_t section_offset, void *dst, size_t dst_len) override
llvm::StringRef StripLinkerSymbolAnnotations(llvm::StringRef symbol_name) const override
static void ParseARMAttributes(lldb_private::DataExtractor &data, uint64_t length, lldb_private::ArchSpec &arch_spec)
lldb_private::DataExtractor GetSegmentData(const elf::ELFProgramHeader &H)
void RelocateSection(lldb_private::Section *section) override
Perform relocations on the section if necessary.
FileAddressToAddressClassMap m_address_class_map
The address class for each symbol in the elf file.
static llvm::StringRef GetPluginDescriptionStatic()
static const uint32_t g_core_uuid_magic
bool IsExecutable() const override
Tells whether this object file is capable of being the main executable for a process.
void DumpDependentModules(lldb_private::Stream *s)
ELF dependent module dump routine.
static void DumpELFHeader_e_type(lldb_private::Stream *s, elf::elf_half e_type)
static size_t GetProgramHeaderInfo(ProgramHeaderColl &program_headers, lldb_private::DataExtractor &object_data, const elf::ELFHeader &header)
std::optional< lldb_private::DataExtractor > GetDynsymDataFromDynamic(uint32_t &num_symbols)
Get the bytes that represent the dynamic symbol table from the .dynamic section from process memory.
DynamicSymbolColl m_dynamic_symbols
Collection of symbols from the dynamic table.
static void DumpELFSectionHeader(lldb_private::Stream *s, const ELFSectionHeaderInfo &sh)
std::vector< ELFSectionHeaderInfo > SectionHeaderColl
static void DumpELFHeader_e_ident_EI_DATA(lldb_private::Stream *s, unsigned char ei_data)
lldb_private::ArchSpec GetArchitecture() override
Get the ArchSpec for this object file.
std::optional< lldb_private::FileSpec > GetDebugLink()
Return the contents of the .gnu_debuglink section, if the object file contains it.
lldb_private::AddressClass GetAddressClass(lldb::addr_t file_addr) override
Get the address type given a file address in an object file.
static void DumpELFSectionHeader_sh_flags(lldb_private::Stream *s, elf::elf_xword sh_flags)
lldb_private::UUID GetUUID() override
Gets the UUID for this object file.
std::optional< uint32_t > GetNumSymbolsFromDynamicGnuHash()
Get the number of symbols from the DT_GNU_HASH dynamic entry.
std::optional< lldb_private::DataExtractor > ReadDataFromDynamic(const elf::ELFDynamic *dyn, uint64_t length, uint64_t offset=0)
Read the bytes pointed to by the dyn dynamic entry.
static void DumpELFProgramHeader_p_type(lldb_private::Stream *s, elf::elf_word p_type)
static lldb_private::Status RefineModuleDetailsFromNote(lldb_private::DataExtractor &data, lldb_private::ArchSpec &arch_spec, lldb_private::UUID &uuid)
size_t SectionIndex(const SectionHeaderCollIter &I)
Returns the index of the given section header.
static void DumpELFProgramHeader_p_flags(lldb_private::Stream *s, elf::elf_word p_flags)
static llvm::StringRef GetPluginNameStatic()
size_t ParseDependentModules()
Scans the dynamic section and locates all dependent modules (shared libraries) populating m_filespec_...
void DumpELFSectionHeaders(lldb_private::Stream *s)
static lldb_private::ObjectFile * CreateInstance(const lldb::ModuleSP &module_sp, lldb::DataExtractorSP extractor_sp, lldb::offset_t data_offset, const lldb_private::FileSpec *file, lldb::offset_t file_offset, lldb::offset_t length)
std::shared_ptr< ObjectFileELF > GetGnuDebugDataObjectFile()
Takes the .gnu_debugdata and returns the decompressed object file that is stored within that section.
static lldb::WritableDataBufferSP MapFileDataWritable(const lldb_private::FileSpec &file, uint64_t Size, uint64_t Offset)
void Dump(lldb_private::Stream *s) override
Dump a description of this object to a Stream.
static uint32_t CalculateELFNotesSegmentsCRC32(const ProgramHeaderColl &program_headers, lldb_private::DataExtractor &data)
lldb_private::UUID m_uuid
ELF build ID.
void DumpELFProgramHeaders(lldb_private::Stream *s)
std::pair< unsigned, FileAddressToAddressClassMap > ParseSymbolTable(lldb_private::Symtab *symbol_table, lldb::user_id_t start_id, lldb_private::Section *symtab)
Populates the symbol table with all non-dynamic linker symbols.
size_t ParseDynamicSymbols()
Parses the dynamic symbol table and populates m_dynamic_symbols.
static lldb_private::ModuleSpecList GetModuleSpecifications(const lldb_private::FileSpec &file, lldb::DataExtractorSP &extractor_sp, lldb::offset_t file_offset, lldb::offset_t length)
std::optional< lldb_private::DataExtractor > GetDynamicData()
Get the bytes that represent the .dynamic section.
ObjectFile::Type CalculateType() override
The object file should be able to calculate its type by looking at its file header and possibly the s...
lldb::SectionType GetSectionType(const ELFSectionHeaderInfo &H) const
bool SetLoadAddress(lldb_private::Target &target, lldb::addr_t value, bool value_is_offset) override
Sets the load address for an entire module, assuming a rigid slide of sections, if possible in the im...
std::unique_ptr< lldb_private::FileSpecList > m_filespec_up
List of file specifications corresponding to the modules (shared libraries) on which this object file...
std::optional< uint32_t > GetNumSymbolsFromDynamicHash()
Get the number of symbols from the DT_HASH dynamic entry.
bool ParseProgramHeaders()
Parses all section headers present in this object file and populates m_program_headers.
std::vector< LoadableData > GetLoadableData(lldb_private::Target &target) override
Loads this objfile to memory.
const ELFSectionHeaderInfo * GetSectionHeaderByIndex(lldb::user_id_t id)
Returns the section header with the given id or NULL.
void CreateSections(lldb_private::SectionList &unified_section_list) override
static bool MagicBytesMatch(lldb::DataBufferSP data_sp, lldb::addr_t offset, lldb::addr_t length)
lldb::user_id_t GetSectionIndexByName(const char *name)
Utility method for looking up a section given its name.
ObjectFileELF(const lldb::ModuleSP &module_sp, lldb::DataExtractorSP extractor_sp, lldb::offset_t data_offset, const lldb_private::FileSpec *file, lldb::offset_t offset, lldb::offset_t length)
uint32_t GetAddressByteSize() const override
Gets the address size in bytes for the current object file.
SectionHeaderColl::const_iterator SectionHeaderCollConstIter
ProgramHeaderColl m_program_headers
Collection of program headers.
void DumpELFDynamic(lldb_private::Stream *s)
ELF dump the .dynamic section.
unsigned ApplyRelocations(lldb_private::Symtab *symtab, const elf::ELFHeader *hdr, const elf::ELFSectionHeader *rel_hdr, const elf::ELFSectionHeader *symtab_hdr, const elf::ELFSectionHeader *debug_hdr, lldb_private::DataExtractor &rel_data, lldb_private::DataExtractor &symtab_data, lldb_private::DataExtractor &debug_data, lldb_private::Section *rel_section)
lldb::ByteOrder GetByteOrder() const override
Gets whether endian swapping should occur when extracting data from this object file.
bool ParseHeader() override
Attempts to parse the object header.
elf::ELFHeader m_header
ELF file header.
std::string m_gnu_debuglink_file
ELF .gnu_debuglink file and crc data if available.
void ParseUnwindSymbols(lldb_private::Symtab *symbol_table, lldb_private::DWARFCallFrameInfo *eh_frame)
std::pair< unsigned, FileAddressToAddressClassMap > ParseSymbols(lldb_private::Symtab *symbol_table, lldb::user_id_t start_id, lldb_private::SectionList *section_list, const size_t num_symbols, const lldb_private::DataExtractor &symtab_data, const lldb_private::DataExtractor &strtab_data)
Helper routine for ParseSymbolTable().
SectionHeaderColl m_section_headers
Collection of section headers.
lldb_private::Address GetEntryPointAddress() override
Returns the address of the Entry Point in this object file - if the object file doesn't have an entry...
ObjectFile::Strata CalculateStrata() override
The object file should be able to calculate the strata of the object file.
void ParseSymtab(lldb_private::Symtab &symtab) override
Parse the symbol table into the provides symbol table object.
unsigned PLTRelocationType()
static lldb_private::ObjectFile * CreateMemoryInstance(const lldb::ModuleSP &module_sp, lldb::WritableDataBufferSP data_sp, const lldb::ProcessSP &process_sp, lldb::addr_t header_addr)
lldb::addr_t m_dynamic_base_addr
The file address of the .dynamic section.
uint32_t GetDependentModules(lldb_private::FileSpecList &files) override
Extract the dependent modules from an object file.
size_t ParseSectionHeaders()
Parses all section headers present in this object file and populates m_section_headers.
static void ParseRISCVAttributes(lldb_private::DataExtractor &data, uint64_t length, lldb_private::ArchSpec &arch_spec)
lldb_private::Address GetBaseAddress() override
Returns base address of this object file.
bool IsStripped() override
Detect if this object file has been stripped of local symbols.
const elf::ELFDynamic * FindDynamicSymbol(unsigned tag)
std::map< lldb::addr_t, lldb_private::AddressClass > FileAddressToAddressClassMap
An ordered map of file address to address class.
llvm::ArrayRef< elf::ELFProgramHeader > ProgramHeaders()
std::optional< lldb_private::DataExtractor > GetDynstrData()
Get the bytes that represent the dynamic string table data.
lldb_private::Address GetImageInfoAddress(lldb_private::Target *target) override
Similar to Process::GetImageInfoAddress().
A section + offset based address range class.
A section + offset based address class.
lldb::addr_t GetLoadAddress(Target *target) const
Get the load address.
bool ResolveAddressUsingFileSections(lldb::addr_t addr, const SectionList *sections)
Resolve a file virtual address using a section list.
lldb::SectionSP GetSection() const
Get const accessor for the section.
bool Slide(int64_t offset)
lldb::addr_t GetFileAddress() const
Get the file address.
bool IsValid() const
Check if the object state is valid.
bool SetOffset(lldb::addr_t offset)
Set accessor for the offset.
An architecture specification class.
@ eLoongArchSubType_loongarch64
@ eLoongArchSubType_unknown
@ eLoongArchSubType_loongarch32
bool IsValid() const
Tests if this ArchSpec is valid.
llvm::Triple & GetTriple()
Architecture triple accessor.
void SetFlags(uint32_t flags)
bool SetArchitecture(ArchitectureType arch_type, uint32_t cpu, uint32_t sub, uint32_t os=0)
Change the architecture object type, CPU type and OS type.
@ eLoongArch_abi_single_float
soft float
@ eLoongArch_abi_double_float
single precision floating point, +f
bool IsMIPS() const
if MIPS architecture return true.
uint32_t GetFlags() const
@ eMIPSSubType_mips32r6el
@ eMIPSSubType_mips64r6el
@ eMIPSSubType_mips64r2el
@ eMIPSSubType_mips32r2el
llvm::Triple::ArchType GetMachine() const
Returns a machine family for the current architecture.
@ eRISCV_float_abi_double
single precision floating point, +f
@ eRISCV_float_abi_quad
double precision floating point, +d
@ eRISCV_float_abi_single
soft float
const char * GetArchitectureName() const
Returns a static string representing the current architecture.
void SetSubtargetFeatures(llvm::SubtargetFeatures &&subtarget_features)
A uniqued constant string class.
llvm::StringRef GetStringRef() const
Get the string value as a llvm::StringRef.
const char * GetCString() const
Get the string value as a C string.
const char * AsCString(const char *value_if_empty) const
Get the string value as a C string.
void ForEachFDEEntries(const std::function< bool(lldb::addr_t, uint32_t, dw_offset_t)> &callback)
A subclass of DataBuffer that stores a data buffer on the heap.
static void ReportWarning(std::string message, std::optional< lldb::user_id_t > debugger_id=std::nullopt, std::once_flag *once=nullptr)
Report warning events.
A class that measures elapsed time in an exception safe way.
bool AppendIfUnique(const FileSpec &file)
Append a FileSpec object if unique.
FileSpec CopyByAppendingPathComponent(llvm::StringRef component) const
const ConstString & GetFilename() const
Filename string const get accessor.
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
std::shared_ptr< WritableDataBuffer > CreateWritableDataBuffer(const llvm::Twine &path, uint64_t size=0, uint64_t offset=0)
static FileSystem & Instance()
void Resolve(llvm::SmallVectorImpl< char > &path, bool force_make_absolute=false)
Resolve path to make it canonical.
ValueType Get() const
Get accessor for all flags.
bool Test(ValueType bit) const
Test a single flag bit.
A class that handles mangled names.
void SetDemangledName(ConstString name)
ConstString GetMangledName() const
Mangled name get accessor.
ConstString GetDemangledName() const
Demangled name get accessor.
void SetMangledName(ConstString name)
ConstString GetName(NamePreference preference=ePreferDemangled) const
Best name get accessor.
lldb::ModuleSP GetModule() const
Get const accessor for the module pointer.
void Append(const ModuleSpec &spec)
void SetObjectSize(uint64_t object_size)
ArchSpec & GetArchitecture()
void SetObjectOffset(uint64_t object_offset)
std::unique_ptr< lldb_private::SectionList > m_sections_up
static lldb::DataBufferSP MapFileData(const FileSpec &file, uint64_t Size, uint64_t Offset)
const lldb::addr_t m_memory_addr
Set if the object file only exists in memory.
static lldb::SectionType GetDWARFSectionTypeFromName(llvm::StringRef name)
Parses the section type from a section name for DWARF sections.
virtual void ParseSymtab(Symtab &symtab)=0
Parse the symbol table into the provides symbol table object.
virtual AddressClass GetAddressClass(lldb::addr_t file_addr)
Get the address type given a file address in an object file.
Symtab * GetSymtab(bool can_create=true)
Gets the symbol table for the currently selected architecture (and object for archives).
DataExtractorNSP m_data_nsp
The data for this object file so things can be parsed lazily.
static lldb::WritableDataBufferSP ReadMemory(const lldb::ProcessSP &process_sp, lldb::addr_t addr, size_t byte_size)
@ eTypeExecutable
A normal executable.
@ eTypeDebugInfo
An object file that contains only debug information.
@ eTypeObjectFile
An intermediate object file.
@ eTypeCoreFile
A core file that has a checkpoint of a program's execution state.
@ eTypeSharedLibrary
A shared library that can be used during execution.
virtual FileSpec & GetFileSpec()
Get accessor to the object file specification.
size_t GetData(lldb::offset_t offset, size_t length, lldb::DataExtractorSP &data_sp) const
virtual SectionList * GetSectionList(bool update_module_section_list=true)
Gets the section list for the currently selected architecture (and object for archives).
ObjectFile(const lldb::ModuleSP &module_sp, const FileSpec *file_spec_ptr, lldb::offset_t file_offset, lldb::offset_t length, lldb::DataExtractorSP extractor_sp, lldb::offset_t data_offset)
Construct with a parent module, offset, and header data.
bool IsInMemory() const
Returns true if the object file exists only in memory.
lldb::ProcessWP m_process_wp
virtual size_t ReadSectionData(Section *section, lldb::offset_t section_offset, void *dst, size_t dst_len)
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
A Progress indicator helper class.
lldb::SectionSP FindSectionByName(ConstString section_dstr) const
static SectionList Merge(SectionList &lhs, SectionList &rhs, MergeCallback filter)
lldb::SectionSP FindSectionByID(lldb::user_id_t sect_id) const
lldb::SectionSP FindSectionContainingFileAddress(lldb::addr_t addr, uint32_t depth=UINT32_MAX) const
size_t AddSection(const lldb::SectionSP §ion_sp)
bool ReplaceSection(lldb::user_id_t sect_id, const lldb::SectionSP §ion_sp, uint32_t depth=UINT32_MAX)
lldb::SectionSP FindSectionByType(lldb::SectionType sect_type, bool check_children, size_t start_idx=0) const
void Dump(llvm::raw_ostream &s, unsigned indent, Target *target, bool show_header, uint32_t depth) const
lldb::SectionSP GetSectionAtIndex(size_t idx) const
ConstString GetName() const
void SetIsRelocated(bool b)
lldb::offset_t GetFileOffset() const
ObjectFile * GetObjectFile()
lldb::offset_t GetFileSize() const
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
static Status FromErrorString(const char *str)
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
void Format(const char *format, Args &&... args)
Forwards the arguments to llvm::formatv and writes to the stream.
llvm::raw_ostream & AsRawOstream()
Returns a raw_ostream that forwards the data to this Stream object.
size_t Indent(llvm::StringRef s="")
Indent the current line in the stream.
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
size_t PutCString(llvm::StringRef cstr)
Output a C string to the stream.
size_t EOL()
Output and End of Line character to the stream.
unsigned GetIndentLevel() const
Get the current indentation level.
void SetSizeIsSynthesized(bool b)
bool GetByteSizeIsValid() const
Address & GetAddressRef()
ConstString GetName() const
void SetByteSize(lldb::addr_t size)
Symbol * FindSymbolByID(lldb::user_id_t uid) const
Symbol * SymbolAtIndex(size_t idx)
Symbol * FindSymbolAtFileAddress(lldb::addr_t file_addr)
Symbol * FindSymbolContainingFileAddress(lldb::addr_t file_addr)
uint32_t AddSymbol(const Symbol &symbol)
void Dump(Stream *s, Target *target, SortOrder sort_type, Mangled::NamePreference name_preference=Mangled::ePreferDemangled)
ObjectFile * GetObjectFile() const
size_t GetNumSymbols() const
bool ReadPointerFromMemory(const Address &addr, Status &error, Address &pointer_addr, bool force_live_memory=false)
uint64_t ReadUnsignedIntegerFromMemory(const Address &addr, size_t integer_byte_size, uint64_t fail_value, Status &error, bool force_live_memory=false)
bool SetSectionLoadAddress(const lldb::SectionSP §ion, lldb::addr_t load_addr, bool warn_multiple=false)
Represents UUID's of various sizes.
uint8_t * GetBytes()
Get a pointer to the data.
#define LLDB_INVALID_CPUTYPE
#define UNUSED_IF_ASSERT_DISABLED(x)
#define LLDB_INVALID_ADDRESS
llvm::Error uncompress(llvm::ArrayRef< uint8_t > InputBuffer, llvm::SmallVectorImpl< uint8_t > &Uncompressed)
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
std::shared_ptr< lldb_private::Process > ProcessSP
ByteOrder
Byte ordering definitions.
std::shared_ptr< lldb_private::DataBuffer > DataBufferSP
std::shared_ptr< lldb_private::Section > SectionSP
std::shared_ptr< lldb_private::WritableDataBuffer > WritableDataBufferSP
@ eSectionTypeELFDynamicSymbols
Elf SHT_DYNSYM section.
@ eSectionTypeContainer
The section contains child sections.
@ eSectionTypeELFDynamicLinkInfo
Elf SHT_DYNAMIC section.
@ eSectionTypeAbsoluteAddress
Dummy section for symbols with absolute address.
@ eSectionTypeELFRelocationEntries
Elf SHT_REL or SHT_REL section.
@ eSectionTypeLLDBFormatters
@ eSectionTypeLLDBTypeSummaries
@ eSectionTypeSwiftModules
@ eSectionTypeDWARFGNUDebugAltLink
@ eSectionTypeELFSymbolTable
Elf SHT_SYMTAB section.
std::shared_ptr< lldb_private::DataExtractor > DataExtractorSP
std::shared_ptr< lldb_private::Module > ModuleSP
bool Parse(const lldb_private::DataExtractor &data, lldb::offset_t *offset)
Parse an ELFNote entry from the given DataExtractor starting at position offset.
Represents an entry in an ELF dynamic table.
elf_addr d_ptr
Pointer value of the table entry.
elf_xword d_val
Integer value of the table entry.
bool Parse(const lldb_private::DataExtractor &data, lldb::offset_t *offset)
Parse an ELFDynamic entry from the given DataExtractor starting at position offset.
elf_sxword d_tag
Type of dynamic table entry.
static unsigned RelocSymbol64(const ELFRel &rel)
Returns the symbol index when the given entry represents a 64-bit relocation.
static unsigned RelocType64(const ELFRel &rel)
Returns the type when the given entry represents a 64-bit relocation.
static unsigned RelocType32(const ELFRel &rel)
Returns the type when the given entry represents a 32-bit relocation.
static unsigned RelocSymbol32(const ELFRel &rel)
Returns the symbol index when the given entry represents a 32-bit relocation.
static unsigned RelocSymbol64(const ELFRela &rela)
Returns the symbol index when the given entry represents a 64-bit relocation.
static unsigned RelocType64(const ELFRela &rela)
Returns the type when the given entry represents a 64-bit relocation.
static unsigned RelocType32(const ELFRela &rela)
Returns the type when the given entry represents a 32-bit relocation.
static unsigned RelocSymbol32(const ELFRela &rela)
Returns the symbol index when the given entry represents a 32-bit relocation.
Represents a symbol within an ELF symbol table.
unsigned char getType() const
Returns the type attribute of the st_info member.
elf_half st_shndx
Section to which this symbol applies.
unsigned char st_info
Symbol type and binding attributes.
unsigned char getBinding() const
Returns the binding attribute of the st_info member.
bool Parse(const lldb_private::DataExtractor &data, lldb::offset_t *offset)
Parse an ELFSymbol entry from the given DataExtractor starting at position offset.
elf_addr st_value
Absolute or relocatable address.
elf_word st_name
Symbol name string index.
elf_xword st_size
Size of the symbol or zero.
unsigned char st_other
Reserved for future use.
llvm::ArrayRef< uint8_t > Contents
lldb::user_id_t GetID() const
Get accessor for the user ID.