1dda28197Spatrick //===-- AddressRange.cpp --------------------------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick
9061da546Spatrick #include "lldb/Core/AddressRange.h"
10061da546Spatrick #include "lldb/Core/Module.h"
11be691f3bSpatrick #include "lldb/Core/Section.h"
12061da546Spatrick #include "lldb/Target/Target.h"
13061da546Spatrick #include "lldb/Utility/ConstString.h"
14061da546Spatrick #include "lldb/Utility/FileSpec.h"
15061da546Spatrick #include "lldb/Utility/Stream.h"
16061da546Spatrick #include "lldb/lldb-defines.h"
17061da546Spatrick
18061da546Spatrick #include "llvm/Support/Compiler.h"
19061da546Spatrick
20061da546Spatrick #include <memory>
21061da546Spatrick
22be691f3bSpatrick #include <cinttypes>
23061da546Spatrick
24061da546Spatrick namespace lldb_private {
25061da546Spatrick class SectionList;
26061da546Spatrick }
27061da546Spatrick
28061da546Spatrick using namespace lldb;
29061da546Spatrick using namespace lldb_private;
30061da546Spatrick
AddressRange()31be691f3bSpatrick AddressRange::AddressRange() : m_base_addr() {}
32061da546Spatrick
AddressRange(addr_t file_addr,addr_t byte_size,const SectionList * section_list)33061da546Spatrick AddressRange::AddressRange(addr_t file_addr, addr_t byte_size,
34061da546Spatrick const SectionList *section_list)
35061da546Spatrick : m_base_addr(file_addr, section_list), m_byte_size(byte_size) {}
36061da546Spatrick
AddressRange(const lldb::SectionSP & section,addr_t offset,addr_t byte_size)37061da546Spatrick AddressRange::AddressRange(const lldb::SectionSP §ion, addr_t offset,
38061da546Spatrick addr_t byte_size)
39061da546Spatrick : m_base_addr(section, offset), m_byte_size(byte_size) {}
40061da546Spatrick
AddressRange(const Address & so_addr,addr_t byte_size)41061da546Spatrick AddressRange::AddressRange(const Address &so_addr, addr_t byte_size)
42061da546Spatrick : m_base_addr(so_addr), m_byte_size(byte_size) {}
43061da546Spatrick
44be691f3bSpatrick AddressRange::~AddressRange() = default;
45061da546Spatrick
Contains(const Address & addr) const46be691f3bSpatrick bool AddressRange::Contains(const Address &addr) const {
47be691f3bSpatrick SectionSP range_sect_sp = GetBaseAddress().GetSection();
48be691f3bSpatrick SectionSP addr_sect_sp = addr.GetSection();
49be691f3bSpatrick if (range_sect_sp) {
50be691f3bSpatrick if (!addr_sect_sp ||
51be691f3bSpatrick range_sect_sp->GetModule() != addr_sect_sp->GetModule())
52be691f3bSpatrick return false; // Modules do not match.
53be691f3bSpatrick } else if (addr_sect_sp) {
54be691f3bSpatrick return false; // Range has no module but "addr" does because addr has a
55be691f3bSpatrick // section
56be691f3bSpatrick }
57be691f3bSpatrick // Either the modules match, or both have no module, so it is ok to compare
58be691f3bSpatrick // the file addresses in this case only.
59be691f3bSpatrick return ContainsFileAddress(addr);
60be691f3bSpatrick }
61be691f3bSpatrick
ContainsFileAddress(const Address & addr) const62061da546Spatrick bool AddressRange::ContainsFileAddress(const Address &addr) const {
63061da546Spatrick if (addr.GetSection() == m_base_addr.GetSection())
64061da546Spatrick return (addr.GetOffset() - m_base_addr.GetOffset()) < GetByteSize();
65061da546Spatrick addr_t file_base_addr = GetBaseAddress().GetFileAddress();
66061da546Spatrick if (file_base_addr == LLDB_INVALID_ADDRESS)
67061da546Spatrick return false;
68061da546Spatrick
69061da546Spatrick addr_t file_addr = addr.GetFileAddress();
70061da546Spatrick if (file_addr == LLDB_INVALID_ADDRESS)
71061da546Spatrick return false;
72061da546Spatrick
73061da546Spatrick if (file_base_addr <= file_addr)
74061da546Spatrick return (file_addr - file_base_addr) < GetByteSize();
75061da546Spatrick
76061da546Spatrick return false;
77061da546Spatrick }
78061da546Spatrick
ContainsFileAddress(addr_t file_addr) const79061da546Spatrick bool AddressRange::ContainsFileAddress(addr_t file_addr) const {
80061da546Spatrick if (file_addr == LLDB_INVALID_ADDRESS)
81061da546Spatrick return false;
82061da546Spatrick
83061da546Spatrick addr_t file_base_addr = GetBaseAddress().GetFileAddress();
84061da546Spatrick if (file_base_addr == LLDB_INVALID_ADDRESS)
85061da546Spatrick return false;
86061da546Spatrick
87061da546Spatrick if (file_base_addr <= file_addr)
88061da546Spatrick return (file_addr - file_base_addr) < GetByteSize();
89061da546Spatrick
90061da546Spatrick return false;
91061da546Spatrick }
92061da546Spatrick
ContainsLoadAddress(const Address & addr,Target * target) const93061da546Spatrick bool AddressRange::ContainsLoadAddress(const Address &addr,
94061da546Spatrick Target *target) const {
95061da546Spatrick if (addr.GetSection() == m_base_addr.GetSection())
96061da546Spatrick return (addr.GetOffset() - m_base_addr.GetOffset()) < GetByteSize();
97061da546Spatrick addr_t load_base_addr = GetBaseAddress().GetLoadAddress(target);
98061da546Spatrick if (load_base_addr == LLDB_INVALID_ADDRESS)
99061da546Spatrick return false;
100061da546Spatrick
101061da546Spatrick addr_t load_addr = addr.GetLoadAddress(target);
102061da546Spatrick if (load_addr == LLDB_INVALID_ADDRESS)
103061da546Spatrick return false;
104061da546Spatrick
105061da546Spatrick if (load_base_addr <= load_addr)
106061da546Spatrick return (load_addr - load_base_addr) < GetByteSize();
107061da546Spatrick
108061da546Spatrick return false;
109061da546Spatrick }
110061da546Spatrick
ContainsLoadAddress(addr_t load_addr,Target * target) const111061da546Spatrick bool AddressRange::ContainsLoadAddress(addr_t load_addr, Target *target) const {
112061da546Spatrick if (load_addr == LLDB_INVALID_ADDRESS)
113061da546Spatrick return false;
114061da546Spatrick
115061da546Spatrick addr_t load_base_addr = GetBaseAddress().GetLoadAddress(target);
116061da546Spatrick if (load_base_addr == LLDB_INVALID_ADDRESS)
117061da546Spatrick return false;
118061da546Spatrick
119061da546Spatrick if (load_base_addr <= load_addr)
120061da546Spatrick return (load_addr - load_base_addr) < GetByteSize();
121061da546Spatrick
122061da546Spatrick return false;
123061da546Spatrick }
124061da546Spatrick
Extend(const AddressRange & rhs_range)125061da546Spatrick bool AddressRange::Extend(const AddressRange &rhs_range) {
126061da546Spatrick addr_t lhs_end_addr = GetBaseAddress().GetFileAddress() + GetByteSize();
127061da546Spatrick addr_t rhs_base_addr = rhs_range.GetBaseAddress().GetFileAddress();
128061da546Spatrick
129061da546Spatrick if (!ContainsFileAddress(rhs_range.GetBaseAddress()) &&
130061da546Spatrick lhs_end_addr != rhs_base_addr)
131061da546Spatrick // The ranges don't intersect at all on the right side of this range.
132061da546Spatrick return false;
133061da546Spatrick
134061da546Spatrick addr_t rhs_end_addr = rhs_base_addr + rhs_range.GetByteSize();
135061da546Spatrick if (lhs_end_addr >= rhs_end_addr)
136061da546Spatrick // The rhs range totally overlaps this one, nothing to add.
137061da546Spatrick return false;
138061da546Spatrick
139061da546Spatrick m_byte_size += rhs_end_addr - lhs_end_addr;
140061da546Spatrick return true;
141061da546Spatrick }
142061da546Spatrick
Clear()143061da546Spatrick void AddressRange::Clear() {
144061da546Spatrick m_base_addr.Clear();
145061da546Spatrick m_byte_size = 0;
146061da546Spatrick }
147061da546Spatrick
Dump(Stream * s,Target * target,Address::DumpStyle style,Address::DumpStyle fallback_style) const148061da546Spatrick bool AddressRange::Dump(Stream *s, Target *target, Address::DumpStyle style,
149061da546Spatrick Address::DumpStyle fallback_style) const {
150061da546Spatrick addr_t vmaddr = LLDB_INVALID_ADDRESS;
151061da546Spatrick int addr_size = sizeof(addr_t);
152061da546Spatrick if (target)
153061da546Spatrick addr_size = target->GetArchitecture().GetAddressByteSize();
154061da546Spatrick
155061da546Spatrick bool show_module = false;
156061da546Spatrick switch (style) {
157061da546Spatrick default:
158061da546Spatrick break;
159061da546Spatrick case Address::DumpStyleSectionNameOffset:
160061da546Spatrick case Address::DumpStyleSectionPointerOffset:
161061da546Spatrick s->PutChar('[');
162061da546Spatrick m_base_addr.Dump(s, target, style, fallback_style);
163061da546Spatrick s->PutChar('-');
164061da546Spatrick DumpAddress(s->AsRawOstream(), m_base_addr.GetOffset() + GetByteSize(),
165061da546Spatrick addr_size);
166061da546Spatrick s->PutChar(')');
167061da546Spatrick return true;
168061da546Spatrick break;
169061da546Spatrick
170061da546Spatrick case Address::DumpStyleModuleWithFileAddress:
171061da546Spatrick show_module = true;
172*f6aab3d8Srobert [[fallthrough]];
173061da546Spatrick case Address::DumpStyleFileAddress:
174061da546Spatrick vmaddr = m_base_addr.GetFileAddress();
175061da546Spatrick break;
176061da546Spatrick
177061da546Spatrick case Address::DumpStyleLoadAddress:
178061da546Spatrick vmaddr = m_base_addr.GetLoadAddress(target);
179061da546Spatrick break;
180061da546Spatrick }
181061da546Spatrick
182061da546Spatrick if (vmaddr != LLDB_INVALID_ADDRESS) {
183061da546Spatrick if (show_module) {
184061da546Spatrick ModuleSP module_sp(GetBaseAddress().GetModule());
185061da546Spatrick if (module_sp)
186061da546Spatrick s->Printf("%s", module_sp->GetFileSpec().GetFilename().AsCString(
187061da546Spatrick "<Unknown>"));
188061da546Spatrick }
189061da546Spatrick DumpAddressRange(s->AsRawOstream(), vmaddr, vmaddr + GetByteSize(),
190061da546Spatrick addr_size);
191061da546Spatrick return true;
192061da546Spatrick } else if (fallback_style != Address::DumpStyleInvalid) {
193061da546Spatrick return Dump(s, target, fallback_style, Address::DumpStyleInvalid);
194061da546Spatrick }
195061da546Spatrick
196061da546Spatrick return false;
197061da546Spatrick }
198061da546Spatrick
DumpDebug(Stream * s) const199061da546Spatrick void AddressRange::DumpDebug(Stream *s) const {
200061da546Spatrick s->Printf("%p: AddressRange section = %p, offset = 0x%16.16" PRIx64
201061da546Spatrick ", byte_size = 0x%16.16" PRIx64 "\n",
202061da546Spatrick static_cast<const void *>(this),
203061da546Spatrick static_cast<void *>(m_base_addr.GetSection().get()),
204061da546Spatrick m_base_addr.GetOffset(), GetByteSize());
205061da546Spatrick }
206