mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-23 04:32:30 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			100 lines
		
	
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			100 lines
		
	
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020-2021, Itamar S. <itamar8910@gmail.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include "AttributeValue.h"
 | |
| #include "DwarfTypes.h"
 | |
| #include <AK/ByteBuffer.h>
 | |
| #include <AK/DeprecatedString.h>
 | |
| #include <AK/NonnullRefPtr.h>
 | |
| #include <AK/RedBlackTree.h>
 | |
| #include <AK/RefCounted.h>
 | |
| #include <LibDebug/Dwarf/DIE.h>
 | |
| #include <LibELF/Image.h>
 | |
| 
 | |
| namespace Debug::Dwarf {
 | |
| 
 | |
| class CompilationUnit;
 | |
| 
 | |
| class DwarfInfo {
 | |
|     AK_MAKE_NONCOPYABLE(DwarfInfo);
 | |
|     AK_MAKE_NONMOVABLE(DwarfInfo);
 | |
| 
 | |
| public:
 | |
|     explicit DwarfInfo(ELF::Image const&);
 | |
|     ~DwarfInfo();
 | |
| 
 | |
|     ReadonlyBytes debug_info_data() const { return m_debug_info_data; }
 | |
|     ReadonlyBytes abbreviation_data() const { return m_abbreviation_data; }
 | |
|     ReadonlyBytes debug_strings_data() const { return m_debug_strings_data; }
 | |
|     ReadonlyBytes debug_line_strings_data() const { return m_debug_line_strings_data; }
 | |
|     ReadonlyBytes debug_range_lists_data() const { return m_debug_range_lists_data; }
 | |
|     ReadonlyBytes debug_str_offsets_data() const { return m_debug_str_offsets_data; }
 | |
|     ReadonlyBytes debug_addr_data() const { return m_debug_addr_data; }
 | |
|     ReadonlyBytes debug_ranges_data() const { return m_debug_ranges_data; }
 | |
| 
 | |
|     template<typename Callback>
 | |
|     ErrorOr<void> for_each_compilation_unit(Callback) const;
 | |
| 
 | |
|     ErrorOr<AttributeValue> get_attribute_value(AttributeDataForm form, ssize_t implicit_const_value,
 | |
|         SeekableStream& debug_info_stream, CompilationUnit const* unit = nullptr) const;
 | |
| 
 | |
|     ErrorOr<Optional<DIE>> get_die_at_address(FlatPtr) const;
 | |
| 
 | |
|     // Note that even if there is a DIE at the given offset,
 | |
|     // but it does not exist in the DIE cache (because for example
 | |
|     // it does not contain an address range), then this function will not return it.
 | |
|     // To get any DIE object at a given offset in a compilation unit,
 | |
|     // use CompilationUnit::get_die_at_offset.
 | |
|     ErrorOr<Optional<DIE>> get_cached_die_at_offset(FlatPtr) const;
 | |
| 
 | |
| private:
 | |
|     ErrorOr<void> populate_compilation_units();
 | |
|     ErrorOr<void> build_cached_dies() const;
 | |
| 
 | |
|     ReadonlyBytes section_data(StringView section_name) const;
 | |
| 
 | |
|     ELF::Image const& m_elf;
 | |
|     ReadonlyBytes m_debug_info_data;
 | |
|     ReadonlyBytes m_abbreviation_data;
 | |
|     ReadonlyBytes m_debug_strings_data;
 | |
|     ReadonlyBytes m_debug_line_data;
 | |
|     ReadonlyBytes m_debug_line_strings_data;
 | |
|     ReadonlyBytes m_debug_range_lists_data;
 | |
|     ReadonlyBytes m_debug_str_offsets_data;
 | |
|     ReadonlyBytes m_debug_addr_data;
 | |
|     ReadonlyBytes m_debug_ranges_data;
 | |
| 
 | |
|     Vector<NonnullOwnPtr<Dwarf::CompilationUnit>> m_compilation_units;
 | |
| 
 | |
|     struct DIERange {
 | |
|         FlatPtr start_address { 0 };
 | |
|         FlatPtr end_address { 0 };
 | |
|     };
 | |
| 
 | |
|     struct DIEAndRange {
 | |
|         DIE die;
 | |
|         DIERange range;
 | |
|     };
 | |
| 
 | |
|     using DIEStartAddress = FlatPtr;
 | |
| 
 | |
|     mutable RedBlackTree<DIEStartAddress, DIEAndRange> m_cached_dies_by_range;
 | |
|     mutable RedBlackTree<FlatPtr, DIE> m_cached_dies_by_offset;
 | |
|     mutable bool m_built_cached_dies { false };
 | |
| };
 | |
| 
 | |
| template<typename Callback>
 | |
| ErrorOr<void> DwarfInfo::for_each_compilation_unit(Callback callback) const
 | |
| {
 | |
|     for (auto const& unit : m_compilation_units) {
 | |
|         TRY(callback(*unit));
 | |
|     }
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| }
 | 
