|  | //===-- 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<SBSection>(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 &)); | 
|  | } | 
|  |  | 
|  | } | 
|  | } |