[rtsan] Add fork/execve interceptors (#117198)
[llvm-project.git] / llvm / test / Transforms / PGOProfile / vtable_profile.ll
blobaae1e2d8b4e491e00fa8b78b7b52f48a961c29e2
1 ; RUN: opt < %s -passes=pgo-instr-gen -enable-vtable-value-profiling -S 2>&1 | FileCheck %s --check-prefix=GEN --implicit-check-not="VTable value profiling is presently not supported"
2 ; RUN: opt < %s -passes=pgo-instr-gen,instrprof -enable-vtable-value-profiling -S 2>&1 | FileCheck %s --check-prefix=LOWER --implicit-check-not="VTable value profiling is presently not supported"
4 source_filename = "vtable_local.ll"
5 target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128"
6 target triple = "x86_64-unknown-linux-gnu"
8 ; The test IR is generated based on the following C++ program.
9 ; Base1 has external linkage and Base2 has local linkage.
10 ; class Derived uses multiple inheritance so its virtual table
11 ; global variable contains two vtables. func1 is loaded from
12 ; the vtable compatible with class Base1, and func2 is loaded
13 ; from the vtable compatible with class Base2.
15 ; class Base1 {
16 ; public:
17 ;   virtual int func1(int a) ;
18 ; };
20 ; namespace {
21 ; class Base2 {
22 ; public:
23 ;   __attribute__((noinline)) virtual int func2(int a) {
24 ;     return a;
25 ;   }
26 ; };
27 ; }
29 ; class Derived : public Base1, public Base2 {
30 ; public:
31 ;   Derived(int c) : v(c) {}
32 ; private:
33 ;   int v;
34 ; };
36 ; Derived* createType();
38 ; int func(int a) {
39 ;   Derived* d = createType();
40 ;   return d->func2(a) + d->func1(a);
41 ; }
43 target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-i128:128-f80:128-n8:16:32:64-S128"
44 target triple = "x86_64-unknown-linux-gnu"
46 @_ZTV7Derived = constant { [3 x ptr], [3 x ptr] } { [3 x ptr] [ptr null, ptr null, ptr @_ZN5Base15func1Ei], [3 x ptr] [ptr inttoptr (i64 -8 to ptr), ptr null, ptr @_ZN12_GLOBAL__N_15Base25func2Ei] }, !type !0, !type !3, !type !6, !type !8, !type !10
47 @_ZTV5Base1 = available_externally constant { [3 x ptr] } { [3 x ptr] [ptr null, ptr null, ptr @_ZN5Base15func1Ei] }, !type !0
48 @_ZTVN12_GLOBAL__N_15Base2E = internal constant { [3 x ptr] } { [3 x ptr] [ptr null, ptr null, ptr @_ZN12_GLOBAL__N_15Base25func2Ei] }, !type !11, !type !8; !vcall_visibility !12
49 @llvm.compiler.used = appending global [1 x ptr] [ptr @_ZTV5Base1], section "llvm.metadata"
51 ; GEN: __llvm_profile_raw_version = comdat any
52 ; GEN: __llvm_profile_raw_version = hidden constant i64 72057594037927946, comdat
53 ; GEN: __profn__Z4funci = private constant [8 x i8] c"_Z4funci"
55 ; LOWER: $__profvt__ZTV7Derived = comdat nodeduplicate
56 ; LOWER: $"__profvt_vtable_local.ll;_ZTVN12_GLOBAL__N_15Base2E" = comdat nodeduplicate
57 ; LOWER: @__profvt__ZTV7Derived = global { i64, ptr, i32 } { i64 -4576307468236080025, ptr @_ZTV7Derived, i32 48 }, section "__llvm_prf_vtab", comdat, align 8
58 ; LOWER: @"__profvt_vtable_local.ll;_ZTVN12_GLOBAL__N_15Base2E" = internal global { i64, ptr, i32 } { i64 1419990121885302679, ptr @_ZTVN12_GLOBAL__N_15Base2E, i32 24 }, section "__llvm_prf_vtab", comdat, align 8
59 ; LOWER: @__llvm_prf_vnm = private constant {{.*}}, section "__llvm_prf_vns", align 1
60 ; LOWER: @llvm.used = appending global [5 x ptr] [ptr @__profvt__ZTV7Derived, ptr @"__profvt_vtable_local.ll;_ZTVN12_GLOBAL__N_15Base2E", ptr @__llvm_prf_vnodes, ptr @__llvm_prf_nm, ptr @__llvm_prf_vnm], section "llvm.metadata"
62 define i32 @_Z4funci(i32 %a) {
63 entry:
64   %call = call ptr @_Z10createTypev()
65   %add.ptr = getelementptr inbounds i8, ptr %call, i64 8
66   %vtable = load ptr, ptr %add.ptr
67 ; GEN: [[P1:%[0-9]+]] = ptrtoint ptr %vtable to i64
68 ; GEN: call void @llvm.instrprof.value.profile(ptr @__profn__Z4funci, i64 [[CFGHash:[0-9]+]], i64 [[P1]], i32 2, i32 0)
69 ; LOWER: [[P1:%[0-9]+]] = ptrtoint ptr %vtable to i64
70 ; LOWER: call void @__llvm_profile_instrument_target(i64 [[P1]], ptr @__profd__Z4funci, i32 2)
71   %vfunc1 = load ptr, ptr %vtable
72   %call1 = call i32 %vfunc1(ptr %add.ptr, i32 %a)
73   %vtable2 = load ptr, ptr %call
74 ; GEN: [[P2:%[0-9]+]] = ptrtoint ptr %vtable2 to i64
75 ; GEN: call void @llvm.instrprof.value.profile(ptr @__profn__Z4funci, i64 [[CFGHash]], i64 [[P2]], i32 2, i32 1)
76 ; LOWER: [[P2:%[0-9]+]] = ptrtoint ptr %vtable2 to i64
77 ; LOWER: call void @__llvm_profile_instrument_target(i64 [[P2]], ptr @__profd__Z4funci, i32 3)
78   %vfunc2 = load ptr, ptr %vtable2
79   %call4 = call i32 %vfunc2(ptr %call, i32 %a)
80   %add = add nsw i32 %call1, %call4
81   ret i32 %add
84 declare ptr @_Z10createTypev()
85 declare i32 @_ZN12_GLOBAL__N_15Base25func2Ei(ptr %this, i32 %a)
86 declare i32 @_ZN5Base15func1Ei(ptr, i32)
88 !0 = !{i64 16, !"_ZTS5Base1"}
89 !3 = !{i64 16, !"_ZTS7Derived"}
90 !6 = !{i64 40, !7}
91 !7 = distinct !{}
92 !8 = !{i64 16, !9}
93 !9 = distinct !{}
94 !10 = !{i64 40, !9}
95 !11 = !{i64 16, !7}