| // Tests that we assign appropriate identifiers to indirect calls and targets |
| // specifically for C++ class and instance methods. |
| |
| // RUN: %clang_cc1 -triple x86_64-unknown-linux -fexperimental-call-graph-section \ |
| // RUN: -emit-llvm -o %t %s |
| // RUN: FileCheck --check-prefix=FT %s < %t |
| // RUN: FileCheck --check-prefix=CST %s < %t |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // Class definitions (check for indirect target metadata) |
| |
| class Cls1 { |
| public: |
| // FT-LABEL: define {{.*}} ptr @_ZN4Cls18receiverEPcPf( |
| // FT-SAME: {{.*}} !type [[F_TCLS1RECEIVER:![0-9]+]] |
| static int *receiver(char *a, float *b) { return 0; } |
| }; |
| |
| class Cls2 { |
| public: |
| int *(*fp)(char *, float *); |
| |
| // FT-LABEL: define {{.*}} i32 @_ZN4Cls22f1Ecfd( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F1:![0-9]+]] |
| int f1(char a, float b, double c) { return 0; } |
| |
| // FT-LABEL: define {{.*}} ptr @_ZN4Cls22f2EPcPfPd( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F2:![0-9]+]] |
| int *f2(char *a, float *b, double *c) { return 0; } |
| |
| // FT-LABEL: define {{.*}} void @_ZN4Cls22f3E4Cls1( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F3F4:![0-9]+]] |
| void f3(Cls1 a) {} |
| |
| // FT-LABEL: define {{.*}} void @_ZN4Cls22f4E4Cls1( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F3F4]] |
| void f4(const Cls1 a) {} |
| |
| // FT-LABEL: define {{.*}} void @_ZN4Cls22f5EP4Cls1( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F5:![0-9]+]] |
| void f5(Cls1 *a) {} |
| |
| // FT-LABEL: define {{.*}} void @_ZN4Cls22f6EPK4Cls1( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F6:![0-9]+]] |
| void f6(const Cls1 *a) {} |
| |
| // FT-LABEL: define {{.*}} void @_ZN4Cls22f7ER4Cls1( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F7:![0-9]+]] |
| void f7(Cls1 &a) {} |
| |
| // FT-LABEL: define {{.*}} void @_ZN4Cls22f8ERK4Cls1( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F8:![0-9]+]] |
| void f8(const Cls1 &a) {} |
| |
| // FT-LABEL: define {{.*}} void @_ZNK4Cls22f9Ev( |
| // FT-SAME: {{.*}} !type [[F_TCLS2F9:![0-9]+]] |
| void f9() const {} |
| }; |
| |
| // FT: [[F_TCLS1RECEIVER]] = !{i64 0, !"_ZTSFPiPcPfE.generalized"} |
| // FT: [[F_TCLS2F1]] = !{i64 0, !"_ZTSFicfdE.generalized"} |
| // FT: [[F_TCLS2F2]] = !{i64 0, !"_ZTSFPiPcPfPdE.generalized"} |
| // FT: [[F_TCLS2F3F4]] = !{i64 0, !"_ZTSFv4Cls1E.generalized"} |
| // FT: [[F_TCLS2F5]] = !{i64 0, !"_ZTSFvP4Cls1E.generalized"} |
| // FT: [[F_TCLS2F6]] = !{i64 0, !"_ZTSFvPK4Cls1E.generalized"} |
| // FT: [[F_TCLS2F7]] = !{i64 0, !"_ZTSFvR4Cls1E.generalized"} |
| // FT: [[F_TCLS2F8]] = !{i64 0, !"_ZTSFvRK4Cls1E.generalized"} |
| // FT: [[F_TCLS2F9]] = !{i64 0, !"_ZTSKFvvE.generalized"} |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // Callsites (check for indirect callsites' callee_type metadata ) |
| |
| // CST-LABEL: define {{.*}} @_Z3foov |
| void foo() { |
| Cls2 ObjCls2; |
| ObjCls2.fp = &Cls1::receiver; |
| |
| // CST: call noundef ptr %{{.*}}, !callee_type [[F_TCLS1RECEIVER_CT:![0-9]+]] |
| ObjCls2.fp(0, 0); |
| |
| auto fp_f1 = &Cls2::f1; |
| auto fp_f2 = &Cls2::f2; |
| auto fp_f3 = &Cls2::f3; |
| auto fp_f4 = &Cls2::f4; |
| auto fp_f5 = &Cls2::f5; |
| auto fp_f6 = &Cls2::f6; |
| auto fp_f7 = &Cls2::f7; |
| auto fp_f8 = &Cls2::f8; |
| auto fp_f9 = &Cls2::f9; |
| |
| Cls2 *ObjCls2Ptr = &ObjCls2; |
| Cls1 Cls1Param; |
| |
| // CST: call noundef i32 %{{.*}}, !callee_type [[F_TCLS2F1_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f1)(0, 0, 0); |
| |
| // CST: call noundef ptr %{{.*}}, !callee_type [[F_TCLS2F2_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f2)(0, 0, 0); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F3F4_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f3)(Cls1Param); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F3F4_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f4)(Cls1Param); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F5_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f5)(&Cls1Param); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F6_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f6)(&Cls1Param); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F7_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f7)(Cls1Param); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F8_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f8)(Cls1Param); |
| |
| // CST: call void %{{.*}}, !callee_type [[F_TCLS2F9_CT:![0-9]+]] |
| (ObjCls2Ptr->*fp_f9)(); |
| } |
| |
| // CST: [[F_TCLS1RECEIVER_CT]] = !{[[F_TCLS1RECEIVER:![0-9]+]]} |
| // CST: [[F_TCLS1RECEIVER]] = !{i64 0, !"_ZTSFPiPcPfE.generalized"} |
| |
| // CST: [[F_TCLS2F1_CT]] = !{[[F_TCLS2F1:![0-9]+]]} |
| // CST: [[F_TCLS2F1]] = !{i64 0, !"_ZTSFicfdE.generalized"} |
| |
| // CST: [[F_TCLS2F2_CT]] = !{[[F_TCLS2F2:![0-9]+]]} |
| // CST: [[F_TCLS2F2]] = !{i64 0, !"_ZTSFPiPcPfPdE.generalized"} |
| |
| // CST: [[F_TCLS2F3F4_CT]] = !{[[F_TCLS2F3F4:![0-9]+]]} |
| // CST: [[F_TCLS2F3F4]] = !{i64 0, !"_ZTSFv4Cls1E.generalized"} |
| |
| // CST: [[F_TCLS2F5_CT]] = !{[[F_TCLS2F5:![0-9]+]]} |
| // CST: [[F_TCLS2F5]] = !{i64 0, !"_ZTSFvP4Cls1E.generalized"} |
| |
| // CST: [[F_TCLS2F6_CT]] = !{[[F_TCLS2F6:![0-9]+]]} |
| // CST: [[F_TCLS2F6]] = !{i64 0, !"_ZTSFvPK4Cls1E.generalized"} |
| |
| // CST: [[F_TCLS2F7_CT]] = !{[[F_TCLS2F7:![0-9]+]]} |
| // CST: [[F_TCLS2F7]] = !{i64 0, !"_ZTSFvR4Cls1E.generalized"} |
| |
| // CST: [[F_TCLS2F8_CT]] = !{[[F_TCLS2F8:![0-9]+]]} |
| // CST: [[F_TCLS2F8]] = !{i64 0, !"_ZTSFvRK4Cls1E.generalized"} |
| |
| // CST: [[F_TCLS2F9_CT]] = !{[[F_TCLS2F9:![0-9]+]]} |
| // CST: [[F_TCLS2F9]] = !{i64 0, !"_ZTSKFvvE.generalized"} |