[MLIR] Prevent invalid IR from being passed outside of RemoveDeadValues (#121079)
[llvm-project.git] / llvm / lib / DebugInfo / PDB / UDTLayout.cpp
blob679881054f9da96513ba666b5c4400d629a84618
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 //===----------------------------------------------------------------------===//
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>
33 using namespace llvm;
34 using namespace llvm::pdb;
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);
43 static uint32_t getTypeLength(const PDBSymbol &Symbol) {
44 auto SymbolType = getSymbolType(Symbol);
45 const IPDBRawSymbol &RawType = SymbolType->getRawSymbol();
47 return RawType.getLength();
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);
60 uint32_t LayoutItemBase::deepPaddingSize() const {
61 return UsedBytes.size() - UsedBytes.count();
64 uint32_t LayoutItemBase::tailPadding() const {
65 int Last = UsedBytes.find_last();
67 return UsedBytes.size() - (Last + 1);
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();
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)) {
89 const PDBSymbolData &DataMemberLayoutItem::getDataMember() {
90 return *cast<PDBSymbolData>(Symbol);
93 bool DataMemberLayoutItem::hasUDTLayout() const { return UdtLayout != nullptr; }
95 const ClassLayout &DataMemberLayoutItem::getUDTLayout() const {
96 return *UdtLayout;
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();
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);
115 initializeChildren(Sym);
116 if (LayoutSize < Size)
117 UsedBytes.resize(LayoutSize);
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;
130 return Abs;
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);
145 ClassLayout::ClassLayout(std::unique_ptr<PDBSymbolTypeUDT> UDT)
146 : ClassLayout(*UDT) {
147 OwnedStorage = std::move(UDT);
150 uint32_t ClassLayout::immediatePadding() const {
151 return SizeOf - ImmediateUsedBytes.count();
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);
165 IsVirtualBase = Base->isVirtualBaseClass();
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;
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));
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));
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());
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));
214 AllBases.push_back(BL.get());
215 addChildToLayout(std::move(BL));
217 NonVirtualBases = AllBases;
219 assert(VTables.size() <= 1);
220 if (!VTables.empty()) {
221 auto VTLayout =
222 std::make_unique<VTableLayoutItem>(*this, std::move(VTables[0]));
224 VTable = VTLayout.get();
226 addChildToLayout(std::move(VTLayout));
229 for (auto &Data : Members) {
230 auto DM = std::make_unique<DataMemberLayoutItem>(*this, std::move(Data));
232 addChildToLayout(std::move(DM));
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));
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());
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));
264 VirtualBases = ArrayRef(AllBases).drop_front(NonVirtualBases.size());
266 if (Parent != nullptr)
267 LayoutSize = UsedBytes.find_last() + 1;
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;
277 return false;
280 void UDTLayoutBase::addChildToLayout(std::unique_ptr<LayoutItemBase> Child) {
281 uint32_t Begin = Child->getOffsetInParent();
283 if (!Child->isElided()) {
284 BitVector ChildBytes = Child->usedBytes();
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;
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());
300 LayoutItems.insert(Loc, Child.get());
304 ChildStorage.push_back(std::move(Child));