mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 04:02:34 +00:00

Previously, when trying to debug variables with more complex types (such as String), we would crash the debugger simply because it didn't know how to handle types that were irrelevant anyways. Now we just skip data we don't yet know how to handle.
155 lines
5.4 KiB
C++
155 lines
5.4 KiB
C++
/*
|
|
* Copyright (c) 2020, Itamar S. <itamar8910@gmail.com>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/NonnullOwnPtrVector.h>
|
|
#include <AK/NonnullRefPtr.h>
|
|
#include <AK/Optional.h>
|
|
#include <AK/Vector.h>
|
|
#include <LibDebug/Dwarf/DIE.h>
|
|
#include <LibDebug/Dwarf/DwarfInfo.h>
|
|
#include <LibDebug/Dwarf/LineProgram.h>
|
|
#include <LibELF/Image.h>
|
|
#include <sys/arch/i386/regs.h>
|
|
|
|
namespace Debug {
|
|
|
|
class DebugInfo {
|
|
public:
|
|
explicit DebugInfo(NonnullOwnPtr<const ELF::Image>, String source_root = {}, FlatPtr base_address = 0);
|
|
|
|
const ELF::Image& elf() const { return *m_elf; }
|
|
|
|
struct SourcePosition {
|
|
FlyString file_path;
|
|
size_t line_number { 0 };
|
|
Optional<u32> address_of_first_statement;
|
|
|
|
SourcePosition()
|
|
: SourcePosition(String::empty(), 0)
|
|
{
|
|
}
|
|
SourcePosition(String file_path, size_t line_number)
|
|
: file_path(file_path)
|
|
, line_number(line_number)
|
|
{
|
|
}
|
|
SourcePosition(String file_path, size_t line_number, u32 address_of_first_statement)
|
|
: file_path(file_path)
|
|
, line_number(line_number)
|
|
, address_of_first_statement(address_of_first_statement)
|
|
{
|
|
}
|
|
|
|
bool operator==(const SourcePosition& other) const { return file_path == other.file_path && line_number == other.line_number; }
|
|
bool operator!=(const SourcePosition& other) const { return !(*this == other); }
|
|
|
|
static SourcePosition from_line_info(const Dwarf::LineProgram::LineInfo&);
|
|
};
|
|
|
|
struct VariableInfo {
|
|
enum class LocationType {
|
|
None,
|
|
Address,
|
|
Register,
|
|
};
|
|
String name;
|
|
String type_name;
|
|
LocationType location_type { LocationType::None };
|
|
union {
|
|
u32 address;
|
|
} location_data { 0 };
|
|
|
|
union {
|
|
u32 as_u32;
|
|
u32 as_i32;
|
|
const char* as_string;
|
|
} constant_data { 0 };
|
|
|
|
Dwarf::EntryTag type_tag;
|
|
OwnPtr<VariableInfo> type;
|
|
NonnullOwnPtrVector<VariableInfo> members;
|
|
VariableInfo* parent { nullptr };
|
|
|
|
bool is_enum_type() const { return type && type->type_tag == Dwarf::EntryTag::EnumerationType; }
|
|
};
|
|
|
|
struct VariablesScope {
|
|
bool is_function { false };
|
|
String name;
|
|
u32 address_low { 0 };
|
|
u32 address_high { 0 }; // Non-inclusive - the lowest address after address_low that's not in this scope
|
|
Vector<Dwarf::DIE> dies_of_variables;
|
|
};
|
|
|
|
NonnullOwnPtrVector<VariableInfo> get_variables_in_current_scope(const PtraceRegisters&) const;
|
|
|
|
Optional<SourcePosition> get_source_position(u32 address) const;
|
|
|
|
struct SourcePositionAndAddress {
|
|
String file;
|
|
size_t line;
|
|
FlatPtr address;
|
|
};
|
|
|
|
Optional<SourcePositionAndAddress> get_address_from_source_position(const String& file, size_t line) const;
|
|
|
|
template<typename Callback>
|
|
void for_each_source_position(Callback callback) const
|
|
{
|
|
FlyString previous_file = "";
|
|
size_t previous_line = 0;
|
|
for (const auto& line_info : m_sorted_lines) {
|
|
if (line_info.file == previous_file && line_info.line == previous_line)
|
|
continue;
|
|
previous_file = line_info.file;
|
|
previous_line = line_info.line;
|
|
callback({ line_info.file, line_info.line, line_info.address });
|
|
}
|
|
}
|
|
|
|
String name_of_containing_function(u32 address) const;
|
|
Vector<SourcePosition> source_lines_in_scope(const VariablesScope&) const;
|
|
Optional<VariablesScope> get_containing_function(u32 address) const;
|
|
|
|
private:
|
|
void prepare_variable_scopes();
|
|
void prepare_lines();
|
|
void parse_scopes_impl(const Dwarf::DIE& die);
|
|
OwnPtr<VariableInfo> create_variable_info(const Dwarf::DIE& variable_die, const PtraceRegisters&, u32 address_offset = 0) const;
|
|
static bool is_variable_tag_supported(const Dwarf::EntryTag& tag);
|
|
|
|
NonnullOwnPtr<const ELF::Image> m_elf;
|
|
String m_source_root;
|
|
FlatPtr m_base_address { 0 };
|
|
Dwarf::DwarfInfo m_dwarf_info;
|
|
|
|
Vector<VariablesScope> m_scopes;
|
|
Vector<Dwarf::LineProgram::LineInfo> m_sorted_lines;
|
|
};
|
|
|
|
}
|