Home | History | Annotate | Download | only in Inline
      1 ; Test for a subtle bug when computing analyses during inlining and mutating
      2 ; the SCC structure. Without care, this can fail to invalidate analyses.
      3 ;
      4 ; RUN: opt < %s -passes='cgscc(inline,function(verify<domtree>))' -debug-pass-manager -S 2>&1 | FileCheck %s
      5 
      6 ; First we check that the passes run in the way we expect. Otherwise this test
      7 ; may stop testing anything.
      8 ;
      9 ; CHECK-LABEL: Starting llvm::Module pass manager run.
     10 ; CHECK: Running pass: InlinerPass on (test1_f, test1_g, test1_h)
     11 ; CHECK: Running analysis: FunctionAnalysisManagerCGSCCProxy on (test1_f, test1_g, test1_h)
     12 ; CHECK: Running analysis: DominatorTreeAnalysis on test1_f
     13 ; CHECK: Running analysis: DominatorTreeAnalysis on test1_g
     14 ; CHECK: Invalidating all non-preserved analyses for: (test1_f)
     15 ; CHECK: Invalidating all non-preserved analyses for: test1_f
     16 ; CHECK: Invalidating analysis: DominatorTreeAnalysis on test1_f
     17 ; CHECK: Invalidating analysis: LoopAnalysis on test1_f
     18 ; CHECK: Invalidating analysis: BranchProbabilityAnalysis on test1_f
     19 ; CHECK: Invalidating analysis: BlockFrequencyAnalysis on test1_f
     20 ; CHECK: Invalidating all non-preserved analyses for: (test1_g, test1_h)
     21 ; CHECK: Invalidating all non-preserved analyses for: test1_g
     22 ; CHECK: Invalidating analysis: DominatorTreeAnalysis on test1_g
     23 ; CHECK: Invalidating analysis: LoopAnalysis on test1_g
     24 ; CHECK: Invalidating analysis: BranchProbabilityAnalysis on test1_g
     25 ; CHECK: Invalidating analysis: BlockFrequencyAnalysis on test1_g
     26 ; CHECK: Invalidating all non-preserved analyses for: test1_h
     27 ; CHECK: Invalidating analysis: DominatorTreeAnalysis on test1_h
     28 ; CHECK: Invalidating analysis: LoopAnalysis on test1_h
     29 ; CHECK: Invalidating analysis: BranchProbabilityAnalysis on test1_h
     30 ; CHECK: Invalidating analysis: BlockFrequencyAnalysis on test1_h
     31 ; CHECK-NOT: Invalidating analysis:
     32 ; CHECK: Starting llvm::Function pass manager run.
     33 ; CHECK-NEXT: Running pass: DominatorTreeVerifierPass on test1_g
     34 ; CHECK-NEXT: Running analysis: DominatorTreeAnalysis on test1_g
     35 ; CHECK-NEXT: Finished llvm::Function pass manager run.
     36 ; CHECK-NEXT: Starting llvm::Function pass manager run.
     37 ; CHECK-NEXT: Running pass: DominatorTreeVerifierPass on test1_h
     38 ; CHECK-NEXT: Running analysis: DominatorTreeAnalysis on test1_h
     39 ; CHECK-NEXT: Finished llvm::Function pass manager run.
     40 ; CHECK-NOT: Invalidating analysis:
     41 ; CHECK: Running pass: DominatorTreeVerifierPass on test1_f
     42 ; CHECK-NEXT: Running analysis: DominatorTreeAnalysis on test1_f
     43 
     44 ; An external function used to control branches.
     45 declare i1 @flag()
     46 ; CHECK-LABEL: declare i1 @flag()
     47 
     48 ; The utility function with interesting control flow that gets inlined below to
     49 ; perturb the dominator tree.
     50 define internal void @callee() {
     51 entry:
     52   %ptr = alloca i8
     53   %flag = call i1 @flag()
     54   br i1 %flag, label %then, label %else
     55 
     56 then:
     57   store volatile i8 42, i8* %ptr
     58   br label %return
     59 
     60 else:
     61   store volatile i8 -42, i8* %ptr
     62   br label %return
     63 
     64 return:
     65   ret void
     66 }
     67 
     68 ; The 'test1_' prefixed functions work to carefully test that incrementally
     69 ; reducing an SCC in the inliner cannot accidentially leave stale function
     70 ; analysis results due to failing to invalidate them for all the functions.
     71 
     72 ; The inliner visits this last function. It can't actually break any cycles
     73 ; here, but because we visit this function we compute fresh analyses for it.
     74 ; These analyses are then invalidated when we inline callee disrupting the
     75 ; CFG, and it is important that they be freed.
     76 define void @test1_h() {
     77 ; CHECK-LABEL: define void @test1_h()
     78 entry:
     79   call void @test1_g()
     80 ; CHECK: call void @test1_g()
     81 
     82   ; Pull interesting CFG into this function.
     83   call void @callee()
     84 ; CHECK-NOT: call void @callee()
     85 
     86   ret void
     87 ; CHECK: ret void
     88 }
     89 
     90 ; We visit this function second and here we inline the edge to 'test1_f'
     91 ; separating it into its own SCC. The current SCC is now just 'test1_g' and
     92 ; 'test1_h'.
     93 define void @test1_g() {
     94 ; CHECK-LABEL: define void @test1_g()
     95 entry:
     96   ; This edge gets inlined away.
     97   call void @test1_f()
     98 ; CHECK-NOT: call void @test1_f()
     99 ; CHECK: call void @test1_g()
    100 
    101   ; We force this edge to survive inlining.
    102   call void @test1_h() noinline
    103 ; CHECK: call void @test1_h()
    104 
    105   ; Pull interesting CFG into this function.
    106   call void @callee()
    107 ; CHECK-NOT: call void @callee()
    108 
    109   ret void
    110 ; CHECK: ret void
    111 }
    112 
    113 ; We visit this function first in the inliner, and while we inline callee
    114 ; perturbing the CFG, we don't inline anything else and the SCC structure
    115 ; remains in tact.
    116 define void @test1_f() {
    117 ; CHECK-LABEL: define void @test1_f()
    118 entry:
    119   ; We force this edge to survive inlining.
    120   call void @test1_g() noinline
    121 ; CHECK: call void @test1_g()
    122 
    123   ; Pull interesting CFG into this function.
    124   call void @callee()
    125 ; CHECK-NOT: call void @callee()
    126 
    127   ret void
    128 ; CHECK: ret void
    129 }
    130 
    131 ; The 'test2_' prefixed code works to carefully trigger forming an SCC with
    132 ; a dominator tree for one of the functions but not the other and without even
    133 ; a function analysis manager proxy for the SCC that things get merged into.
    134 ; Without proper handling when updating the call graph this will find a stale
    135 ; dominator tree.
    136 
    137 @test2_global = external global i32, align 4
    138 
    139 define void @test2_hoge(i1 (i32*)* %arg) {
    140 ; CHECK-LABEL: define void @test2_hoge(
    141 bb:
    142   %tmp2 = call zeroext i1 %arg(i32* @test2_global)
    143 ; CHECK: call zeroext i1 %arg(
    144   br label %bb3
    145 
    146 bb3:
    147   %tmp5 = call zeroext i1 %arg(i32* @test2_global)
    148 ; CHECK: call zeroext i1 %arg(
    149   br i1 %tmp5, label %bb3, label %bb6
    150 
    151 bb6:
    152   ret void
    153 }
    154 
    155 define zeroext i1 @test2_widget(i32* %arg) {
    156 ; CHECK-LABEL: define zeroext i1 @test2_widget(
    157 bb:
    158   %tmp1 = alloca i8, align 1
    159   %tmp2 = alloca i32, align 4
    160   call void @test2_quux()
    161 ; CHECK-NOT:     call
    162 ;
    163 ; CHECK:         call zeroext i1 @test2_widget(i32* @test2_global)
    164 ; CHECK-NEXT:    br label %[[NEW_BB:.*]]
    165 ;
    166 ; CHECK:       [[NEW_BB]]:
    167 ; CHECK-NEXT:    call zeroext i1 @test2_widget(i32* @test2_global)
    168 ;
    169 ; CHECK:       {{.*}}:
    170 
    171   call void @test2_hoge.1(i32* %arg)
    172 ; CHECK-NEXT:    call void @test2_hoge.1(
    173 
    174   %tmp4 = call zeroext i1 @test2_barney(i32* %tmp2)
    175   %tmp5 = zext i1 %tmp4 to i32
    176   store i32 %tmp5, i32* %tmp2, align 4
    177   %tmp6 = call zeroext i1 @test2_barney(i32* null)
    178   call void @test2_ham(i8* %tmp1)
    179 ; CHECK:         call void @test2_ham(
    180 
    181   call void @test2_quux()
    182 ; CHECK-NOT:     call
    183 ;
    184 ; CHECK:         call zeroext i1 @test2_widget(i32* @test2_global)
    185 ; CHECK-NEXT:    br label %[[NEW_BB:.*]]
    186 ;
    187 ; CHECK:       [[NEW_BB]]:
    188 ; CHECK-NEXT:    call zeroext i1 @test2_widget(i32* @test2_global)
    189 ;
    190 ; CHECK:       {{.*}}:
    191   ret i1 true
    192 ; CHECK-NEXT:    ret i1 true
    193 }
    194 
    195 define internal void @test2_quux() {
    196 ; CHECK-NOT: @test2_quux
    197 bb:
    198   call void @test2_hoge(i1 (i32*)* @test2_widget)
    199   ret void
    200 }
    201 
    202 declare void @test2_hoge.1(i32*)
    203 
    204 declare zeroext i1 @test2_barney(i32*)
    205 
    206 declare void @test2_ham(i8*)
    207