xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/SymbolFile/DWARF/NameToDIE.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1dda28197Spatrick //===-- NameToDIE.cpp -----------------------------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick 
9061da546Spatrick #include "NameToDIE.h"
10061da546Spatrick #include "DWARFUnit.h"
11*f6aab3d8Srobert #include "lldb/Core/DataFileCache.h"
12061da546Spatrick #include "lldb/Symbol/ObjectFile.h"
13061da546Spatrick #include "lldb/Utility/ConstString.h"
14*f6aab3d8Srobert #include "lldb/Utility/DataEncoder.h"
15*f6aab3d8Srobert #include "lldb/Utility/DataExtractor.h"
16061da546Spatrick #include "lldb/Utility/RegularExpression.h"
17061da546Spatrick #include "lldb/Utility/Stream.h"
18061da546Spatrick #include "lldb/Utility/StreamString.h"
19*f6aab3d8Srobert #include <optional>
20061da546Spatrick 
21061da546Spatrick using namespace lldb;
22061da546Spatrick using namespace lldb_private;
23061da546Spatrick 
Finalize()24061da546Spatrick void NameToDIE::Finalize() {
25*f6aab3d8Srobert   m_map.Sort(std::less<DIERef>());
26061da546Spatrick   m_map.SizeToFit();
27061da546Spatrick }
28061da546Spatrick 
Insert(ConstString name,const DIERef & die_ref)29061da546Spatrick void NameToDIE::Insert(ConstString name, const DIERef &die_ref) {
30061da546Spatrick   m_map.Append(name, die_ref);
31061da546Spatrick }
32061da546Spatrick 
Find(ConstString name,llvm::function_ref<bool (DIERef ref)> callback) const33dda28197Spatrick bool NameToDIE::Find(ConstString name,
34dda28197Spatrick                      llvm::function_ref<bool(DIERef ref)> callback) const {
35dda28197Spatrick   for (const auto &entry : m_map.equal_range(name))
36dda28197Spatrick     if (!callback(entry.value))
37dda28197Spatrick       return false;
38dda28197Spatrick   return true;
39061da546Spatrick }
40061da546Spatrick 
Find(const RegularExpression & regex,llvm::function_ref<bool (DIERef ref)> callback) const41dda28197Spatrick bool NameToDIE::Find(const RegularExpression &regex,
42dda28197Spatrick                      llvm::function_ref<bool(DIERef ref)> callback) const {
43dda28197Spatrick   for (const auto &entry : m_map)
44dda28197Spatrick     if (regex.Execute(entry.cstring.GetCString())) {
45dda28197Spatrick       if (!callback(entry.value))
46dda28197Spatrick         return false;
47dda28197Spatrick     }
48dda28197Spatrick   return true;
49061da546Spatrick }
50061da546Spatrick 
FindAllEntriesForUnit(DWARFUnit & s_unit,llvm::function_ref<bool (DIERef ref)> callback) const51dda28197Spatrick void NameToDIE::FindAllEntriesForUnit(
52*f6aab3d8Srobert     DWARFUnit &s_unit, llvm::function_ref<bool(DIERef ref)> callback) const {
53*f6aab3d8Srobert   lldbassert(!s_unit.GetSymbolFileDWARF().GetDwoNum());
54*f6aab3d8Srobert   const DWARFUnit &ns_unit = s_unit.GetNonSkeletonUnit();
55061da546Spatrick   const uint32_t size = m_map.GetSize();
56061da546Spatrick   for (uint32_t i = 0; i < size; ++i) {
57061da546Spatrick     const DIERef &die_ref = m_map.GetValueAtIndexUnchecked(i);
58*f6aab3d8Srobert     if (ns_unit.GetSymbolFileDWARF().GetDwoNum() == die_ref.dwo_num() &&
59*f6aab3d8Srobert         ns_unit.GetDebugSection() == die_ref.section() &&
60*f6aab3d8Srobert         ns_unit.GetOffset() <= die_ref.die_offset() &&
61*f6aab3d8Srobert         die_ref.die_offset() < ns_unit.GetNextUnitOffset()) {
62dda28197Spatrick       if (!callback(die_ref))
63dda28197Spatrick         return;
64061da546Spatrick     }
65dda28197Spatrick   }
66061da546Spatrick }
67061da546Spatrick 
Dump(Stream * s)68061da546Spatrick void NameToDIE::Dump(Stream *s) {
69061da546Spatrick   const uint32_t size = m_map.GetSize();
70061da546Spatrick   for (uint32_t i = 0; i < size; ++i) {
71061da546Spatrick     s->Format("{0} \"{1}\"\n", m_map.GetValueAtIndexUnchecked(i),
72061da546Spatrick               m_map.GetCStringAtIndexUnchecked(i));
73061da546Spatrick   }
74061da546Spatrick }
75061da546Spatrick 
ForEach(std::function<bool (ConstString name,const DIERef & die_ref)> const & callback) const76061da546Spatrick void NameToDIE::ForEach(
77061da546Spatrick     std::function<bool(ConstString name, const DIERef &die_ref)> const
78061da546Spatrick         &callback) const {
79061da546Spatrick   const uint32_t size = m_map.GetSize();
80061da546Spatrick   for (uint32_t i = 0; i < size; ++i) {
81061da546Spatrick     if (!callback(m_map.GetCStringAtIndexUnchecked(i),
82061da546Spatrick                   m_map.GetValueAtIndexUnchecked(i)))
83061da546Spatrick       break;
84061da546Spatrick   }
85061da546Spatrick }
86061da546Spatrick 
Append(const NameToDIE & other)87061da546Spatrick void NameToDIE::Append(const NameToDIE &other) {
88061da546Spatrick   const uint32_t size = other.m_map.GetSize();
89061da546Spatrick   for (uint32_t i = 0; i < size; ++i) {
90061da546Spatrick     m_map.Append(other.m_map.GetCStringAtIndexUnchecked(i),
91061da546Spatrick                  other.m_map.GetValueAtIndexUnchecked(i));
92061da546Spatrick   }
93061da546Spatrick }
94*f6aab3d8Srobert 
95*f6aab3d8Srobert constexpr llvm::StringLiteral kIdentifierNameToDIE("N2DI");
96*f6aab3d8Srobert 
Decode(const DataExtractor & data,lldb::offset_t * offset_ptr,const StringTableReader & strtab)97*f6aab3d8Srobert bool NameToDIE::Decode(const DataExtractor &data, lldb::offset_t *offset_ptr,
98*f6aab3d8Srobert                        const StringTableReader &strtab) {
99*f6aab3d8Srobert   m_map.Clear();
100*f6aab3d8Srobert   llvm::StringRef identifier((const char *)data.GetData(offset_ptr, 4), 4);
101*f6aab3d8Srobert   if (identifier != kIdentifierNameToDIE)
102*f6aab3d8Srobert     return false;
103*f6aab3d8Srobert   const uint32_t count = data.GetU32(offset_ptr);
104*f6aab3d8Srobert   m_map.Reserve(count);
105*f6aab3d8Srobert   for (uint32_t i = 0; i < count; ++i) {
106*f6aab3d8Srobert     llvm::StringRef str(strtab.Get(data.GetU32(offset_ptr)));
107*f6aab3d8Srobert     // No empty strings allowed in the name to DIE maps.
108*f6aab3d8Srobert     if (str.empty())
109*f6aab3d8Srobert       return false;
110*f6aab3d8Srobert     if (std::optional<DIERef> die_ref = DIERef::Decode(data, offset_ptr))
111*f6aab3d8Srobert       m_map.Append(ConstString(str), *die_ref);
112*f6aab3d8Srobert     else
113*f6aab3d8Srobert       return false;
114*f6aab3d8Srobert   }
115*f6aab3d8Srobert   // We must sort the UniqueCStringMap after decoding it since it is a vector
116*f6aab3d8Srobert   // of UniqueCStringMap::Entry objects which contain a ConstString and type T.
117*f6aab3d8Srobert   // ConstString objects are sorted by "const char *" and then type T and
118*f6aab3d8Srobert   // the "const char *" are point values that will depend on the order in which
119*f6aab3d8Srobert   // ConstString objects are created and in which of the 256 string pools they
120*f6aab3d8Srobert   // are created in. So after we decode all of the entries, we must sort the
121*f6aab3d8Srobert   // name map to ensure name lookups succeed. If we encode and decode within
122*f6aab3d8Srobert   // the same process we wouldn't need to sort, so unit testing didn't catch
123*f6aab3d8Srobert   // this issue when first checked in.
124*f6aab3d8Srobert   m_map.Sort(std::less<DIERef>());
125*f6aab3d8Srobert   return true;
126*f6aab3d8Srobert }
127*f6aab3d8Srobert 
Encode(DataEncoder & encoder,ConstStringTable & strtab) const128*f6aab3d8Srobert void NameToDIE::Encode(DataEncoder &encoder, ConstStringTable &strtab) const {
129*f6aab3d8Srobert   encoder.AppendData(kIdentifierNameToDIE);
130*f6aab3d8Srobert   encoder.AppendU32(m_map.GetSize());
131*f6aab3d8Srobert   for (const auto &entry : m_map) {
132*f6aab3d8Srobert     // Make sure there are no empty strings.
133*f6aab3d8Srobert     assert((bool)entry.cstring);
134*f6aab3d8Srobert     encoder.AppendU32(strtab.Add(entry.cstring));
135*f6aab3d8Srobert     entry.value.Encode(encoder);
136*f6aab3d8Srobert   }
137*f6aab3d8Srobert }
138*f6aab3d8Srobert 
operator ==(const NameToDIE & rhs) const139*f6aab3d8Srobert bool NameToDIE::operator==(const NameToDIE &rhs) const {
140*f6aab3d8Srobert   const size_t size = m_map.GetSize();
141*f6aab3d8Srobert   if (size != rhs.m_map.GetSize())
142*f6aab3d8Srobert     return false;
143*f6aab3d8Srobert   for (size_t i = 0; i < size; ++i) {
144*f6aab3d8Srobert     if (m_map.GetCStringAtIndex(i) != rhs.m_map.GetCStringAtIndex(i))
145*f6aab3d8Srobert       return false;
146*f6aab3d8Srobert     if (m_map.GetValueRefAtIndexUnchecked(i) !=
147*f6aab3d8Srobert         rhs.m_map.GetValueRefAtIndexUnchecked(i))
148*f6aab3d8Srobert       return false;
149*f6aab3d8Srobert   }
150*f6aab3d8Srobert   return true;
151*f6aab3d8Srobert }
152