CompilationUnit.cpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. /*
  2. * Copyright (c) 2020, Itamar S. <itamar8910@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "CompilationUnit.h"
  7. #include <AK/ByteReader.h>
  8. #include <LibDebug/Dwarf/DIE.h>
  9. #include <LibDebug/Dwarf/DwarfInfo.h>
  10. #include <LibDebug/Dwarf/LineProgram.h>
  11. namespace Debug::Dwarf {
  12. CompilationUnit::CompilationUnit(DwarfInfo const& dwarf_info, u32 offset, CompilationUnitHeader const& header, NonnullOwnPtr<LineProgram>&& line_program)
  13. : m_dwarf_info(dwarf_info)
  14. , m_offset(offset)
  15. , m_header(header)
  16. , m_abbreviations(dwarf_info, header.abbrev_offset())
  17. , m_line_program(move(line_program))
  18. {
  19. VERIFY(header.version() < 5 || header.unit_type() == CompilationUnitType::Full);
  20. }
  21. CompilationUnit::~CompilationUnit() = default;
  22. DIE CompilationUnit::root_die() const
  23. {
  24. return DIE(*this, m_offset + m_header.header_size());
  25. }
  26. DIE CompilationUnit::get_die_at_offset(u32 die_offset) const
  27. {
  28. VERIFY(die_offset >= offset() && die_offset < offset() + size());
  29. return DIE(*this, die_offset);
  30. }
  31. LineProgram const& CompilationUnit::line_program() const
  32. {
  33. return *m_line_program;
  34. }
  35. ErrorOr<Optional<FlatPtr>> CompilationUnit::base_address() const
  36. {
  37. if (m_has_cached_base_address)
  38. return m_cached_base_address;
  39. auto die = root_die();
  40. auto res = TRY(die.get_attribute(Attribute::LowPc));
  41. if (res.has_value()) {
  42. m_cached_base_address = TRY(res->as_addr());
  43. }
  44. m_has_cached_base_address = true;
  45. return m_cached_base_address;
  46. }
  47. ErrorOr<u64> CompilationUnit::address_table_base() const
  48. {
  49. if (m_has_cached_address_table_base)
  50. return m_cached_address_table_base;
  51. auto die = root_die();
  52. auto res = TRY(die.get_attribute(Attribute::AddrBase));
  53. if (res.has_value()) {
  54. VERIFY(res->form() == AttributeDataForm::SecOffset);
  55. m_cached_address_table_base = res->as_unsigned();
  56. }
  57. m_has_cached_address_table_base = true;
  58. return m_cached_address_table_base;
  59. }
  60. ErrorOr<u64> CompilationUnit::string_offsets_base() const
  61. {
  62. if (m_has_cached_string_offsets_base)
  63. return m_cached_string_offsets_base;
  64. auto die = root_die();
  65. auto res = TRY(die.get_attribute(Attribute::StrOffsetsBase));
  66. if (res.has_value()) {
  67. VERIFY(res->form() == AttributeDataForm::SecOffset);
  68. m_cached_string_offsets_base = res->as_unsigned();
  69. }
  70. m_has_cached_string_offsets_base = true;
  71. return m_cached_string_offsets_base;
  72. }
  73. ErrorOr<u64> CompilationUnit::range_lists_base() const
  74. {
  75. if (m_has_cached_range_lists_base)
  76. return m_cached_range_lists_base;
  77. auto die = root_die();
  78. auto res = TRY(die.get_attribute(Attribute::RngListsBase));
  79. if (res.has_value()) {
  80. VERIFY(res->form() == AttributeDataForm::SecOffset);
  81. m_cached_range_lists_base = res->as_unsigned();
  82. }
  83. m_has_cached_range_lists_base = true;
  84. return m_cached_range_lists_base;
  85. }
  86. ErrorOr<FlatPtr> CompilationUnit::get_address(size_t index) const
  87. {
  88. auto base = TRY(address_table_base());
  89. auto debug_addr_data = dwarf_info().debug_addr_data();
  90. VERIFY(base < debug_addr_data.size());
  91. auto addresses = debug_addr_data.slice(base);
  92. VERIFY(index * sizeof(FlatPtr) < addresses.size());
  93. FlatPtr value { 0 };
  94. ByteReader::load<FlatPtr>(addresses.offset_pointer(index * sizeof(FlatPtr)), value);
  95. return value;
  96. }
  97. ErrorOr<char const*> CompilationUnit::get_string(size_t index) const
  98. {
  99. auto base = TRY(string_offsets_base());
  100. auto debug_str_offsets_data = dwarf_info().debug_str_offsets_data();
  101. VERIFY(base < debug_str_offsets_data.size());
  102. // FIXME: This assumes DWARF32
  103. auto offsets = debug_str_offsets_data.slice(base);
  104. VERIFY(index * sizeof(u32) < offsets.size());
  105. auto offset = ByteReader::load32(offsets.offset_pointer(index * sizeof(u32)));
  106. return bit_cast<char const*>(dwarf_info().debug_strings_data().offset(offset));
  107. }
  108. }