xref: /freebsd/contrib/llvm-project/llvm/lib/DebugInfo/DWARF/DWARFAbbreviationDeclaration.cpp (revision 51015e6d0f570239b0c2088dc6cf2b018928375d)
1 //===- DWARFAbbreviationDeclaration.cpp -----------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "llvm/DebugInfo/DWARF/DWARFAbbreviationDeclaration.h"
10 
11 #include "llvm/ADT/None.h"
12 #include "llvm/ADT/Optional.h"
13 #include "llvm/BinaryFormat/Dwarf.h"
14 #include "llvm/DebugInfo/DWARF/DWARFDataExtractor.h"
15 #include "llvm/DebugInfo/DWARF/DWARFFormValue.h"
16 #include "llvm/DebugInfo/DWARF/DWARFUnit.h"
17 #include "llvm/Support/DataExtractor.h"
18 #include "llvm/Support/FormatVariadic.h"
19 #include "llvm/Support/raw_ostream.h"
20 #include <cstddef>
21 #include <cstdint>
22 
23 using namespace llvm;
24 using namespace dwarf;
25 
26 void DWARFAbbreviationDeclaration::clear() {
27   Code = 0;
28   Tag = DW_TAG_null;
29   CodeByteSize = 0;
30   HasChildren = false;
31   AttributeSpecs.clear();
32   FixedAttributeSize.reset();
33 }
34 
35 DWARFAbbreviationDeclaration::DWARFAbbreviationDeclaration() {
36   clear();
37 }
38 
39 bool
40 DWARFAbbreviationDeclaration::extract(DataExtractor Data,
41                                       uint64_t* OffsetPtr) {
42   clear();
43   const uint64_t Offset = *OffsetPtr;
44   Code = Data.getULEB128(OffsetPtr);
45   if (Code == 0) {
46     return false;
47   }
48   CodeByteSize = *OffsetPtr - Offset;
49   Tag = static_cast<llvm::dwarf::Tag>(Data.getULEB128(OffsetPtr));
50   if (Tag == DW_TAG_null) {
51     clear();
52     return false;
53   }
54   uint8_t ChildrenByte = Data.getU8(OffsetPtr);
55   HasChildren = (ChildrenByte == DW_CHILDREN_yes);
56   // Assign a value to our optional FixedAttributeSize member variable. If
57   // this member variable still has a value after the while loop below, then
58   // all attribute data in this abbreviation declaration has a fixed byte size.
59   FixedAttributeSize = FixedSizeInfo();
60 
61   // Read all of the abbreviation attributes and forms.
62   while (true) {
63     auto A = static_cast<Attribute>(Data.getULEB128(OffsetPtr));
64     auto F = static_cast<Form>(Data.getULEB128(OffsetPtr));
65     if (A && F) {
66       bool IsImplicitConst = (F == DW_FORM_implicit_const);
67       if (IsImplicitConst) {
68         int64_t V = Data.getSLEB128(OffsetPtr);
69         AttributeSpecs.push_back(AttributeSpec(A, F, V));
70         continue;
71       }
72       Optional<uint8_t> ByteSize;
73       // If this abbrevation still has a fixed byte size, then update the
74       // FixedAttributeSize as needed.
75       switch (F) {
76       case DW_FORM_addr:
77         if (FixedAttributeSize)
78           ++FixedAttributeSize->NumAddrs;
79         break;
80 
81       case DW_FORM_ref_addr:
82         if (FixedAttributeSize)
83           ++FixedAttributeSize->NumRefAddrs;
84         break;
85 
86       case DW_FORM_strp:
87       case DW_FORM_GNU_ref_alt:
88       case DW_FORM_GNU_strp_alt:
89       case DW_FORM_line_strp:
90       case DW_FORM_sec_offset:
91       case DW_FORM_strp_sup:
92         if (FixedAttributeSize)
93           ++FixedAttributeSize->NumDwarfOffsets;
94         break;
95 
96       default:
97         // The form has a byte size that doesn't depend on Params.
98         // If it's a fixed size, keep track of it.
99         if ((ByteSize = dwarf::getFixedFormByteSize(F, dwarf::FormParams()))) {
100           if (FixedAttributeSize)
101             FixedAttributeSize->NumBytes += *ByteSize;
102           break;
103         }
104         // Indicate we no longer have a fixed byte size for this
105         // abbreviation by clearing the FixedAttributeSize optional value
106         // so it doesn't have a value.
107         FixedAttributeSize.reset();
108         break;
109       }
110       // Record this attribute and its fixed size if it has one.
111       AttributeSpecs.push_back(AttributeSpec(A, F, ByteSize));
112     } else if (A == 0 && F == 0) {
113       // We successfully reached the end of this abbreviation declaration
114       // since both attribute and form are zero.
115       break;
116     } else {
117       // Attribute and form pairs must either both be non-zero, in which case
118       // they are added to the abbreviation declaration, or both be zero to
119       // terminate the abbrevation declaration. In this case only one was
120       // zero which is an error.
121       clear();
122       return false;
123     }
124   }
125   return true;
126 }
127 
128 void DWARFAbbreviationDeclaration::dump(raw_ostream &OS) const {
129   OS << '[' << getCode() << "] ";
130   OS << formatv("{0}", getTag());
131   OS << "\tDW_CHILDREN_" << (hasChildren() ? "yes" : "no") << '\n';
132   for (const AttributeSpec &Spec : AttributeSpecs) {
133     OS << formatv("\t{0}\t{1}", Spec.Attr, Spec.Form);
134     if (Spec.isImplicitConst())
135       OS << '\t' << Spec.getImplicitConstValue();
136     OS << '\n';
137   }
138   OS << '\n';
139 }
140 
141 Optional<uint32_t>
142 DWARFAbbreviationDeclaration::findAttributeIndex(dwarf::Attribute Attr) const {
143   for (uint32_t i = 0, e = AttributeSpecs.size(); i != e; ++i) {
144     if (AttributeSpecs[i].Attr == Attr)
145       return i;
146   }
147   return None;
148 }
149 
150 uint64_t DWARFAbbreviationDeclaration::getAttributeOffsetFromIndex(
151     uint32_t AttrIndex, uint64_t DIEOffset, const DWARFUnit &U) const {
152   DWARFDataExtractor DebugInfoData = U.getDebugInfoExtractor();
153 
154   // Add the byte size of ULEB that for the abbrev Code so we can start
155   // skipping the attribute data.
156   uint64_t Offset = DIEOffset + CodeByteSize;
157   for (uint32_t CurAttrIdx = 0; CurAttrIdx != AttrIndex; ++CurAttrIdx)
158     // Match Offset along until we get to the attribute we want.
159     if (auto FixedSize = AttributeSpecs[CurAttrIdx].getByteSize(U))
160       Offset += *FixedSize;
161     else
162       DWARFFormValue::skipValue(AttributeSpecs[CurAttrIdx].Form, DebugInfoData,
163                                 &Offset, U.getFormParams());
164   return Offset;
165 }
166 
167 Optional<DWARFFormValue>
168 DWARFAbbreviationDeclaration::getAttributeValueFromOffset(
169     uint32_t AttrIndex, uint64_t Offset, const DWARFUnit &U) const {
170   assert(AttributeSpecs.size() > AttrIndex &&
171          "Attribute Index is out of bounds.");
172 
173   // We have arrived at the attribute to extract, extract if from Offset.
174   const AttributeSpec &Spec = AttributeSpecs[AttrIndex];
175   if (Spec.isImplicitConst())
176     return DWARFFormValue::createFromSValue(Spec.Form,
177                                             Spec.getImplicitConstValue());
178 
179   DWARFFormValue FormValue(Spec.Form);
180   DWARFDataExtractor DebugInfoData = U.getDebugInfoExtractor();
181   if (FormValue.extractValue(DebugInfoData, &Offset, U.getFormParams(), &U))
182     return FormValue;
183   return None;
184 }
185 
186 Optional<DWARFFormValue>
187 DWARFAbbreviationDeclaration::getAttributeValue(const uint64_t DIEOffset,
188                                                 const dwarf::Attribute Attr,
189                                                 const DWARFUnit &U) const {
190   // Check if this abbreviation has this attribute without needing to skip
191   // any data so we can return quickly if it doesn't.
192   Optional<uint32_t> MatchAttrIndex = findAttributeIndex(Attr);
193   if (!MatchAttrIndex)
194     return None;
195 
196   uint64_t Offset = getAttributeOffsetFromIndex(*MatchAttrIndex, DIEOffset, U);
197 
198   return getAttributeValueFromOffset(*MatchAttrIndex, Offset, U);
199 }
200 
201 size_t DWARFAbbreviationDeclaration::FixedSizeInfo::getByteSize(
202     const DWARFUnit &U) const {
203   size_t ByteSize = NumBytes;
204   if (NumAddrs)
205     ByteSize += NumAddrs * U.getAddressByteSize();
206   if (NumRefAddrs)
207     ByteSize += NumRefAddrs * U.getRefAddrByteSize();
208   if (NumDwarfOffsets)
209     ByteSize += NumDwarfOffsets * U.getDwarfOffsetByteSize();
210   return ByteSize;
211 }
212 
213 Optional<int64_t> DWARFAbbreviationDeclaration::AttributeSpec::getByteSize(
214     const DWARFUnit &U) const {
215   if (isImplicitConst())
216     return 0;
217   if (ByteSize.HasByteSize)
218     return ByteSize.ByteSize;
219   Optional<int64_t> S;
220   auto FixedByteSize = dwarf::getFixedFormByteSize(Form, U.getFormParams());
221   if (FixedByteSize)
222     S = *FixedByteSize;
223   return S;
224 }
225 
226 Optional<size_t> DWARFAbbreviationDeclaration::getFixedAttributesByteSize(
227     const DWARFUnit &U) const {
228   if (FixedAttributeSize)
229     return FixedAttributeSize->getByteSize(U);
230   return None;
231 }
232