forked from OSchip/llvm-project
Improve support of PDB as an external layout source
Summary: This patch improves support of PDB as an external layout source in the next cases: - Multiple non-virtual inheritance from packed base classes. When using external layout, there's no need to align `NonVirtualSize` of a base class. It may cause an overlapping when the next base classes will be layouted (but there is a slightly different case in the test because I can't find a way to specify a base offset); - Support of nameless structs and unions. There is no info about nameless child structs and unions in Microsoft cl-emitted PDBs. Instead all its fields are just treated as outer structure's (union's) fields. This also causes a fields overlapping, and makes it possible for unions to have fields located at a non-zero offset. Reviewers: rsmith, zturner, rnk, mstorsjo, majnemer Reviewed By: rnk Subscribers: cfe-commits Tags: #clang Differential Revision: https://reviews.llvm.org/D49871 llvm-svn: 338353
This commit is contained in:
parent
615540d0f2
commit
09240ef776
|
@ -2452,7 +2452,9 @@ void MicrosoftRecordLayoutBuilder::cxxLayout(const CXXRecordDecl *RD) {
|
||||||
auto RoundingAlignment = Alignment;
|
auto RoundingAlignment = Alignment;
|
||||||
if (!MaxFieldAlignment.isZero())
|
if (!MaxFieldAlignment.isZero())
|
||||||
RoundingAlignment = std::min(RoundingAlignment, MaxFieldAlignment);
|
RoundingAlignment = std::min(RoundingAlignment, MaxFieldAlignment);
|
||||||
NonVirtualSize = Size = Size.alignTo(RoundingAlignment);
|
if (!UseExternalLayout)
|
||||||
|
Size = Size.alignTo(RoundingAlignment);
|
||||||
|
NonVirtualSize = Size;
|
||||||
RequiredAlignment = std::max(
|
RequiredAlignment = std::max(
|
||||||
RequiredAlignment, Context.toCharUnitsFromBits(RD->getMaxAlignment()));
|
RequiredAlignment, Context.toCharUnitsFromBits(RD->getMaxAlignment()));
|
||||||
layoutVirtualBases(RD);
|
layoutVirtualBases(RD);
|
||||||
|
@ -2653,21 +2655,16 @@ void MicrosoftRecordLayoutBuilder::layoutField(const FieldDecl *FD) {
|
||||||
LastFieldIsNonZeroWidthBitfield = false;
|
LastFieldIsNonZeroWidthBitfield = false;
|
||||||
ElementInfo Info = getAdjustedElementInfo(FD);
|
ElementInfo Info = getAdjustedElementInfo(FD);
|
||||||
Alignment = std::max(Alignment, Info.Alignment);
|
Alignment = std::max(Alignment, Info.Alignment);
|
||||||
if (IsUnion) {
|
CharUnits FieldOffset;
|
||||||
placeFieldAtOffset(CharUnits::Zero());
|
if (UseExternalLayout)
|
||||||
Size = std::max(Size, Info.Size);
|
FieldOffset =
|
||||||
} else {
|
Context.toCharUnitsFromBits(External.getExternalFieldOffset(FD));
|
||||||
CharUnits FieldOffset;
|
else if (IsUnion)
|
||||||
if (UseExternalLayout) {
|
FieldOffset = CharUnits::Zero();
|
||||||
FieldOffset =
|
else
|
||||||
Context.toCharUnitsFromBits(External.getExternalFieldOffset(FD));
|
FieldOffset = Size.alignTo(Info.Alignment);
|
||||||
assert(FieldOffset >= Size && "field offset already allocated");
|
placeFieldAtOffset(FieldOffset);
|
||||||
} else {
|
Size = std::max(Size, FieldOffset + Info.Size);
|
||||||
FieldOffset = Size.alignTo(Info.Alignment);
|
|
||||||
}
|
|
||||||
placeFieldAtOffset(FieldOffset);
|
|
||||||
Size = FieldOffset + Info.Size;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MicrosoftRecordLayoutBuilder::layoutBitField(const FieldDecl *FD) {
|
void MicrosoftRecordLayoutBuilder::layoutBitField(const FieldDecl *FD) {
|
||||||
|
@ -2692,18 +2689,17 @@ void MicrosoftRecordLayoutBuilder::layoutBitField(const FieldDecl *FD) {
|
||||||
}
|
}
|
||||||
LastFieldIsNonZeroWidthBitfield = true;
|
LastFieldIsNonZeroWidthBitfield = true;
|
||||||
CurrentBitfieldSize = Info.Size;
|
CurrentBitfieldSize = Info.Size;
|
||||||
if (IsUnion) {
|
if (UseExternalLayout) {
|
||||||
placeFieldAtOffset(CharUnits::Zero());
|
|
||||||
Size = std::max(Size, Info.Size);
|
|
||||||
// TODO: Add a Sema warning that MS ignores bitfield alignment in unions.
|
|
||||||
} else if (UseExternalLayout) {
|
|
||||||
auto FieldBitOffset = External.getExternalFieldOffset(FD);
|
auto FieldBitOffset = External.getExternalFieldOffset(FD);
|
||||||
placeFieldAtBitOffset(FieldBitOffset);
|
placeFieldAtBitOffset(FieldBitOffset);
|
||||||
auto NewSize = Context.toCharUnitsFromBits(
|
auto NewSize = Context.toCharUnitsFromBits(
|
||||||
llvm::alignTo(FieldBitOffset + Width, Context.getCharWidth()));
|
llvm::alignTo(FieldBitOffset + Width, Context.getCharWidth()));
|
||||||
assert(NewSize >= Size && "bit field offset already allocated");
|
Size = std::max(Size, NewSize);
|
||||||
Size = NewSize;
|
|
||||||
Alignment = std::max(Alignment, Info.Alignment);
|
Alignment = std::max(Alignment, Info.Alignment);
|
||||||
|
} else if (IsUnion) {
|
||||||
|
placeFieldAtOffset(CharUnits::Zero());
|
||||||
|
Size = std::max(Size, Info.Size);
|
||||||
|
// TODO: Add a Sema warning that MS ignores bitfield alignment in unions.
|
||||||
} else {
|
} else {
|
||||||
// Allocate a new block of memory and place the bitfield in it.
|
// Allocate a new block of memory and place the bitfield in it.
|
||||||
CharUnits FieldOffset = Size.alignTo(Info.Alignment);
|
CharUnits FieldOffset = Size.alignTo(Info.Alignment);
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
|
||||||
|
*** Dumping AST Record Layout
|
||||||
|
Type: struct S
|
||||||
|
|
||||||
|
Layout: <ASTRecordLayout
|
||||||
|
Size:64
|
||||||
|
Alignment:32
|
||||||
|
FieldOffsets: [0, 32, 32]>
|
||||||
|
|
||||||
|
*** Dumping AST Record Layout
|
||||||
|
Type: union U
|
||||||
|
|
||||||
|
Layout: <ASTRecordLayout
|
||||||
|
Size:96
|
||||||
|
Alignment:32
|
||||||
|
FieldOffsets: [0, 0, 32, 64, 68, 73]>
|
|
@ -0,0 +1,18 @@
|
||||||
|
|
||||||
|
*** Dumping AST Record Layout
|
||||||
|
Type: class B<0>
|
||||||
|
|
||||||
|
Layout: <ASTRecordLayout
|
||||||
|
FieldOffsets: [0, 32]>
|
||||||
|
|
||||||
|
*** Dumping AST Record Layout
|
||||||
|
Type: class B<1>
|
||||||
|
|
||||||
|
Layout: <ASTRecordLayout
|
||||||
|
FieldOffsets: [0, 32]>
|
||||||
|
|
||||||
|
*** Dumping AST Record Layout
|
||||||
|
Type: class C
|
||||||
|
|
||||||
|
Layout: <ASTRecordLayout
|
||||||
|
FieldOffsets: [80]>
|
|
@ -0,0 +1,33 @@
|
||||||
|
// RUN: %clang_cc1 -w -fdump-record-layouts-simple -foverride-record-layout=%S/Inputs/override-layout-nameless-struct-union.layout %s | FileCheck %s
|
||||||
|
|
||||||
|
// CHECK: Type: struct S
|
||||||
|
// CHECK: Size:64
|
||||||
|
// CHECK: Alignment:32
|
||||||
|
// CHECK: FieldOffsets: [0, 32, 32]
|
||||||
|
struct S {
|
||||||
|
short _s;
|
||||||
|
//union {
|
||||||
|
int _su0;
|
||||||
|
char _su1;
|
||||||
|
//};
|
||||||
|
};
|
||||||
|
|
||||||
|
// CHECK: Type: union U
|
||||||
|
// CHECK: Size:96
|
||||||
|
// CHECK: Alignment:32
|
||||||
|
// CHECK: FieldOffsets: [0, 0, 32, 64, 68, 73]
|
||||||
|
union U {
|
||||||
|
short _u;
|
||||||
|
//struct {
|
||||||
|
char _us0;
|
||||||
|
int _us1;
|
||||||
|
unsigned _us20 : 4;
|
||||||
|
unsigned _us21 : 5;
|
||||||
|
unsigned _us22 : 6;
|
||||||
|
//};
|
||||||
|
};
|
||||||
|
|
||||||
|
void use_structs() {
|
||||||
|
S ss[sizeof(S)];
|
||||||
|
U us[sizeof(U)];
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
// RUN: %clang_cc1 -w -fdump-record-layouts-simple -foverride-record-layout=%S/Inputs/override-layout-packed-base.layout %s | FileCheck %s
|
||||||
|
|
||||||
|
// CHECK: Type: class B<0>
|
||||||
|
// CHECK: FieldOffsets: [0, 32]
|
||||||
|
|
||||||
|
// CHECK: Type: class B<1>
|
||||||
|
// CHECK: FieldOffsets: [0, 32]
|
||||||
|
|
||||||
|
//#pragma pack(push, 1)
|
||||||
|
template<int I>
|
||||||
|
class B {
|
||||||
|
int _b1;
|
||||||
|
char _b2;
|
||||||
|
};
|
||||||
|
//#pragma pack(pop)
|
||||||
|
|
||||||
|
// CHECK: Type: class C
|
||||||
|
// CHECK: FieldOffsets: [80]
|
||||||
|
|
||||||
|
class C : B<0>, B<1> {
|
||||||
|
char _c;
|
||||||
|
};
|
||||||
|
|
||||||
|
void use_structs() {
|
||||||
|
C cs[sizeof(C)];
|
||||||
|
}
|
Loading…
Reference in New Issue