CompilationUnit.cpp 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  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 "DIE.h"
  8. namespace Debug::Dwarf {
  9. CompilationUnit::CompilationUnit(DwarfInfo const& dwarf_info, u32 offset, CompilationUnitHeader const& header, NonnullOwnPtr<LineProgram>&& line_program)
  10. : m_dwarf_info(dwarf_info)
  11. , m_offset(offset)
  12. , m_header(header)
  13. , m_abbreviations(dwarf_info, header.abbrev_offset())
  14. , m_line_program(move(line_program))
  15. {
  16. VERIFY(header.version() < 5 || header.unit_type() == CompilationUnitType::Full);
  17. }
  18. DIE CompilationUnit::root_die() const
  19. {
  20. return DIE(*this, m_offset + m_header.header_size());
  21. }
  22. DIE CompilationUnit::get_die_at_offset(u32 die_offset) const
  23. {
  24. VERIFY(die_offset >= offset() && die_offset < offset() + size());
  25. return DIE(*this, die_offset);
  26. }
  27. Optional<FlatPtr> CompilationUnit::base_address() const
  28. {
  29. if (m_has_cached_base_address)
  30. return m_cached_base_address;
  31. auto die = root_die();
  32. auto res = die.get_attribute(Attribute::LowPc);
  33. if (res.has_value()) {
  34. m_cached_base_address = res->as_addr();
  35. }
  36. m_has_cached_base_address = true;
  37. return m_cached_base_address;
  38. }
  39. u64 CompilationUnit::address_table_base() const
  40. {
  41. if (m_has_cached_address_table_base)
  42. return m_cached_address_table_base;
  43. auto die = root_die();
  44. auto res = die.get_attribute(Attribute::AddrBase);
  45. if (res.has_value()) {
  46. VERIFY(res->form() == AttributeDataForm::SecOffset);
  47. m_cached_address_table_base = res->as_unsigned();
  48. }
  49. m_has_cached_address_table_base = true;
  50. return m_cached_address_table_base;
  51. }
  52. u64 CompilationUnit::string_offsets_base() const
  53. {
  54. if (m_has_cached_string_offsets_base)
  55. return m_cached_string_offsets_base;
  56. auto die = root_die();
  57. auto res = die.get_attribute(Attribute::StrOffsetsBase);
  58. if (res.has_value()) {
  59. VERIFY(res->form() == AttributeDataForm::SecOffset);
  60. m_cached_string_offsets_base = res->as_unsigned();
  61. }
  62. m_has_cached_string_offsets_base = true;
  63. return m_cached_string_offsets_base;
  64. }
  65. u64 CompilationUnit::range_lists_base() const
  66. {
  67. if (m_has_cached_range_lists_base)
  68. return m_cached_range_lists_base;
  69. auto die = root_die();
  70. auto res = die.get_attribute(Attribute::RngListsBase);
  71. if (res.has_value()) {
  72. VERIFY(res->form() == AttributeDataForm::SecOffset);
  73. m_cached_range_lists_base = res->as_unsigned();
  74. }
  75. m_has_cached_range_lists_base = true;
  76. return m_cached_range_lists_base;
  77. }
  78. FlatPtr CompilationUnit::get_address(size_t index) const
  79. {
  80. auto base = address_table_base();
  81. auto debug_addr_data = dwarf_info().debug_addr_data();
  82. VERIFY(base < debug_addr_data.size());
  83. auto addresses = reinterpret_cast<FlatPtr const*>(debug_addr_data.offset(base));
  84. VERIFY(base + index * sizeof(FlatPtr) < debug_addr_data.size());
  85. return addresses[index];
  86. }
  87. char const* CompilationUnit::get_string(size_t index) const
  88. {
  89. auto base = string_offsets_base();
  90. auto debug_str_offsets_data = dwarf_info().debug_str_offsets_data();
  91. VERIFY(base < debug_str_offsets_data.size());
  92. // FIXME: This assumes DWARF32
  93. auto offsets = reinterpret_cast<u32 const*>(debug_str_offsets_data.offset(base));
  94. VERIFY(base + index * sizeof(u32) < debug_str_offsets_data.size());
  95. auto offset = offsets[index];
  96. return reinterpret_cast<char const*>(dwarf_info().debug_strings_data().offset(offset));
  97. }
  98. }