forked from OSchip/llvm-project
Mark static member functions as static in CodeViewDebug
Summary: To improve CodeView quality for static member functions, we need to make the static explicit. In addition to a small change in LLVM's CodeViewDebug to return the appropriate MethodKind, this requires a small change in Clang to note the staticness in the debug info metadata. Subscribers: aprantl, hiraditya Differential Revision: https://reviews.llvm.org/D37715 llvm-svn: 313192
This commit is contained in:
parent
1929ffd452
commit
d91bf3998f
|
@ -1408,6 +1408,8 @@ llvm::DISubprogram *CGDebugInfo::CreateCXXMemberFunction(
|
|||
ContainingType = RecordTy;
|
||||
}
|
||||
|
||||
if (Method->isStatic())
|
||||
Flags |= llvm::DINode::FlagStaticMember;
|
||||
if (Method->isImplicit())
|
||||
Flags |= llvm::DINode::FlagArtificial;
|
||||
Flags |= getAccessFlag(Method->getAccess(), Method->getParent());
|
||||
|
|
|
@ -6,6 +6,7 @@ struct Foo {
|
|||
virtual void f();
|
||||
virtual void g();
|
||||
virtual void h();
|
||||
static void i(int, int);
|
||||
struct Nested {};
|
||||
};
|
||||
Foo f;
|
||||
|
@ -18,7 +19,7 @@ Foo::Nested n;
|
|||
// CHECK-SAME: elements: ![[elements:[0-9]+]]
|
||||
// CHECK-SAME: identifier: ".?AUFoo@@"
|
||||
|
||||
// CHECK: ![[elements]] = !{![[vshape:[0-9]+]], ![[vptr:[0-9]+]], ![[Nested]], ![[f:[0-9]+]], ![[g:[0-9]+]], ![[h:[0-9]+]]}
|
||||
// CHECK: ![[elements]] = !{![[vshape:[0-9]+]], ![[vptr:[0-9]+]], ![[Nested]], ![[f:[0-9]+]], ![[g:[0-9]+]], ![[h:[0-9]+]], ![[i:[0-9]+]]}
|
||||
|
||||
// CHECK: ![[vshape]] = !DIDerivedType(tag: DW_TAG_pointer_type, name: "__vtbl_ptr_type", baseType: null, size: 96)
|
||||
|
||||
|
@ -38,3 +39,9 @@ Foo::Nested n;
|
|||
// CHECK: ![[h]] = !DISubprogram(name: "h",
|
||||
// CHECK-SAME: containingType: ![[Foo]], virtuality: DW_VIRTUALITY_virtual, virtualIndex: 2,
|
||||
// CHECK-SAME: flags: DIFlagPrototyped | DIFlagIntroducedVirtual,
|
||||
|
||||
// CHECK: ![[i]] = !DISubprogram(name: "i",
|
||||
// CHECK-SAME: flags: DIFlagPrototyped | DIFlagStaticMember
|
||||
// CHECK-NEXT: ![[dummy:[0-9]+]] = !DISubroutineType(types: ![[Signature:[0-9]+]])
|
||||
// CHECK: ![[Signature]] = !{null, ![[BasicInt:[0-9]+]], ![[BasicInt]]}
|
||||
// CHECK: ![[BasicInt]] = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed)
|
||||
|
|
|
@ -325,8 +325,9 @@ TypeIndex CodeViewDebug::getMemberFunctionType(const DISubprogram *SP,
|
|||
// function type, as the complete class type is likely to reference this
|
||||
// member function type.
|
||||
TypeLoweringScope S(*this);
|
||||
TypeIndex TI =
|
||||
lowerTypeMemberFunction(SP->getType(), Class, SP->getThisAdjustment());
|
||||
const bool IsStaticMethod = (SP->getFlags() & DINode::FlagStaticMember) != 0;
|
||||
TypeIndex TI = lowerTypeMemberFunction(
|
||||
SP->getType(), Class, SP->getThisAdjustment(), IsStaticMethod);
|
||||
return recordTypeIndexForDINode(SP, TI, Class);
|
||||
}
|
||||
|
||||
|
@ -1221,7 +1222,8 @@ TypeIndex CodeViewDebug::lowerType(const DIType *Ty, const DIType *ClassTy) {
|
|||
// The member function type of a member function pointer has no
|
||||
// ThisAdjustment.
|
||||
return lowerTypeMemberFunction(cast<DISubroutineType>(Ty), ClassTy,
|
||||
/*ThisAdjustment=*/0);
|
||||
/*ThisAdjustment=*/0,
|
||||
/*IsStaticMethod=*/false);
|
||||
}
|
||||
return lowerTypeFunction(cast<DISubroutineType>(Ty));
|
||||
case dwarf::DW_TAG_enumeration_type:
|
||||
|
@ -1542,7 +1544,8 @@ TypeIndex CodeViewDebug::lowerTypeFunction(const DISubroutineType *Ty) {
|
|||
|
||||
TypeIndex CodeViewDebug::lowerTypeMemberFunction(const DISubroutineType *Ty,
|
||||
const DIType *ClassTy,
|
||||
int ThisAdjustment) {
|
||||
int ThisAdjustment,
|
||||
bool IsStaticMethod) {
|
||||
// Lower the containing class type.
|
||||
TypeIndex ClassType = getTypeIndex(ClassTy);
|
||||
|
||||
|
@ -1558,7 +1561,7 @@ TypeIndex CodeViewDebug::lowerTypeMemberFunction(const DISubroutineType *Ty,
|
|||
ArgTypeIndices = ReturnAndArgTypesRef.drop_front();
|
||||
}
|
||||
TypeIndex ThisTypeIndex = TypeIndex::Void();
|
||||
if (!ArgTypeIndices.empty()) {
|
||||
if (!IsStaticMethod && !ArgTypeIndices.empty()) {
|
||||
ThisTypeIndex = ArgTypeIndices.front();
|
||||
ArgTypeIndices = ArgTypeIndices.drop_front();
|
||||
}
|
||||
|
@ -1568,9 +1571,7 @@ TypeIndex CodeViewDebug::lowerTypeMemberFunction(const DISubroutineType *Ty,
|
|||
|
||||
CallingConvention CC = dwarfCCToCodeView(Ty->getCC());
|
||||
|
||||
// TODO: Need to use the correct values for:
|
||||
// FunctionOptions
|
||||
// ThisPointerAdjustment.
|
||||
// TODO: Need to use the correct values for FunctionOptions.
|
||||
MemberFunctionRecord MFR(ReturnTypeIndex, ClassType, ThisTypeIndex, CC,
|
||||
FunctionOptions::None, ArgTypeIndices.size(),
|
||||
ArgListIndex, ThisAdjustment);
|
||||
|
@ -1612,6 +1613,9 @@ static MethodOptions translateMethodOptionFlags(const DISubprogram *SP) {
|
|||
|
||||
static MethodKind translateMethodKindFlags(const DISubprogram *SP,
|
||||
bool Introduced) {
|
||||
if (SP->getFlags() & DINode::FlagStaticMember)
|
||||
return MethodKind::Static;
|
||||
|
||||
switch (SP->getVirtuality()) {
|
||||
case dwarf::DW_VIRTUALITY_none:
|
||||
break;
|
||||
|
@ -1624,8 +1628,6 @@ static MethodKind translateMethodKindFlags(const DISubprogram *SP,
|
|||
llvm_unreachable("unhandled virtuality case");
|
||||
}
|
||||
|
||||
// FIXME: Get Clang to mark DISubprogram as static and do something with it.
|
||||
|
||||
return MethodKind::Vanilla;
|
||||
}
|
||||
|
||||
|
|
|
@ -287,7 +287,8 @@ class LLVM_LIBRARY_VISIBILITY CodeViewDebug : public DebugHandlerBase {
|
|||
codeview::TypeIndex lowerTypeVFTableShape(const DIDerivedType *Ty);
|
||||
codeview::TypeIndex lowerTypeMemberFunction(const DISubroutineType *Ty,
|
||||
const DIType *ClassTy,
|
||||
int ThisAdjustment);
|
||||
int ThisAdjustment,
|
||||
bool IsStaticMethod);
|
||||
codeview::TypeIndex lowerTypeEnum(const DICompositeType *Ty);
|
||||
codeview::TypeIndex lowerTypeClass(const DICompositeType *Ty);
|
||||
codeview::TypeIndex lowerTypeUnion(const DICompositeType *Ty);
|
||||
|
|
|
@ -0,0 +1,103 @@
|
|||
; RUN: llc < %s -filetype=obj | llvm-readobj - -codeview | FileCheck %s
|
||||
|
||||
; Check for the appropriate MethodKind below.
|
||||
|
||||
; C++ source used to generate IR:
|
||||
; $ cat t.cpp
|
||||
; struct A {
|
||||
; static void f();
|
||||
; void g();
|
||||
; static void h(int x, int y);
|
||||
; };
|
||||
; A *p = new A;
|
||||
; $ clang t.cpp -S -emit-llvm -g -gcodeview -o t.ll
|
||||
|
||||
; CHECK: OneMethod {
|
||||
; CHECK-NEXT: TypeLeafKind: LF_ONEMETHOD (0x1511)
|
||||
; CHECK-NEXT: AccessSpecifier: Public (0x3)
|
||||
; CHECK-NEXT: MethodKind: Static (0x2)
|
||||
; CHECK-NEXT: Type: void A::() ({{.*}})
|
||||
; CHECK-NEXT: Name: f
|
||||
; CHECK-NEXT: }
|
||||
; CHECK-NEXT: OneMethod {
|
||||
; CHECK-NEXT: TypeLeafKind: LF_ONEMETHOD (0x1511)
|
||||
; CHECK-NEXT: AccessSpecifier: Public (0x3)
|
||||
; CHECK-NEXT: Type: void A::() ({{.*}})
|
||||
; CHECK-NEXT: Name: g
|
||||
; CHECK-NEXT: }
|
||||
; CHECK-NEXT: OneMethod {
|
||||
; CHECK-NEXT: TypeLeafKind: LF_ONEMETHOD (0x1511)
|
||||
; CHECK-NEXT: AccessSpecifier: Public (0x3)
|
||||
; CHECK-NEXT: MethodKind: Static (0x2)
|
||||
; CHECK-NEXT: Type: void A::(int, int) ({{.*}})
|
||||
; CHECK-NEXT: Name: h
|
||||
; CHECK-NEXT: }
|
||||
|
||||
; ModuleID = 't.cpp'
|
||||
source_filename = "t.cpp"
|
||||
target datalayout = "e-m:x-p:32:32-i64:64-f80:32-n8:16:32-a:0:32-S32"
|
||||
target triple = "i686-pc-windows-msvc19.0.24215"
|
||||
|
||||
%struct.A = type { i8 }
|
||||
|
||||
@"\01?p@@3PAUA@@A" = global %struct.A* null, align 4, !dbg !0
|
||||
@llvm.global_ctors = appending global [1 x { i32, void ()*, i8* }] [{ i32, void ()*, i8* } { i32 65535, void ()* @_GLOBAL__sub_I_t.cpp, i8* null }]
|
||||
|
||||
; Function Attrs: noinline
|
||||
define internal void @"\01??__Ep@@YAXXZ"() #0 !dbg !25 {
|
||||
entry:
|
||||
%call = call i8* @"\01??2@YAPAXI@Z"(i32 1) #2, !dbg !26
|
||||
%0 = bitcast i8* %call to %struct.A*, !dbg !26
|
||||
store %struct.A* %0, %struct.A** @"\01?p@@3PAUA@@A", align 4, !dbg !26
|
||||
ret void, !dbg !27
|
||||
}
|
||||
|
||||
; Function Attrs: nobuiltin
|
||||
declare noalias i8* @"\01??2@YAPAXI@Z"(i32) #1
|
||||
|
||||
; Function Attrs: noinline
|
||||
define internal void @_GLOBAL__sub_I_t.cpp() #0 !dbg !28 {
|
||||
entry:
|
||||
call void @"\01??__Ep@@YAXXZ"(), !dbg !30
|
||||
ret void
|
||||
}
|
||||
|
||||
attributes #0 = { noinline "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf" "no-infs-fp-math"="false" "no-jump-tables"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="pentium4" "target-features"="+fxsr,+mmx,+sse,+sse2,+x87" "unsafe-fp-math"="false" "use-soft-float"="false" }
|
||||
attributes #1 = { nobuiltin "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="pentium4" "target-features"="+fxsr,+mmx,+sse,+sse2,+x87" "unsafe-fp-math"="false" "use-soft-float"="false" }
|
||||
attributes #2 = { builtin }
|
||||
|
||||
!llvm.dbg.cu = !{!2}
|
||||
!llvm.module.flags = !{!20, !21, !22, !23}
|
||||
!llvm.ident = !{!24}
|
||||
|
||||
!0 = !DIGlobalVariableExpression(var: !1, expr: !DIExpression())
|
||||
!1 = distinct !DIGlobalVariable(name: "p", linkageName: "\01?p@@3PAUA@@A", scope: !2, file: !3, line: 7, type: !6, isLocal: false, isDefinition: true)
|
||||
!2 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !3, producer: "clang version 6.0.0 ", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, enums: !4, globals: !5)
|
||||
!3 = !DIFile(filename: "t.cpp", directory: "D:\5Csrc\5Ct", checksumkind: CSK_MD5, checksum: "168f4e5caded7d526f64a37783785c64")
|
||||
!4 = !{}
|
||||
!5 = !{!0}
|
||||
!6 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !7, size: 32)
|
||||
!7 = distinct !DICompositeType(tag: DW_TAG_structure_type, name: "A", file: !3, line: 1, size: 8, elements: !8, identifier: ".?AUA@@")
|
||||
!8 = !{!9, !12, !16}
|
||||
!9 = !DISubprogram(name: "f", linkageName: "\01?f@A@@SAXXZ", scope: !7, file: !3, line: 2, type: !10, isLocal: false, isDefinition: false, scopeLine: 2, flags: DIFlagPrototyped | DIFlagStaticMember, isOptimized: false)
|
||||
!10 = !DISubroutineType(types: !11)
|
||||
!11 = !{null}
|
||||
!12 = !DISubprogram(name: "g", linkageName: "\01?g@A@@QAEXXZ", scope: !7, file: !3, line: 3, type: !13, isLocal: false, isDefinition: false, scopeLine: 3, flags: DIFlagPrototyped, isOptimized: false)
|
||||
!13 = !DISubroutineType(cc: DW_CC_BORLAND_thiscall, types: !14)
|
||||
!14 = !{null, !15}
|
||||
!15 = !DIDerivedType(tag: DW_TAG_pointer_type, baseType: !7, size: 32, flags: DIFlagArtificial | DIFlagObjectPointer)
|
||||
!16 = !DISubprogram(name: "h", linkageName: "\01?h@A@@SAXHH@Z", scope: !7, file: !3, line: 4, type: !17, isLocal: false, isDefinition: false, scopeLine: 4, flags: DIFlagPrototyped | DIFlagStaticMember, isOptimized: false)
|
||||
!17 = !DISubroutineType(types: !18)
|
||||
!18 = !{null, !19, !19}
|
||||
!19 = !DIBasicType(name: "int", size: 32, encoding: DW_ATE_signed)
|
||||
!20 = !{i32 1, !"NumRegisterParameters", i32 0}
|
||||
!21 = !{i32 2, !"CodeView", i32 1}
|
||||
!22 = !{i32 2, !"Debug Info Version", i32 3}
|
||||
!23 = !{i32 1, !"wchar_size", i32 2}
|
||||
!24 = !{!"clang version 6.0.0 "}
|
||||
!25 = distinct !DISubprogram(name: "??__Ep@@YAXXZ", scope: !3, file: !3, line: 7, type: !10, isLocal: true, isDefinition: true, scopeLine: 7, flags: DIFlagPrototyped, isOptimized: false, unit: !2, variables: !4)
|
||||
!26 = !DILocation(line: 7, column: 8, scope: !25)
|
||||
!27 = !DILocation(line: 7, column: 12, scope: !25)
|
||||
!28 = distinct !DISubprogram(linkageName: "_GLOBAL__sub_I_t.cpp", scope: !3, file: !3, type: !29, isLocal: true, isDefinition: true, flags: DIFlagArtificial, isOptimized: false, unit: !2, variables: !4)
|
||||
!29 = !DISubroutineType(types: !4)
|
||||
!30 = !DILocation(line: 0, scope: !28)
|
Loading…
Reference in New Issue