Home | History | Annotate | Download | only in X86
      1 ; RUN: llc < %s -march=x86-64 -mtriple=x86_64-unknown-linux-gnu -asm-verbose=false | FileCheck %s
      2 
      3 ; These tests check for loop branching structure, and that the loop align
      4 ; directive is placed in the expected place.
      5 
      6 ; CodeGen should insert a branch into the middle of the loop in
      7 ; order to avoid a branch within the loop.
      8 
      9 ; CHECK-LABEL: simple:
     10 ;      CHECK:   jmp   .LBB0_1
     11 ; CHECK-NEXT:   align
     12 ; CHECK-NEXT: .LBB0_2:
     13 ; CHECK-NEXT:   callq loop_latch
     14 ; CHECK-NEXT: .LBB0_1:
     15 ; CHECK-NEXT:   callq loop_header
     16 
     17 define void @simple() nounwind {
     18 entry:
     19   br label %loop
     20 
     21 loop:
     22   call void @loop_header()
     23   %t0 = tail call i32 @get()
     24   %t1 = icmp slt i32 %t0, 0
     25   br i1 %t1, label %done, label %bb
     26 
     27 bb:
     28   call void @loop_latch()
     29   br label %loop
     30 
     31 done:
     32   call void @exit()
     33   ret void
     34 }
     35 
     36 ; CodeGen should move block_a to the top of the loop so that it
     37 ; falls through into the loop, avoiding a branch within the loop.
     38 
     39 ; CHECK-LABEL: slightly_more_involved:
     40 ;      CHECK:   jmp .LBB1_1
     41 ; CHECK-NEXT:   align
     42 ; CHECK-NEXT: .LBB1_4:
     43 ; CHECK-NEXT:   callq bar99
     44 ; CHECK-NEXT: .LBB1_1:
     45 ; CHECK-NEXT:   callq body
     46 
     47 define void @slightly_more_involved() nounwind {
     48 entry:
     49   br label %loop
     50 
     51 loop:
     52   call void @body()
     53   %t0 = call i32 @get()
     54   %t1 = icmp slt i32 %t0, 2
     55   br i1 %t1, label %block_a, label %bb
     56 
     57 bb:
     58   %t2 = call i32 @get()
     59   %t3 = icmp slt i32 %t2, 99
     60   br i1 %t3, label %exit, label %loop
     61 
     62 block_a:
     63   call void @bar99()
     64   br label %loop
     65 
     66 exit:
     67   call void @exit()
     68   ret void
     69 }
     70 
     71 ; Same as slightly_more_involved, but block_a is now a CFG diamond with
     72 ; fallthrough edges which should be preserved.
     73 ; "callq block_a_merge_func" is tail duped.
     74 
     75 ; CHECK-LABEL: yet_more_involved:
     76 ;      CHECK:   jmp .LBB2_1
     77 ; CHECK-NEXT:   align
     78 ; CHECK-NEXT: .LBB2_5:
     79 ; CHECK-NEXT:   callq block_a_true_func
     80 ; CHECK-NEXT:   callq block_a_merge_func
     81 ; CHECK-NEXT: .LBB2_1:
     82 ; CHECK-NEXT:   callq body
     83 ;
     84 ; LBB2_4
     85 ;      CHECK:   callq bar99
     86 ; CHECK-NEXT:   callq get
     87 ; CHECK-NEXT:   cmpl $2999, %eax
     88 ; CHECK-NEXT:   jle .LBB2_5
     89 ; CHECK-NEXT:   callq block_a_false_func
     90 ; CHECK-NEXT:   callq block_a_merge_func
     91 ; CHECK-NEXT:   jmp .LBB2_1
     92 
     93 define void @yet_more_involved() nounwind {
     94 entry:
     95   br label %loop
     96 
     97 loop:
     98   call void @body()
     99   %t0 = call i32 @get()
    100   %t1 = icmp slt i32 %t0, 2
    101   br i1 %t1, label %block_a, label %bb
    102 
    103 bb:
    104   %t2 = call i32 @get()
    105   %t3 = icmp slt i32 %t2, 99
    106   br i1 %t3, label %exit, label %loop
    107 
    108 block_a:
    109   call void @bar99()
    110   %z0 = call i32 @get()
    111   %z1 = icmp slt i32 %z0, 3000
    112   br i1 %z1, label %block_a_true, label %block_a_false
    113 
    114 block_a_true:
    115   call void @block_a_true_func()
    116   br label %block_a_merge
    117 
    118 block_a_false:
    119   call void @block_a_false_func()
    120   br label %block_a_merge
    121 
    122 block_a_merge:
    123   call void @block_a_merge_func()
    124   br label %loop
    125 
    126 exit:
    127   call void @exit()
    128   ret void
    129 }
    130 
    131 ; CodeGen should move the CFG islands that are part of the loop but don't
    132 ; conveniently fit anywhere so that they are at least contiguous with the
    133 ; loop.
    134 
    135 ; CHECK-LABEL: cfg_islands:
    136 ;      CHECK:   jmp     .LBB3_1
    137 ; CHECK-NEXT:   align
    138 ; CHECK-NEXT: .LBB3_7:
    139 ; CHECK-NEXT:   callq   bar100
    140 ; CHECK-NEXT: .LBB3_1:
    141 ; CHECK-NEXT:   callq   loop_header
    142 ;      CHECK:   jl .LBB3_7
    143 ;      CHECK:   jge .LBB3_3
    144 ; CHECK-NEXT:   callq   bar101
    145 ; CHECK-NEXT:   jmp     .LBB3_1
    146 ; CHECK-NEXT:   align
    147 ; CHECK-NEXT: .LBB3_3:
    148 ;      CHECK:   jge .LBB3_4
    149 ; CHECK-NEXT:   callq   bar102
    150 ; CHECK-NEXT:   jmp     .LBB3_1
    151 ; CHECK-NEXT: .LBB3_4:
    152 ;      CHECK:   jl .LBB3_6
    153 ; CHECK-NEXT:   callq   loop_latch
    154 ; CHECK-NEXT:   jmp     .LBB3_1
    155 ; CHECK-NEXT: .LBB3_6:
    156 
    157 define void @cfg_islands() nounwind {
    158 entry:
    159   br label %loop
    160 
    161 loop:
    162   call void @loop_header()
    163   %t0 = call i32 @get()
    164   %t1 = icmp slt i32 %t0, 100
    165   br i1 %t1, label %block100, label %bb
    166 
    167 bb:
    168   %t2 = call i32 @get()
    169   %t3 = icmp slt i32 %t2, 101
    170   br i1 %t3, label %block101, label %bb1
    171 
    172 bb1:
    173   %t4 = call i32 @get()
    174   %t5 = icmp slt i32 %t4, 102
    175   br i1 %t5, label %block102, label %bb2
    176 
    177 bb2:
    178   %t6 = call i32 @get()
    179   %t7 = icmp slt i32 %t6, 103
    180   br i1 %t7, label %exit, label %bb3
    181 
    182 bb3:
    183   call void @loop_latch()
    184   br label %loop
    185 
    186 exit:
    187   call void @exit()
    188   ret void
    189 
    190 block100:
    191   call void @bar100()
    192   br label %loop
    193 
    194 block101:
    195   call void @bar101()
    196   br label %loop
    197 
    198 block102:
    199   call void @bar102()
    200   br label %loop
    201 }
    202 
    203 ; CHECK-LABEL: check_minsize:
    204 ;      CHECK:   jmp   .LBB4_1
    205 ; CHECK-NOT:   align
    206 ; CHECK-NEXT: .LBB4_2:
    207 ; CHECK-NEXT:   callq loop_latch
    208 ; CHECK-NEXT: .LBB4_1:
    209 ; CHECK-NEXT:   callq loop_header
    210 
    211 
    212 define void @check_minsize() minsize nounwind {
    213 entry:
    214   br label %loop
    215 
    216 loop:
    217   call void @loop_header()
    218   %t0 = tail call i32 @get()
    219   %t1 = icmp slt i32 %t0, 0
    220   br i1 %t1, label %done, label %bb
    221 
    222 bb:
    223   call void @loop_latch()
    224   br label %loop
    225 
    226 done:
    227   call void @exit()
    228   ret void
    229 }
    230 
    231 declare void @bar99() nounwind
    232 declare void @bar100() nounwind
    233 declare void @bar101() nounwind
    234 declare void @bar102() nounwind
    235 declare void @body() nounwind
    236 declare void @exit() nounwind
    237 declare void @loop_header() nounwind
    238 declare void @loop_latch() nounwind
    239 declare i32 @get() nounwind
    240 declare void @block_a_true_func() nounwind
    241 declare void @block_a_false_func() nounwind
    242 declare void @block_a_merge_func() nounwind
    243