xref: /freebsd/contrib/llvm-project/llvm/lib/DebugInfo/PDB/UDTLayout.cpp (revision 51015e6d0f570239b0c2088dc6cf2b018928375d)
1 //===- UDTLayout.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/PDB/UDTLayout.h"
10 #include "llvm/ADT/ArrayRef.h"
11 #include "llvm/ADT/BitVector.h"
12 #include "llvm/ADT/STLExtras.h"
13 #include "llvm/DebugInfo/PDB/IPDBEnumChildren.h"
14 #include "llvm/DebugInfo/PDB/IPDBLineNumber.h"
15 #include "llvm/DebugInfo/PDB/IPDBRawSymbol.h"
16 #include "llvm/DebugInfo/PDB/IPDBSession.h"
17 #include "llvm/DebugInfo/PDB/PDBSymbol.h"
18 #include "llvm/DebugInfo/PDB/PDBSymbolData.h"
19 #include "llvm/DebugInfo/PDB/PDBSymbolFunc.h"
20 #include "llvm/DebugInfo/PDB/PDBSymbolTypeBaseClass.h"
21 #include "llvm/DebugInfo/PDB/PDBSymbolTypeBuiltin.h"
22 #include "llvm/DebugInfo/PDB/PDBSymbolTypeFunctionSig.h"
23 #include "llvm/DebugInfo/PDB/PDBSymbolTypePointer.h"
24 #include "llvm/DebugInfo/PDB/PDBSymbolTypeUDT.h"
25 #include "llvm/DebugInfo/PDB/PDBSymbolTypeVTable.h"
26 #include "llvm/DebugInfo/PDB/PDBTypes.h"
27 #include "llvm/Support/Casting.h"
28 #include <algorithm>
29 #include <cassert>
30 #include <cstdint>
31 #include <memory>
32 
33 using namespace llvm;
34 using namespace llvm::pdb;
35 
36 static std::unique_ptr<PDBSymbol> getSymbolType(const PDBSymbol &Symbol) {
37   const IPDBSession &Session = Symbol.getSession();
38   const IPDBRawSymbol &RawSymbol = Symbol.getRawSymbol();
39   uint32_t TypeId = RawSymbol.getTypeId();
40   return Session.getSymbolById(TypeId);
41 }
42 
43 static uint32_t getTypeLength(const PDBSymbol &Symbol) {
44   auto SymbolType = getSymbolType(Symbol);
45   const IPDBRawSymbol &RawType = SymbolType->getRawSymbol();
46 
47   return RawType.getLength();
48 }
49 
50 LayoutItemBase::LayoutItemBase(const UDTLayoutBase *Parent,
51                                const PDBSymbol *Symbol, const std::string &Name,
52                                uint32_t OffsetInParent, uint32_t Size,
53                                bool IsElided)
54     : Symbol(Symbol), Parent(Parent), Name(Name),
55       OffsetInParent(OffsetInParent), SizeOf(Size), LayoutSize(Size),
56       IsElided(IsElided) {
57   UsedBytes.resize(SizeOf, true);
58 }
59 
60 uint32_t LayoutItemBase::deepPaddingSize() const {
61   return UsedBytes.size() - UsedBytes.count();
62 }
63 
64 uint32_t LayoutItemBase::tailPadding() const {
65   int Last = UsedBytes.find_last();
66 
67   return UsedBytes.size() - (Last + 1);
68 }
69 
70 DataMemberLayoutItem::DataMemberLayoutItem(
71     const UDTLayoutBase &Parent, std::unique_ptr<PDBSymbolData> Member)
72     : LayoutItemBase(&Parent, Member.get(), Member->getName(),
73                      Member->getOffset(), getTypeLength(*Member), false),
74       DataMember(std::move(Member)) {
75   auto Type = DataMember->getType();
76   if (auto UDT = unique_dyn_cast<PDBSymbolTypeUDT>(Type)) {
77     UdtLayout = std::make_unique<ClassLayout>(std::move(UDT));
78     UsedBytes = UdtLayout->usedBytes();
79   }
80 }
81 
82 VBPtrLayoutItem::VBPtrLayoutItem(const UDTLayoutBase &Parent,
83                                  std::unique_ptr<PDBSymbolTypeBuiltin> Sym,
84                                  uint32_t Offset, uint32_t Size)
85     : LayoutItemBase(&Parent, Sym.get(), "<vbptr>", Offset, Size, false),
86       Type(std::move(Sym)) {
87 }
88 
89 const PDBSymbolData &DataMemberLayoutItem::getDataMember() {
90   return *cast<PDBSymbolData>(Symbol);
91 }
92 
93 bool DataMemberLayoutItem::hasUDTLayout() const { return UdtLayout != nullptr; }
94 
95 const ClassLayout &DataMemberLayoutItem::getUDTLayout() const {
96   return *UdtLayout;
97 }
98 
99 VTableLayoutItem::VTableLayoutItem(const UDTLayoutBase &Parent,
100                                    std::unique_ptr<PDBSymbolTypeVTable> VT)
101     : LayoutItemBase(&Parent, VT.get(), "<vtbl>", 0, getTypeLength(*VT), false),
102       VTable(std::move(VT)) {
103   auto VTableType = cast<PDBSymbolTypePointer>(VTable->getType());
104   ElementSize = VTableType->getLength();
105 }
106 
107 UDTLayoutBase::UDTLayoutBase(const UDTLayoutBase *Parent, const PDBSymbol &Sym,
108                              const std::string &Name, uint32_t OffsetInParent,
109                              uint32_t Size, bool IsElided)
110     : LayoutItemBase(Parent, &Sym, Name, OffsetInParent, Size, IsElided) {
111   // UDT storage comes from a union of all the children's storage, so start out
112   // uninitialized.
113   UsedBytes.reset(0, Size);
114 
115   initializeChildren(Sym);
116   if (LayoutSize < Size)
117     UsedBytes.resize(LayoutSize);
118 }
119 
120 uint32_t UDTLayoutBase::tailPadding() const {
121   uint32_t Abs = LayoutItemBase::tailPadding();
122   if (!LayoutItems.empty()) {
123     const LayoutItemBase *Back = LayoutItems.back();
124     uint32_t ChildPadding = Back->LayoutItemBase::tailPadding();
125     if (Abs < ChildPadding)
126       Abs = 0;
127     else
128       Abs -= ChildPadding;
129   }
130   return Abs;
131 }
132 
133 ClassLayout::ClassLayout(const PDBSymbolTypeUDT &UDT)
134     : UDTLayoutBase(nullptr, UDT, UDT.getName(), 0, UDT.getLength(), false),
135       UDT(UDT) {
136   ImmediateUsedBytes.resize(SizeOf, false);
137   for (auto &LI : LayoutItems) {
138     uint32_t Begin = LI->getOffsetInParent();
139     uint32_t End = Begin + LI->getLayoutSize();
140     End = std::min(SizeOf, End);
141     ImmediateUsedBytes.set(Begin, End);
142   }
143 }
144 
145 ClassLayout::ClassLayout(std::unique_ptr<PDBSymbolTypeUDT> UDT)
146     : ClassLayout(*UDT) {
147   OwnedStorage = std::move(UDT);
148 }
149 
150 uint32_t ClassLayout::immediatePadding() const {
151   return SizeOf - ImmediateUsedBytes.count();
152 }
153 
154 BaseClassLayout::BaseClassLayout(const UDTLayoutBase &Parent,
155                                  uint32_t OffsetInParent, bool Elide,
156                                  std::unique_ptr<PDBSymbolTypeBaseClass> B)
157     : UDTLayoutBase(&Parent, *B, B->getName(), OffsetInParent, B->getLength(),
158                     Elide),
159       Base(std::move(B)) {
160   if (isEmptyBase()) {
161     // Special case an empty base so that it doesn't get treated as padding.
162     UsedBytes.resize(1);
163     UsedBytes.set(0);
164   }
165   IsVirtualBase = Base->isVirtualBaseClass();
166 }
167 
168 void UDTLayoutBase::initializeChildren(const PDBSymbol &Sym) {
169   // Handled bases first, followed by VTables, followed by data members,
170   // followed by functions, followed by other.  This ordering is necessary
171   // so that bases and vtables get initialized before any functions which
172   // may override them.
173   UniquePtrVector<PDBSymbolTypeBaseClass> Bases;
174   UniquePtrVector<PDBSymbolTypeVTable> VTables;
175   UniquePtrVector<PDBSymbolData> Members;
176   UniquePtrVector<PDBSymbolTypeBaseClass> VirtualBaseSyms;
177 
178   auto Children = Sym.findAllChildren();
179   while (auto Child = Children->getNext()) {
180     if (auto Base = unique_dyn_cast<PDBSymbolTypeBaseClass>(Child)) {
181       if (Base->isVirtualBaseClass())
182         VirtualBaseSyms.push_back(std::move(Base));
183       else
184         Bases.push_back(std::move(Base));
185     }
186     else if (auto Data = unique_dyn_cast<PDBSymbolData>(Child)) {
187       if (Data->getDataKind() == PDB_DataKind::Member)
188         Members.push_back(std::move(Data));
189       else
190         Other.push_back(std::move(Data));
191     } else if (auto VT = unique_dyn_cast<PDBSymbolTypeVTable>(Child))
192       VTables.push_back(std::move(VT));
193     else if (auto Func = unique_dyn_cast<PDBSymbolFunc>(Child))
194       Funcs.push_back(std::move(Func));
195     else {
196       Other.push_back(std::move(Child));
197     }
198   }
199 
200   // We don't want to have any re-allocations in the list of bases, so make
201   // sure to reserve enough space so that our ArrayRefs don't get invalidated.
202   AllBases.reserve(Bases.size() + VirtualBaseSyms.size());
203 
204   // Only add non-virtual bases to the class first.  Only at the end of the
205   // class, after all non-virtual bases and data members have been added do we
206   // add virtual bases.  This way the offsets are correctly aligned when we go
207   // to lay out virtual bases.
208   for (auto &Base : Bases) {
209     uint32_t Offset = Base->getOffset();
210     // Non-virtual bases never get elided.
211     auto BL = std::make_unique<BaseClassLayout>(*this, Offset, false,
212                                                  std::move(Base));
213 
214     AllBases.push_back(BL.get());
215     addChildToLayout(std::move(BL));
216   }
217   NonVirtualBases = AllBases;
218 
219   assert(VTables.size() <= 1);
220   if (!VTables.empty()) {
221     auto VTLayout =
222         std::make_unique<VTableLayoutItem>(*this, std::move(VTables[0]));
223 
224     VTable = VTLayout.get();
225 
226     addChildToLayout(std::move(VTLayout));
227   }
228 
229   for (auto &Data : Members) {
230     auto DM = std::make_unique<DataMemberLayoutItem>(*this, std::move(Data));
231 
232     addChildToLayout(std::move(DM));
233   }
234 
235   // Make sure add virtual bases before adding functions, since functions may be
236   // overrides of virtual functions declared in a virtual base, so the VTables
237   // and virtual intros need to be correctly initialized.
238   for (auto &VB : VirtualBaseSyms) {
239     int VBPO = VB->getVirtualBasePointerOffset();
240     if (!hasVBPtrAtOffset(VBPO)) {
241       if (auto VBP = VB->getRawSymbol().getVirtualBaseTableType()) {
242         auto VBPL = std::make_unique<VBPtrLayoutItem>(*this, std::move(VBP),
243                                                        VBPO, VBP->getLength());
244         VBPtr = VBPL.get();
245         addChildToLayout(std::move(VBPL));
246       }
247     }
248 
249     // Virtual bases always go at the end.  So just look for the last place we
250     // ended when writing something, and put our virtual base there.
251     // Note that virtual bases get elided unless this is a top-most derived
252     // class.
253     uint32_t Offset = UsedBytes.find_last() + 1;
254     bool Elide = (Parent != nullptr);
255     auto BL =
256         std::make_unique<BaseClassLayout>(*this, Offset, Elide, std::move(VB));
257     AllBases.push_back(BL.get());
258 
259     // Only lay this virtual base out directly inside of *this* class if this
260     // is a top-most derived class.  Keep track of it regardless, but only
261     // physically lay it out if it's a topmost derived class.
262     addChildToLayout(std::move(BL));
263   }
264   VirtualBases = makeArrayRef(AllBases).drop_front(NonVirtualBases.size());
265 
266   if (Parent != nullptr)
267     LayoutSize = UsedBytes.find_last() + 1;
268 }
269 
270 bool UDTLayoutBase::hasVBPtrAtOffset(uint32_t Off) const {
271   if (VBPtr && VBPtr->getOffsetInParent() == Off)
272     return true;
273   for (BaseClassLayout *BL : AllBases) {
274     if (BL->hasVBPtrAtOffset(Off - BL->getOffsetInParent()))
275       return true;
276   }
277   return false;
278 }
279 
280 void UDTLayoutBase::addChildToLayout(std::unique_ptr<LayoutItemBase> Child) {
281   uint32_t Begin = Child->getOffsetInParent();
282 
283   if (!Child->isElided()) {
284     BitVector ChildBytes = Child->usedBytes();
285 
286     // Suppose the child occupies 4 bytes starting at offset 12 in a 32 byte
287     // class.  When we call ChildBytes.resize(32), the Child's storage will
288     // still begin at offset 0, so we need to shift it left by offset bytes
289     // to get it into the right position.
290     ChildBytes.resize(UsedBytes.size());
291     ChildBytes <<= Child->getOffsetInParent();
292     UsedBytes |= ChildBytes;
293 
294     if (ChildBytes.count() > 0) {
295       auto Loc = llvm::upper_bound(
296           LayoutItems, Begin, [](uint32_t Off, const LayoutItemBase *Item) {
297             return (Off < Item->getOffsetInParent());
298           });
299 
300       LayoutItems.insert(Loc, Child.get());
301     }
302   }
303 
304   ChildStorage.push_back(std::move(Child));
305 }
306