hanchenye-llvm-project/lldb/source/API/SBSection.cpp

325 lines
7.8 KiB
C++
Raw Normal View History

//===-- SBSection.cpp -------------------------------------------*- C++ -*-===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include "lldb/API/SBSection.h"
#include "lldb/API/SBStream.h"
#include "lldb/Core/DataBuffer.h"
#include "lldb/Core/DataExtractor.h"
#include "lldb/Core/Log.h"
#include "lldb/Core/Section.h"
#include "lldb/Core/Module.h"
namespace lldb_private
{
// We need a section implementation to hold onto a reference to the module
// since if the module goes away and we have anyone still holding onto a
// SBSection object, we could crash.
class SectionImpl
{
public:
SectionImpl (const lldb_private::Section *section = NULL) :
m_module_sp (),
m_section (section)
{
if (section)
m_module_sp = section->GetModule();
}
SectionImpl (const SectionImpl &rhs) :
m_module_sp (rhs.m_module_sp),
m_section (rhs.m_section)
{
}
bool
IsValid () const
{
return m_section != NULL;
}
void
operator = (const SectionImpl &rhs)
{
m_module_sp = rhs.m_module_sp;
m_section = rhs.m_section;
}
void
operator =(const lldb_private::Section *section)
{
m_section = section;
if (section)
m_module_sp.reset(section->GetModule());
else
m_module_sp.reset();
}
const lldb_private::Section *
GetSection () const
{
return m_section;
}
Module *
GetModule()
{
return m_module_sp.get();
}
const lldb::ModuleSP &
GetModuleSP() const
{
return m_module_sp;
}
protected:
lldb::ModuleSP m_module_sp;
const lldb_private::Section *m_section;
};
}
using namespace lldb;
using namespace lldb_private;
SBSection::SBSection () :
m_opaque_ap ()
{
}
SBSection::SBSection (const SBSection &rhs) :
m_opaque_ap ()
{
if (rhs.IsValid())
m_opaque_ap.reset (new SectionImpl (*rhs.m_opaque_ap));
}
SBSection::SBSection (const lldb_private::Section *section) :
m_opaque_ap ()
{
if (section)
m_opaque_ap.reset (new SectionImpl(section));
}
const SBSection &
SBSection::operator = (const SBSection &rhs)
{
if (this != &rhs && rhs.IsValid())
m_opaque_ap.reset (new SectionImpl(*rhs.m_opaque_ap));
else
m_opaque_ap.reset ();
return *this;
}
SBSection::~SBSection ()
{
}
bool
SBSection::IsValid () const
{
return m_opaque_ap.get() != NULL && m_opaque_ap->IsValid();
}
lldb::SBSection
SBSection::FindSubSection (const char *sect_name)
{
lldb::SBSection sb_section;
if (IsValid())
{
ConstString const_sect_name(sect_name);
sb_section.SetSection(m_opaque_ap->GetSection()->GetChildren ().FindSectionByName(const_sect_name).get());
}
return sb_section;
}
size_t
SBSection::GetNumSubSections ()
{
if (IsValid())
return m_opaque_ap->GetSection()->GetChildren ().GetSize();
return 0;
}
lldb::SBSection
SBSection::GetSubSectionAtIndex (size_t idx)
{
lldb::SBSection sb_section;
if (IsValid())
sb_section.SetSection(m_opaque_ap->GetSection()->GetChildren ().GetSectionAtIndex(idx).get());
return sb_section;
}
const lldb_private::Section *
SBSection::GetSection()
{
if (m_opaque_ap.get())
return m_opaque_ap->GetSection();
return NULL;
}
void
SBSection::SetSection (const lldb_private::Section *section)
{
m_opaque_ap.reset (new SectionImpl(section));
}
lldb::addr_t
SBSection::GetFileAddress ()
{
lldb::addr_t file_addr = LLDB_INVALID_ADDRESS;
if (IsValid())
return m_opaque_ap->GetSection()->GetFileAddress();
return file_addr;
}
lldb::addr_t
SBSection::GetByteSize ()
{
if (IsValid())
{
const Section *section = m_opaque_ap->GetSection();
if (section)
return section->GetByteSize();
}
return 0;
}
uint64_t
SBSection::GetFileOffset ()
{
if (IsValid())
{
const Section *section = m_opaque_ap->GetSection();
if (section)
{
Module *module = m_opaque_ap->GetModule();
if (module)
{
ObjectFile *objfile = module->GetObjectFile();
if (objfile)
return objfile->GetOffset() + section->GetFileOffset();
}
return section->GetFileOffset();
}
}
return 0;
}
uint64_t
SBSection::GetFileByteSize ()
{
if (IsValid())
{
const Section *section = m_opaque_ap->GetSection();
if (section)
return section->GetFileSize();
}
return 0;
}
SBData
SBSection::GetSectionData (uint64_t offset, uint64_t size)
{
SBData sb_data;
if (IsValid())
{
const Section *section = m_opaque_ap->GetSection();
if (section)
{
const uint64_t sect_file_size = section->GetFileSize();
if (sect_file_size > 0)
{
Module *module = m_opaque_ap->GetModule();
if (module)
{
ObjectFile *objfile = module->GetObjectFile();
if (objfile)
{
const uint64_t sect_file_offset = objfile->GetOffset() + section->GetFileOffset();
const uint64_t file_offset = sect_file_offset + offset;
uint64_t file_size = size;
if (file_size == UINT64_MAX)
{
file_size = section->GetByteSize();
if (file_size > offset)
file_size -= offset;
else
file_size = 0;
}
DataBufferSP data_buffer_sp (objfile->GetFileSpec().ReadFileContents (file_offset, file_size));
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 sb_data;
}
SectionType
SBSection::GetSectionType ()
{
if (m_opaque_ap.get())
{
const Section *section = m_opaque_ap->GetSection();
if (section)
return section->GetType();
}
return eSectionTypeInvalid;
}
bool
SBSection::operator == (const SBSection &rhs)
{
SectionImpl *lhs_ptr = m_opaque_ap.get();
SectionImpl *rhs_ptr = rhs.m_opaque_ap.get();
if (lhs_ptr && rhs_ptr)
return lhs_ptr->GetSection() == rhs_ptr->GetSection();
return false;
}
bool
SBSection::operator != (const SBSection &rhs)
{
SectionImpl *lhs_ptr = m_opaque_ap.get();
SectionImpl *rhs_ptr = rhs.m_opaque_ap.get();
if (lhs_ptr && rhs_ptr)
return lhs_ptr->GetSection() != rhs_ptr->GetSection();
return false;
}
bool
SBSection::GetDescription (SBStream &description)
{
if (m_opaque_ap.get())
{
description.Printf ("SBSection");
}
else
{
description.Printf ("No value");
}
return true;
}