1 ; RUN: opt < %s -aa-pipeline=basic-aa -passes='cgscc(function-attrs,inline)' -S | FileCheck %s 2 ; This test runs the inliner and the function attribute deduction. It ensures 3 ; that when the inliner mutates the call graph it correctly updates the CGSCC 4 ; iteration so that we can compute refined function attributes. In this way it 5 ; is leveraging function attribute computation to observe correct call graph 6 ; updates. 7 8 ; Boring unknown external function call. 9 ; CHECK: declare void @unknown() 10 declare void @unknown() 11 12 ; Sanity check: this should get annotated as readnone. 13 ; CHECK: Function Attrs: nounwind readnone 14 ; CHECK-NEXT: declare void @readnone() 15 declare void @readnone() readnone nounwind 16 17 ; The 'test1_' prefixed functions are designed to trigger forming a new direct 18 ; call in the inlined body of the function. After that, we form a new SCC and 19 ; using that can deduce precise function attrs. 20 21 ; This function should no longer exist. 22 ; CHECK-NOT: @test1_f() 23 define internal void @test1_f(void()* %p) { 24 entry: 25 call void %p() 26 ret void 27 } 28 29 ; This function should have had 'readnone' deduced for its SCC. 30 ; CHECK: Function Attrs: noinline nounwind readnone 31 ; CHECK-NEXT: define void @test1_g() 32 define void @test1_g() noinline { 33 entry: 34 call void @test1_f(void()* @test1_h) 35 ret void 36 } 37 38 ; This function should have had 'readnone' deduced for its SCC. 39 ; CHECK: Function Attrs: noinline nounwind readnone 40 ; CHECK-NEXT: define void @test1_h() 41 define void @test1_h() noinline { 42 entry: 43 call void @test1_g() 44 call void @readnone() 45 ret void 46 } 47 48 49 ; The 'test2_' prefixed functions are designed to trigger forming a new direct 50 ; call due to RAUW-ing the returned value of a called function into the caller. 51 ; This too should form a new SCC which can then be reasoned about to compute 52 ; precise function attrs. 53 54 ; This function should no longer exist. 55 ; CHECK-NOT: @test2_f() 56 define internal void()* @test2_f() { 57 entry: 58 ret void()* @test2_h 59 } 60 61 ; This function should have had 'readnone' deduced for its SCC. 62 ; CHECK: Function Attrs: noinline nounwind readnone 63 ; CHECK-NEXT: define void @test2_g() 64 define void @test2_g() noinline { 65 entry: 66 %p = call void()* @test2_f() 67 call void %p() 68 ret void 69 } 70 71 ; This function should have had 'readnone' deduced for its SCC. 72 ; CHECK: Function Attrs: noinline nounwind readnone 73 ; CHECK-NEXT: define void @test2_h() 74 define void @test2_h() noinline { 75 entry: 76 call void @test2_g() 77 call void @readnone() 78 ret void 79 } 80 81 82 ; The 'test3_' prefixed functions are designed to inline in a way that causes 83 ; call sites to become trivially dead during the middle of inlining callsites of 84 ; a single function to make sure that the inliner does not get confused by this 85 ; pattern. 86 87 ; CHECK-NOT: @test3_maybe_unknown( 88 define internal void @test3_maybe_unknown(i1 %b) { 89 entry: 90 br i1 %b, label %then, label %exit 91 92 then: 93 call void @unknown() 94 br label %exit 95 96 exit: 97 ret void 98 } 99 100 ; CHECK-NOT: @test3_f( 101 define internal i1 @test3_f() { 102 entry: 103 ret i1 false 104 } 105 106 ; CHECK-NOT: @test3_g( 107 define internal i1 @test3_g(i1 %b) { 108 entry: 109 br i1 %b, label %then1, label %if2 110 111 then1: 112 call void @test3_maybe_unknown(i1 true) 113 br label %if2 114 115 if2: 116 %f = call i1 @test3_f() 117 br i1 %f, label %then2, label %exit 118 119 then2: 120 call void @test3_maybe_unknown(i1 true) 121 br label %exit 122 123 exit: 124 ret i1 false 125 } 126 127 ; FIXME: Currently the inliner doesn't successfully mark this as readnone 128 ; because while it simplifies trivially dead CFGs when inlining callees it 129 ; doesn't simplify the caller's trivially dead CFG and so we end with a dead 130 ; block calling @unknown. 131 ; CHECK-NOT: Function Attrs: readnone 132 ; CHECK: define void @test3_h() 133 define void @test3_h() { 134 entry: 135 %g = call i1 @test3_g(i1 false) 136 br i1 %g, label %then, label %exit 137 138 then: 139 call void @test3_maybe_unknown(i1 true) 140 br label %exit 141 142 exit: 143 call void @test3_maybe_unknown(i1 false) 144 ret void 145 } 146 147 148 ; The 'test4_' prefixed functions are designed to trigger forming a new direct 149 ; call in the inlined body of the function similar to 'test1_'. However, after 150 ; that we continue to inline another edge of the graph forcing us to do a more 151 ; interesting call graph update for the new call edge. Eventually, we still 152 ; form a new SCC and should use that can deduce precise function attrs. 153 154 ; This function should have had 'readnone' deduced for its SCC. 155 ; CHECK: Function Attrs: noinline nounwind readnone 156 ; CHECK-NEXT: define void @test4_f1() 157 define void @test4_f1() noinline { 158 entry: 159 call void @test4_h() 160 ret void 161 } 162 163 ; CHECK-NOT: @test4_f2 164 define internal void @test4_f2() { 165 entry: 166 call void @test4_f1() 167 ret void 168 } 169 170 ; CHECK-NOT: @test4_g 171 define internal void @test4_g(void()* %p) { 172 entry: 173 call void %p() 174 ret void 175 } 176 177 ; This function should have had 'readnone' deduced for its SCC. 178 ; CHECK: Function Attrs: noinline nounwind readnone 179 ; CHECK-NEXT: define void @test4_h() 180 define void @test4_h() noinline { 181 entry: 182 call void @test4_g(void()* @test4_f2) 183 ret void 184 } 185