-
Notifications
You must be signed in to change notification settings - Fork 14.8k
Description
When compiling following source, clang creates a virtual destructor for A
with an artificial vtt
argument. The DILocalVariable
for vtt
correctly has arg: 2
.
However, the field types
of the DISubroutineType
for this virtual destructor seems to miss the additional parameter, as it only has two members and thus represents a function with only one parameter.
Source:
struct B {
virtual ~B() {}
};
struct A : virtual B {
};
A a;
LLVM-IR:
[...]
!5 = distinct !DICompositeType(tag: DW_TAG_structure_type, name: "A", file: !3, line: 5, size: 64, flags: DIFlagTypePassByReference | DIFlagNonTrivial, elements: !6, vtableHolder: !8, identifier: "_ZTS1A")
[...]
!34 = !DISubroutineType(types: !35)
!35 = !{null, !36}
!36 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !5, size: 64, flags: DIFlagArtificial | DIFlagObjectPointer)
[...]
!70 = distinct !DISubprogram(name: "~A", linkageName: "_ZN1AD2Ev", scope: !5, file: !3, line: 5, type: !34, scopeLine: 5, flags: DIFlagArtificial | DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !2, declaration: !44, retainedNodes: !38)
!71 = !DILocalVariable(name: "this", arg: 1, scope: !70, type: !40, flags: DIFlagArtificial | DIFlagObjectPointer)
!72 = !DILocation(line: 0, scope: !70)
!73 = !DILocalVariable(name: "vtt", arg: 2, scope: !70, type: !74, flags: DIFlagArtificial)
!74 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !75, size: 64)
!75 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: null, size: 64)
[...]
I generated the IR using clang 19.1.0-rc2 on x86_64-unknown-linux-gnu
.
DWARF generated from this IR is not affected and includes all formal parameters.
The issue seems to be that in ItaniumCXXABI::addImplicitStructorParams
, the parameter is added, but in CGDebugInfo::getOrCreateMethodType
another type array is read to create the DISubroutineType
.
llvm-project/clang/lib/CodeGen/ItaniumCXXABI.cpp
Lines 1870 to 1890 in d033ae1
void ItaniumCXXABI::addImplicitStructorParams(CodeGenFunction &CGF, | |
QualType &ResTy, | |
FunctionArgList &Params) { | |
const CXXMethodDecl *MD = cast<CXXMethodDecl>(CGF.CurGD.getDecl()); | |
assert(isa<CXXConstructorDecl>(MD) || isa<CXXDestructorDecl>(MD)); | |
// Check if we need a VTT parameter as well. | |
if (NeedsVTTParameter(CGF.CurGD)) { | |
ASTContext &Context = getContext(); | |
// FIXME: avoid the fake decl | |
LangAS AS = CGM.GetGlobalVarAddressSpace(nullptr); | |
QualType Q = Context.getAddrSpaceQualType(Context.VoidPtrTy, AS); | |
QualType T = Context.getPointerType(Q); | |
auto *VTTDecl = ImplicitParamDecl::Create( | |
Context, /*DC=*/nullptr, MD->getLocation(), &Context.Idents.get("vtt"), | |
T, ImplicitParamKind::CXXVTT); | |
Params.insert(Params.begin() + 1, VTTDecl); | |
getStructorImplicitParamDecl(CGF) = VTTDecl; | |
} | |
} |
llvm-project/clang/lib/CodeGen/CGDebugInfo.cpp
Lines 1938 to 1946 in d033ae1
llvm::DISubroutineType * | |
CGDebugInfo::getOrCreateMethodType(const CXXMethodDecl *Method, | |
llvm::DIFile *Unit) { | |
const FunctionProtoType *Func = Method->getType()->getAs<FunctionProtoType>(); | |
if (Method->isStatic()) | |
return cast_or_null<llvm::DISubroutineType>( | |
getOrCreateType(QualType(Func, 0), Unit)); | |
return getOrCreateInstanceMethodType(Method->getThisType(), Func, Unit); | |
} |