//===----------------------------------------------------------------------===// // // 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/Utility/VirtualDataExtractor.h" #include using namespace lldb; using namespace lldb_private; VirtualDataExtractor::VirtualDataExtractor(const void *data, offset_t data_length, ByteOrder byte_order, uint32_t addr_size, LookupTable lookup_table) : DataExtractor(data, data_length, byte_order, addr_size), m_lookup_table(std::move(lookup_table)) { m_lookup_table.Sort(); } VirtualDataExtractor::VirtualDataExtractor(const DataBufferSP &data_sp, ByteOrder byte_order, uint32_t addr_size, LookupTable lookup_table) : DataExtractor(data_sp, byte_order, addr_size), m_lookup_table(std::move(lookup_table)) { m_lookup_table.Sort(); } const VirtualDataExtractor::LookupTable::Entry * VirtualDataExtractor::FindEntry(offset_t virtual_addr) const { // Use RangeDataVector's binary search instead of linear search. return m_lookup_table.FindEntryThatContains(virtual_addr); } bool VirtualDataExtractor::ValidateVirtualRead(offset_t virtual_addr, offset_t length) const { const LookupTable::Entry *entry = FindEntry(virtual_addr); if (!entry) return false; // Assert that the read does not cross entry boundaries. // RangeData.Contains() checks if a range is fully contained. assert(entry->Contains(LookupTable::Range(virtual_addr, length)) && "Read crosses lookup table entry boundary"); // Also validate that the physical offset is within the data buffer. // RangeData.data contains the physical offset. offset_t physical_offset = entry->data + (virtual_addr - entry->base); return ValidOffsetForDataOfSize(physical_offset, length); } const void *VirtualDataExtractor::GetData(offset_t *offset_ptr, offset_t length) const { // Override to treat offset as virtual address. if (!offset_ptr) return nullptr; offset_t virtual_addr = *offset_ptr; if (!ValidateVirtualRead(virtual_addr, length)) return nullptr; const LookupTable::Entry *entry = FindEntry(virtual_addr); assert(entry && "ValidateVirtualRead should have found an entry"); offset_t physical_offset = entry->data + (virtual_addr - entry->base); // Use base class PeekData directly to avoid recursion. const void *result = DataExtractor::PeekData(physical_offset, length); if (result) { // Advance the virtual offset pointer. *offset_ptr += length; } return result; } const uint8_t *VirtualDataExtractor::PeekData(offset_t offset, offset_t length) const { // Override to treat offset as virtual address. if (!ValidateVirtualRead(offset, length)) return nullptr; const LookupTable::Entry *entry = FindEntry(offset); assert(entry && "ValidateVirtualRead should have found an entry"); offset_t physical_offset = entry->data + (offset - entry->base); // Use the base class PeekData with the physical offset. return DataExtractor::PeekData(physical_offset, length); } uint8_t VirtualDataExtractor::GetU8_unchecked(offset_t *offset_ptr) const { offset_t virtual_addr = *offset_ptr; const LookupTable::Entry *entry = FindEntry(virtual_addr); assert(entry && "Unchecked methods require valid virtual address"); offset_t physical_offset = entry->data + (virtual_addr - entry->base); uint8_t result = DataExtractor::GetU8_unchecked(&physical_offset); *offset_ptr += 1; return result; } uint16_t VirtualDataExtractor::GetU16_unchecked(offset_t *offset_ptr) const { offset_t virtual_addr = *offset_ptr; const LookupTable::Entry *entry = FindEntry(virtual_addr); assert(entry && "Unchecked methods require valid virtual address"); offset_t physical_offset = entry->data + (virtual_addr - entry->base); uint16_t result = DataExtractor::GetU16_unchecked(&physical_offset); *offset_ptr += 2; return result; } uint32_t VirtualDataExtractor::GetU32_unchecked(offset_t *offset_ptr) const { offset_t virtual_addr = *offset_ptr; const LookupTable::Entry *entry = FindEntry(virtual_addr); assert(entry && "Unchecked methods require valid virtual address"); offset_t physical_offset = entry->data + (virtual_addr - entry->base); uint32_t result = DataExtractor::GetU32_unchecked(&physical_offset); *offset_ptr += 4; return result; } uint64_t VirtualDataExtractor::GetU64_unchecked(offset_t *offset_ptr) const { offset_t virtual_addr = *offset_ptr; const LookupTable::Entry *entry = FindEntry(virtual_addr); assert(entry && "Unchecked methods require valid virtual address"); offset_t physical_offset = entry->data + (virtual_addr - entry->base); uint64_t result = DataExtractor::GetU64_unchecked(&physical_offset); *offset_ptr += 8; return result; }