//===-- SBSection.cpp -----------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "lldb/API/SBSection.h" #include "SBReproducerPrivate.h" #include "lldb/API/SBStream.h" #include "lldb/API/SBTarget.h" #include "lldb/Core/Module.h" #include "lldb/Core/Section.h" #include "lldb/Symbol/ObjectFile.h" #include "lldb/Utility/DataBuffer.h" #include "lldb/Utility/DataExtractor.h" #include "lldb/Utility/StreamString.h" using namespace lldb; using namespace lldb_private; SBSection::SBSection() : m_opaque_wp() { LLDB_RECORD_CONSTRUCTOR_NO_ARGS(SBSection); } SBSection::SBSection(const SBSection &rhs) : m_opaque_wp(rhs.m_opaque_wp) { LLDB_RECORD_CONSTRUCTOR(SBSection, (const lldb::SBSection &), rhs); } SBSection::SBSection(const lldb::SectionSP §ion_sp) : m_opaque_wp() // Don't init with section_sp otherwise this will throw if // section_sp doesn't contain a valid Section * { if (section_sp) m_opaque_wp = section_sp; } const SBSection &SBSection::operator=(const SBSection &rhs) { LLDB_RECORD_METHOD(const lldb::SBSection &, SBSection, operator=,(const lldb::SBSection &), rhs); m_opaque_wp = rhs.m_opaque_wp; return LLDB_RECORD_RESULT(*this); } SBSection::~SBSection() = default; bool SBSection::IsValid() const { LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBSection, IsValid); return this->operator bool(); } SBSection::operator bool() const { LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBSection, operator bool); SectionSP section_sp(GetSP()); return section_sp && section_sp->GetModule().get() != nullptr; } const char *SBSection::GetName() { LLDB_RECORD_METHOD_NO_ARGS(const char *, SBSection, GetName); SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetName().GetCString(); return nullptr; } lldb::SBSection SBSection::GetParent() { LLDB_RECORD_METHOD_NO_ARGS(lldb::SBSection, SBSection, GetParent); lldb::SBSection sb_section; SectionSP section_sp(GetSP()); if (section_sp) { SectionSP parent_section_sp(section_sp->GetParent()); if (parent_section_sp) sb_section.SetSP(parent_section_sp); } return LLDB_RECORD_RESULT(sb_section); } lldb::SBSection SBSection::FindSubSection(const char *sect_name) { LLDB_RECORD_METHOD(lldb::SBSection, SBSection, FindSubSection, (const char *), sect_name); lldb::SBSection sb_section; if (sect_name) { SectionSP section_sp(GetSP()); if (section_sp) { ConstString const_sect_name(sect_name); sb_section.SetSP( section_sp->GetChildren().FindSectionByName(const_sect_name)); } } return LLDB_RECORD_RESULT(sb_section); } size_t SBSection::GetNumSubSections() { LLDB_RECORD_METHOD_NO_ARGS(size_t, SBSection, GetNumSubSections); SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetChildren().GetSize(); return 0; } lldb::SBSection SBSection::GetSubSectionAtIndex(size_t idx) { LLDB_RECORD_METHOD(lldb::SBSection, SBSection, GetSubSectionAtIndex, (size_t), idx); lldb::SBSection sb_section; SectionSP section_sp(GetSP()); if (section_sp) sb_section.SetSP(section_sp->GetChildren().GetSectionAtIndex(idx)); return LLDB_RECORD_RESULT(sb_section); } lldb::SectionSP SBSection::GetSP() const { return m_opaque_wp.lock(); } void SBSection::SetSP(const lldb::SectionSP §ion_sp) { m_opaque_wp = section_sp; } lldb::addr_t SBSection::GetFileAddress() { LLDB_RECORD_METHOD_NO_ARGS(lldb::addr_t, SBSection, GetFileAddress); lldb::addr_t file_addr = LLDB_INVALID_ADDRESS; SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetFileAddress(); return file_addr; } lldb::addr_t SBSection::GetLoadAddress(lldb::SBTarget &sb_target) { LLDB_RECORD_METHOD(lldb::addr_t, SBSection, GetLoadAddress, (lldb::SBTarget &), sb_target); TargetSP target_sp(sb_target.GetSP()); if (target_sp) { SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetLoadBaseAddress(target_sp.get()); } return LLDB_INVALID_ADDRESS; } lldb::addr_t SBSection::GetByteSize() { LLDB_RECORD_METHOD_NO_ARGS(lldb::addr_t, SBSection, GetByteSize); SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetByteSize(); return 0; } uint64_t SBSection::GetFileOffset() { LLDB_RECORD_METHOD_NO_ARGS(uint64_t, SBSection, GetFileOffset); SectionSP section_sp(GetSP()); if (section_sp) { ModuleSP module_sp(section_sp->GetModule()); if (module_sp) { ObjectFile *objfile = module_sp->GetObjectFile(); if (objfile) return objfile->GetFileOffset() + section_sp->GetFileOffset(); } } return UINT64_MAX; } uint64_t SBSection::GetFileByteSize() { LLDB_RECORD_METHOD_NO_ARGS(uint64_t, SBSection, GetFileByteSize); SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetFileSize(); return 0; } SBData SBSection::GetSectionData() { LLDB_RECORD_METHOD_NO_ARGS(lldb::SBData, SBSection, GetSectionData); return LLDB_RECORD_RESULT(GetSectionData(0, UINT64_MAX)); } SBData SBSection::GetSectionData(uint64_t offset, uint64_t size) { LLDB_RECORD_METHOD(lldb::SBData, SBSection, GetSectionData, (uint64_t, uint64_t), offset, size); SBData sb_data; SectionSP section_sp(GetSP()); if (section_sp) { const uint64_t sect_file_size = section_sp->GetFileSize(); if (sect_file_size > 0) { ModuleSP module_sp(section_sp->GetModule()); if (module_sp) { ObjectFile *objfile = module_sp->GetObjectFile(); if (objfile) { const uint64_t sect_file_offset = objfile->GetFileOffset() + section_sp->GetFileOffset(); const uint64_t file_offset = sect_file_offset + offset; uint64_t file_size = size; if (file_size == UINT64_MAX) { file_size = section_sp->GetByteSize(); if (file_size > offset) file_size -= offset; else file_size = 0; } auto data_buffer_sp = FileSystem::Instance().CreateDataBuffer( objfile->GetFileSpec().GetPath(), file_size, file_offset); if (data_buffer_sp && data_buffer_sp->GetByteSize() > 0) { DataExtractorSP data_extractor_sp( new DataExtractor(data_buffer_sp, objfile->GetByteOrder(), objfile->GetAddressByteSize())); sb_data.SetOpaque(data_extractor_sp); } } } } } return LLDB_RECORD_RESULT(sb_data); } SectionType SBSection::GetSectionType() { LLDB_RECORD_METHOD_NO_ARGS(lldb::SectionType, SBSection, GetSectionType); SectionSP section_sp(GetSP()); if (section_sp.get()) return section_sp->GetType(); return eSectionTypeInvalid; } uint32_t SBSection::GetPermissions() const { LLDB_RECORD_METHOD_CONST_NO_ARGS(uint32_t, SBSection, GetPermissions); SectionSP section_sp(GetSP()); if (section_sp) return section_sp->GetPermissions(); return 0; } uint32_t SBSection::GetTargetByteSize() { LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBSection, GetTargetByteSize); SectionSP section_sp(GetSP()); if (section_sp.get()) return section_sp->GetTargetByteSize(); return 0; } bool SBSection::operator==(const SBSection &rhs) { LLDB_RECORD_METHOD(bool, SBSection, operator==,(const lldb::SBSection &), rhs); SectionSP lhs_section_sp(GetSP()); SectionSP rhs_section_sp(rhs.GetSP()); if (lhs_section_sp && rhs_section_sp) return lhs_section_sp == rhs_section_sp; return false; } bool SBSection::operator!=(const SBSection &rhs) { LLDB_RECORD_METHOD(bool, SBSection, operator!=,(const lldb::SBSection &), rhs); SectionSP lhs_section_sp(GetSP()); SectionSP rhs_section_sp(rhs.GetSP()); return lhs_section_sp != rhs_section_sp; } bool SBSection::GetDescription(SBStream &description) { LLDB_RECORD_METHOD(bool, SBSection, GetDescription, (lldb::SBStream &), description); Stream &strm = description.ref(); SectionSP section_sp(GetSP()); if (section_sp) { const addr_t file_addr = section_sp->GetFileAddress(); strm.Printf("[0x%16.16" PRIx64 "-0x%16.16" PRIx64 ") ", file_addr, file_addr + section_sp->GetByteSize()); section_sp->DumpName(strm.AsRawOstream()); } else { strm.PutCString("No value"); } return true; } namespace lldb_private { namespace repro { template <> void RegisterMethods(Registry &R) { LLDB_REGISTER_CONSTRUCTOR(SBSection, ()); LLDB_REGISTER_CONSTRUCTOR(SBSection, (const lldb::SBSection &)); LLDB_REGISTER_METHOD(const lldb::SBSection &, SBSection, operator=,(const lldb::SBSection &)); LLDB_REGISTER_METHOD_CONST(bool, SBSection, IsValid, ()); LLDB_REGISTER_METHOD_CONST(bool, SBSection, operator bool, ()); LLDB_REGISTER_METHOD(const char *, SBSection, GetName, ()); LLDB_REGISTER_METHOD(lldb::SBSection, SBSection, GetParent, ()); LLDB_REGISTER_METHOD(lldb::SBSection, SBSection, FindSubSection, (const char *)); LLDB_REGISTER_METHOD(size_t, SBSection, GetNumSubSections, ()); LLDB_REGISTER_METHOD(lldb::SBSection, SBSection, GetSubSectionAtIndex, (size_t)); LLDB_REGISTER_METHOD(lldb::addr_t, SBSection, GetFileAddress, ()); LLDB_REGISTER_METHOD(lldb::addr_t, SBSection, GetLoadAddress, (lldb::SBTarget &)); LLDB_REGISTER_METHOD(lldb::addr_t, SBSection, GetByteSize, ()); LLDB_REGISTER_METHOD(uint64_t, SBSection, GetFileOffset, ()); LLDB_REGISTER_METHOD(uint64_t, SBSection, GetFileByteSize, ()); LLDB_REGISTER_METHOD(lldb::SBData, SBSection, GetSectionData, ()); LLDB_REGISTER_METHOD(lldb::SBData, SBSection, GetSectionData, (uint64_t, uint64_t)); LLDB_REGISTER_METHOD(lldb::SectionType, SBSection, GetSectionType, ()); LLDB_REGISTER_METHOD_CONST(uint32_t, SBSection, GetPermissions, ()); LLDB_REGISTER_METHOD(uint32_t, SBSection, GetTargetByteSize, ()); LLDB_REGISTER_METHOD(bool, SBSection, operator==,(const lldb::SBSection &)); LLDB_REGISTER_METHOD(bool, SBSection, operator!=,(const lldb::SBSection &)); LLDB_REGISTER_METHOD(bool, SBSection, GetDescription, (lldb::SBStream &)); } } }