Path: blob/main/contrib/llvm-project/lldb/source/Core/AddressRange.cpp
39587 views
//===-- AddressRange.cpp --------------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#include "lldb/Core/AddressRange.h"9#include "lldb/Core/Module.h"10#include "lldb/Core/Section.h"11#include "lldb/Target/Target.h"12#include "lldb/Utility/ConstString.h"13#include "lldb/Utility/FileSpec.h"14#include "lldb/Utility/Stream.h"15#include "lldb/lldb-defines.h"16#include "lldb/lldb-types.h"1718#include "llvm/Support/Compiler.h"1920#include <memory>2122#include <cinttypes>2324namespace lldb_private {25class SectionList;26}2728using namespace lldb;29using namespace lldb_private;3031AddressRange::AddressRange() : m_base_addr() {}3233AddressRange::AddressRange(addr_t file_addr, addr_t byte_size,34const SectionList *section_list)35: m_base_addr(file_addr, section_list), m_byte_size(byte_size) {}3637AddressRange::AddressRange(const lldb::SectionSP §ion, addr_t offset,38addr_t byte_size)39: m_base_addr(section, offset), m_byte_size(byte_size) {}4041AddressRange::AddressRange(const Address &so_addr, addr_t byte_size)42: m_base_addr(so_addr), m_byte_size(byte_size) {}4344AddressRange::~AddressRange() = default;4546bool AddressRange::Contains(const Address &addr) const {47SectionSP range_sect_sp = GetBaseAddress().GetSection();48SectionSP addr_sect_sp = addr.GetSection();49if (range_sect_sp) {50if (!addr_sect_sp ||51range_sect_sp->GetModule() != addr_sect_sp->GetModule())52return false; // Modules do not match.53} else if (addr_sect_sp) {54return false; // Range has no module but "addr" does because addr has a55// section56}57// Either the modules match, or both have no module, so it is ok to compare58// the file addresses in this case only.59return ContainsFileAddress(addr);60}6162bool AddressRange::ContainsFileAddress(const Address &addr) const {63if (addr.GetSection() == m_base_addr.GetSection())64return (addr.GetOffset() - m_base_addr.GetOffset()) < GetByteSize();65addr_t file_base_addr = GetBaseAddress().GetFileAddress();66if (file_base_addr == LLDB_INVALID_ADDRESS)67return false;6869addr_t file_addr = addr.GetFileAddress();70if (file_addr == LLDB_INVALID_ADDRESS)71return false;7273if (file_base_addr <= file_addr)74return (file_addr - file_base_addr) < GetByteSize();7576return false;77}7879bool AddressRange::ContainsFileAddress(addr_t file_addr) const {80if (file_addr == LLDB_INVALID_ADDRESS)81return false;8283addr_t file_base_addr = GetBaseAddress().GetFileAddress();84if (file_base_addr == LLDB_INVALID_ADDRESS)85return false;8687if (file_base_addr <= file_addr)88return (file_addr - file_base_addr) < GetByteSize();8990return false;91}9293bool AddressRange::ContainsLoadAddress(const Address &addr,94Target *target) const {95if (addr.GetSection() == m_base_addr.GetSection())96return (addr.GetOffset() - m_base_addr.GetOffset()) < GetByteSize();97addr_t load_base_addr = GetBaseAddress().GetLoadAddress(target);98if (load_base_addr == LLDB_INVALID_ADDRESS)99return false;100101addr_t load_addr = addr.GetLoadAddress(target);102if (load_addr == LLDB_INVALID_ADDRESS)103return false;104105if (load_base_addr <= load_addr)106return (load_addr - load_base_addr) < GetByteSize();107108return false;109}110111bool AddressRange::ContainsLoadAddress(addr_t load_addr, Target *target) const {112if (load_addr == LLDB_INVALID_ADDRESS)113return false;114115addr_t load_base_addr = GetBaseAddress().GetLoadAddress(target);116if (load_base_addr == LLDB_INVALID_ADDRESS)117return false;118119if (load_base_addr <= load_addr)120return (load_addr - load_base_addr) < GetByteSize();121122return false;123}124125bool AddressRange::Extend(const AddressRange &rhs_range) {126addr_t lhs_end_addr = GetBaseAddress().GetFileAddress() + GetByteSize();127addr_t rhs_base_addr = rhs_range.GetBaseAddress().GetFileAddress();128129if (!ContainsFileAddress(rhs_range.GetBaseAddress()) &&130lhs_end_addr != rhs_base_addr)131// The ranges don't intersect at all on the right side of this range.132return false;133134addr_t rhs_end_addr = rhs_base_addr + rhs_range.GetByteSize();135if (lhs_end_addr >= rhs_end_addr)136// The rhs range totally overlaps this one, nothing to add.137return false;138139m_byte_size += rhs_end_addr - lhs_end_addr;140return true;141}142143void AddressRange::Clear() {144m_base_addr.Clear();145m_byte_size = 0;146}147148bool AddressRange::IsValid() const {149return m_base_addr.IsValid() && (m_byte_size > 0);150}151152bool AddressRange::Dump(Stream *s, Target *target, Address::DumpStyle style,153Address::DumpStyle fallback_style) const {154addr_t vmaddr = LLDB_INVALID_ADDRESS;155int addr_size = sizeof(addr_t);156if (target)157addr_size = target->GetArchitecture().GetAddressByteSize();158159bool show_module = false;160switch (style) {161default:162break;163case Address::DumpStyleSectionNameOffset:164case Address::DumpStyleSectionPointerOffset:165s->PutChar('[');166m_base_addr.Dump(s, target, style, fallback_style);167s->PutChar('-');168DumpAddress(s->AsRawOstream(), m_base_addr.GetOffset() + GetByteSize(),169addr_size);170s->PutChar(')');171return true;172break;173174case Address::DumpStyleModuleWithFileAddress:175show_module = true;176[[fallthrough]];177case Address::DumpStyleFileAddress:178vmaddr = m_base_addr.GetFileAddress();179break;180181case Address::DumpStyleLoadAddress:182vmaddr = m_base_addr.GetLoadAddress(target);183break;184}185186if (vmaddr != LLDB_INVALID_ADDRESS) {187if (show_module) {188ModuleSP module_sp(GetBaseAddress().GetModule());189if (module_sp)190s->Printf("%s", module_sp->GetFileSpec().GetFilename().AsCString(191"<Unknown>"));192}193DumpAddressRange(s->AsRawOstream(), vmaddr, vmaddr + GetByteSize(),194addr_size);195return true;196} else if (fallback_style != Address::DumpStyleInvalid) {197return Dump(s, target, fallback_style, Address::DumpStyleInvalid);198}199200return false;201}202203void AddressRange::DumpDebug(Stream *s) const {204s->Printf("%p: AddressRange section = %p, offset = 0x%16.16" PRIx64205", byte_size = 0x%16.16" PRIx64 "\n",206static_cast<const void *>(this),207static_cast<void *>(m_base_addr.GetSection().get()),208m_base_addr.GetOffset(), GetByteSize());209}210211bool AddressRange::GetDescription(Stream *s, Target *target) const {212addr_t start_addr = m_base_addr.GetLoadAddress(target);213if (start_addr != LLDB_INVALID_ADDRESS) {214// We have a valid target and the address was resolved, or we have a base215// address with no section. Just print out a raw address range: [<addr>,216// <addr>)217s->Printf("[0x%" PRIx64 "-0x%" PRIx64 ")", start_addr,218start_addr + GetByteSize());219return true;220}221222// Either no target or the address wasn't resolved, print as223// <module>[<file-addr>-<file-addr>)224const char *file_name = "";225const auto section_sp = m_base_addr.GetSection();226if (section_sp) {227if (const auto object_file = section_sp->GetObjectFile())228file_name = object_file->GetFileSpec().GetFilename().AsCString();229}230start_addr = m_base_addr.GetFileAddress();231const addr_t end_addr = (start_addr == LLDB_INVALID_ADDRESS)232? LLDB_INVALID_ADDRESS233: start_addr + GetByteSize();234s->Printf("%s[0x%" PRIx64 "-0x%" PRIx64 ")", file_name, start_addr, end_addr);235return true;236}237238bool AddressRange::operator==(const AddressRange &rhs) {239if (!IsValid() || !rhs.IsValid())240return false;241return m_base_addr == rhs.GetBaseAddress() &&242m_byte_size == rhs.GetByteSize();243}244245bool AddressRange::operator!=(const AddressRange &rhs) {246return !(*this == rhs);247}248249250