blob: 638830ee568d439eb1dde8e472e20325bac74290 [file] [log] [blame]
Benjamin Krameraa2f78f2011-09-13 19:42:23 +00001//===-- DWARFAbbreviationDeclaration.cpp ----------------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Zachary Turner82af9432015-01-30 18:07:45 +000010#include "llvm/DebugInfo/DWARF/DWARFAbbreviationDeclaration.h"
Greg Clayton6f6e4db2016-11-15 01:23:06 +000011#include "llvm/DebugInfo/DWARF/DWARFFormValue.h"
12#include "llvm/DebugInfo/DWARF/DWARFUnit.h"
Benjamin Krameraa2f78f2011-09-13 19:42:23 +000013#include "llvm/Support/Dwarf.h"
Benjamin Kramer21a50922011-09-15 04:15:59 +000014#include "llvm/Support/Format.h"
Benjamin Krameraa2f78f2011-09-13 19:42:23 +000015#include "llvm/Support/raw_ostream.h"
16using namespace llvm;
17using namespace dwarf;
18
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000019void DWARFAbbreviationDeclaration::clear() {
20 Code = 0;
Greg Clayton6c273762016-10-27 16:32:04 +000021 Tag = DW_TAG_null;
Greg Clayton6f6e4db2016-11-15 01:23:06 +000022 CodeByteSize = 0;
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000023 HasChildren = false;
Alexey Samsonov1eabf982014-03-13 07:52:54 +000024 AttributeSpecs.clear();
Greg Clayton6f6e4db2016-11-15 01:23:06 +000025 FixedAttributeSize.reset();
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000026}
27
28DWARFAbbreviationDeclaration::DWARFAbbreviationDeclaration() {
29 clear();
Benjamin Krameraa2f78f2011-09-13 19:42:23 +000030}
31
32bool
Greg Clayton6c273762016-10-27 16:32:04 +000033DWARFAbbreviationDeclaration::extract(DataExtractor Data,
34 uint32_t* OffsetPtr) {
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000035 clear();
Greg Clayton6f6e4db2016-11-15 01:23:06 +000036 const uint32_t Offset = *OffsetPtr;
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000037 Code = Data.getULEB128(OffsetPtr);
38 if (Code == 0) {
39 return false;
40 }
Greg Clayton6f6e4db2016-11-15 01:23:06 +000041 CodeByteSize = *OffsetPtr - Offset;
Greg Clayton6c273762016-10-27 16:32:04 +000042 Tag = static_cast<llvm::dwarf::Tag>(Data.getULEB128(OffsetPtr));
43 if (Tag == DW_TAG_null) {
44 clear();
45 return false;
46 }
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000047 uint8_t ChildrenByte = Data.getU8(OffsetPtr);
48 HasChildren = (ChildrenByte == DW_CHILDREN_yes);
Greg Clayton6f6e4db2016-11-15 01:23:06 +000049 // Assign a value to our optional FixedAttributeSize member variable. If
50 // this member variable still has a value after the while loop below, then
51 // all attribute data in this abbreviation declaration has a fixed byte size.
52 FixedAttributeSize = FixedSizeInfo();
Benjamin Krameraa2f78f2011-09-13 19:42:23 +000053
Greg Clayton6f6e4db2016-11-15 01:23:06 +000054 // Read all of the abbreviation attributes and forms.
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000055 while (true) {
Greg Clayton6c273762016-10-27 16:32:04 +000056 auto A = static_cast<Attribute>(Data.getULEB128(OffsetPtr));
57 auto F = static_cast<Form>(Data.getULEB128(OffsetPtr));
58 if (A && F) {
Greg Clayton6f6e4db2016-11-15 01:23:06 +000059 auto FixedFormByteSize = DWARFFormValue::getFixedByteSize(F);
60 AttributeSpecs.push_back(AttributeSpec(A, F, FixedFormByteSize));
61 // If this abbrevation still has a fixed byte size, then update the
62 // FixedAttributeSize as needed.
63 if (FixedAttributeSize) {
64 if (FixedFormByteSize)
65 FixedAttributeSize->NumBytes += *FixedFormByteSize;
66 else {
67 switch (F) {
68 case DW_FORM_addr:
69 ++FixedAttributeSize->NumAddrs;
70 break;
71
72 case DW_FORM_ref_addr:
73 ++FixedAttributeSize->NumRefAddrs;
74 break;
75
76 case DW_FORM_strp:
77 case DW_FORM_GNU_ref_alt:
78 case DW_FORM_GNU_strp_alt:
79 case DW_FORM_line_strp:
80 case DW_FORM_sec_offset:
81 case DW_FORM_strp_sup:
82 case DW_FORM_ref_sup:
83 ++FixedAttributeSize->NumDwarfOffsets;
84 break;
85
86 default:
87 // Indicate we no longer have a fixed byte size for this
88 // abbreviation by clearing the FixedAttributeSize optional value
89 // so it doesn't have a value.
90 FixedAttributeSize.reset();
91 break;
92 }
93 }
94 }
Greg Clayton6c273762016-10-27 16:32:04 +000095 } else if (A == 0 && F == 0) {
96 // We successfully reached the end of this abbreviation declaration
97 // since both attribute and form are zero.
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +000098 break;
Greg Clayton6c273762016-10-27 16:32:04 +000099 } else {
100 // Attribute and form pairs must either both be non-zero, in which case
101 // they are added to the abbreviation declaration, or both be zero to
102 // terminate the abbrevation declaration. In this case only one was
103 // zero which is an error.
104 clear();
105 return false;
106 }
Alexey Samsonovd5cc93c2013-10-31 17:20:14 +0000107 }
108 return true;
Benjamin Krameraa2f78f2011-09-13 19:42:23 +0000109}
110
111void DWARFAbbreviationDeclaration::dump(raw_ostream &OS) const {
Mehdi Amini149f6ea2016-10-05 05:59:29 +0000112 auto tagString = TagString(getTag());
Benjamin Kramer21a50922011-09-15 04:15:59 +0000113 OS << '[' << getCode() << "] ";
Mehdi Amini149f6ea2016-10-05 05:59:29 +0000114 if (!tagString.empty())
Benjamin Kramer21a50922011-09-15 04:15:59 +0000115 OS << tagString;
116 else
117 OS << format("DW_TAG_Unknown_%x", getTag());
118 OS << "\tDW_CHILDREN_" << (hasChildren() ? "yes" : "no") << '\n';
Alexey Samsonov1eabf982014-03-13 07:52:54 +0000119 for (const AttributeSpec &Spec : AttributeSpecs) {
Benjamin Kramer21a50922011-09-15 04:15:59 +0000120 OS << '\t';
Mehdi Amini149f6ea2016-10-05 05:59:29 +0000121 auto attrString = AttributeString(Spec.Attr);
122 if (!attrString.empty())
Benjamin Kramer21a50922011-09-15 04:15:59 +0000123 OS << attrString;
124 else
Alexey Samsonov1eabf982014-03-13 07:52:54 +0000125 OS << format("DW_AT_Unknown_%x", Spec.Attr);
Benjamin Kramer21a50922011-09-15 04:15:59 +0000126 OS << '\t';
Mehdi Amini149f6ea2016-10-05 05:59:29 +0000127 auto formString = FormEncodingString(Spec.Form);
128 if (!formString.empty())
Benjamin Kramer21a50922011-09-15 04:15:59 +0000129 OS << formString;
130 else
Alexey Samsonov1eabf982014-03-13 07:52:54 +0000131 OS << format("DW_FORM_Unknown_%x", Spec.Form);
Benjamin Kramer21a50922011-09-15 04:15:59 +0000132 OS << '\n';
Benjamin Kramer4137b6a2011-09-15 04:00:58 +0000133 }
Benjamin Krameraa2f78f2011-09-13 19:42:23 +0000134 OS << '\n';
135}
136
Greg Clayton6f6e4db2016-11-15 01:23:06 +0000137Optional<uint32_t>
138DWARFAbbreviationDeclaration::findAttributeIndex(dwarf::Attribute Attr) const {
Alexey Samsonov1eabf982014-03-13 07:52:54 +0000139 for (uint32_t i = 0, e = AttributeSpecs.size(); i != e; ++i) {
Greg Clayton6f6e4db2016-11-15 01:23:06 +0000140 if (AttributeSpecs[i].Attr == Attr)
Benjamin Krameraa2f78f2011-09-13 19:42:23 +0000141 return i;
142 }
Greg Clayton6f6e4db2016-11-15 01:23:06 +0000143 return None;
144}
145
146bool DWARFAbbreviationDeclaration::getAttributeValue(
147 const uint32_t DIEOffset, const dwarf::Attribute Attr, const DWARFUnit &U,
148 DWARFFormValue &FormValue) const {
149 Optional<uint32_t> MatchAttrIndex = findAttributeIndex(Attr);
150 if (!MatchAttrIndex)
151 return false;
152
153 auto DebugInfoData = U.getDebugInfoExtractor();
154
155 // Add the byte size of ULEB that for the abbrev Code so we can start
156 // skipping the attribute data.
157 uint32_t Offset = DIEOffset + CodeByteSize;
158 uint32_t AttrIndex = 0;
159 for (const auto &Spec : AttributeSpecs) {
160 if (*MatchAttrIndex == AttrIndex) {
161 // We have arrived at the attribute to extract, extract if from Offset.
162 FormValue.setForm(Spec.Form);
163 return FormValue.extractValue(DebugInfoData, &Offset, &U);
164 }
165 // March Offset along until we get to the attribute we want.
166 if (Optional<uint8_t> FixedSize = Spec.getByteSize(U))
167 Offset += *FixedSize;
168 else
169 DWARFFormValue::skipValue(Spec.Form, DebugInfoData, &Offset, &U);
170 ++AttrIndex;
171 }
172 return false;
173}
174
175size_t DWARFAbbreviationDeclaration::FixedSizeInfo::getByteSize(
176 const DWARFUnit &U) const {
177 size_t ByteSize = NumBytes;
178 if (NumAddrs)
179 ByteSize += NumAddrs * U.getAddressByteSize();
180 if (NumRefAddrs)
181 ByteSize += NumRefAddrs * U.getRefAddrByteSize();
182 if (NumDwarfOffsets)
183 ByteSize += NumDwarfOffsets * U.getDwarfOffsetByteSize();
184 return ByteSize;
185}
186
187Optional<uint8_t> DWARFAbbreviationDeclaration::AttributeSpec::getByteSize(
188 const DWARFUnit &U) const {
189 return ByteSize ? ByteSize : DWARFFormValue::getFixedByteSize(Form, &U);
190}
191
192Optional<size_t> DWARFAbbreviationDeclaration::getFixedAttributesByteSize(
193 const DWARFUnit &U) const {
194 if (FixedAttributeSize)
195 return FixedAttributeSize->getByteSize(U);
196 return None;
Benjamin Krameraa2f78f2011-09-13 19:42:23 +0000197}