blob: 86ed3ee2337a7cee6dc9b0ae6a4f1c40c45cccef [file] [log] [blame]
// 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"}