Home | History | Annotate | Download | only in SimplifyCFG
      1 ; RUN: opt -S -simplifycfg < %s | FileCheck %s
      2 ; RUN: opt -S -default-data-layout="p:32:32-p1:16:16" -simplifycfg < %s | FileCheck -check-prefix=CHECK -check-prefix=DL %s
      3 
      4 declare void @foo1()
      5 
      6 declare void @foo2()
      7 
      8 define void @test1(i32 %V) {
      9         %C1 = icmp eq i32 %V, 4         ; <i1> [#uses=1]
     10         %C2 = icmp eq i32 %V, 17                ; <i1> [#uses=1]
     11         %CN = or i1 %C1, %C2            ; <i1> [#uses=1]
     12         br i1 %CN, label %T, label %F
     13 T:              ; preds = %0
     14         call void @foo1( )
     15         ret void
     16 F:              ; preds = %0
     17         call void @foo2( )
     18         ret void
     19 ; CHECK-LABEL: @test1(
     20 ; CHECK:  switch i32 %V, label %F [
     21 ; CHECK:    i32 17, label %T
     22 ; CHECK:    i32 4, label %T
     23 ; CHECK:  ]
     24 }
     25 
     26 define void @test1_ptr(i32* %V) {
     27         %C1 = icmp eq i32* %V, inttoptr (i32 4 to i32*)
     28         %C2 = icmp eq i32* %V, inttoptr (i32 17 to i32*)
     29         %CN = or i1 %C1, %C2            ; <i1> [#uses=1]
     30         br i1 %CN, label %T, label %F
     31 T:              ; preds = %0
     32         call void @foo1( )
     33         ret void
     34 F:              ; preds = %0
     35         call void @foo2( )
     36         ret void
     37 ; CHECK-LABEL: @test1_ptr(
     38 ; DL:  %magicptr = ptrtoint i32* %V to i32
     39 ; DL:  switch i32 %magicptr, label %F [
     40 ; DL:    i32 17, label %T
     41 ; DL:    i32 4, label %T
     42 ; DL:  ]
     43 }
     44 
     45 define void @test1_ptr_as1(i32 addrspace(1)* %V) {
     46         %C1 = icmp eq i32 addrspace(1)* %V, inttoptr (i32 4 to i32 addrspace(1)*)
     47         %C2 = icmp eq i32 addrspace(1)* %V, inttoptr (i32 17 to i32 addrspace(1)*)
     48         %CN = or i1 %C1, %C2            ; <i1> [#uses=1]
     49         br i1 %CN, label %T, label %F
     50 T:              ; preds = %0
     51         call void @foo1( )
     52         ret void
     53 F:              ; preds = %0
     54         call void @foo2( )
     55         ret void
     56 ; CHECK-LABEL: @test1_ptr_as1(
     57 ; DL:  %magicptr = ptrtoint i32 addrspace(1)* %V to i16
     58 ; DL:  switch i16 %magicptr, label %F [
     59 ; DL:    i16 17, label %T
     60 ; DL:    i16 4, label %T
     61 ; DL:  ]
     62 }
     63 
     64 define void @test2(i32 %V) {
     65         %C1 = icmp ne i32 %V, 4         ; <i1> [#uses=1]
     66         %C2 = icmp ne i32 %V, 17                ; <i1> [#uses=1]
     67         %CN = and i1 %C1, %C2           ; <i1> [#uses=1]
     68         br i1 %CN, label %T, label %F
     69 T:              ; preds = %0
     70         call void @foo1( )
     71         ret void
     72 F:              ; preds = %0
     73         call void @foo2( )
     74         ret void
     75 ; CHECK-LABEL: @test2(
     76 ; CHECK:  switch i32 %V, label %T [
     77 ; CHECK:    i32 17, label %F
     78 ; CHECK:    i32 4, label %F
     79 ; CHECK:  ]
     80 }
     81 
     82 define void @test3(i32 %V) {
     83         %C1 = icmp eq i32 %V, 4         ; <i1> [#uses=1]
     84         br i1 %C1, label %T, label %N
     85 N:              ; preds = %0
     86         %C2 = icmp eq i32 %V, 17                ; <i1> [#uses=1]
     87         br i1 %C2, label %T, label %F
     88 T:              ; preds = %N, %0
     89         call void @foo1( )
     90         ret void
     91 F:              ; preds = %N
     92         call void @foo2( )
     93         ret void
     94 
     95 ; CHECK-LABEL: @test3(
     96 ; CHECK: switch i32 %V, label %F [
     97 ; CHECK:     i32 4, label %T
     98 ; CHECK:     i32 17, label %T
     99 ; CHECK:   ]
    100 }
    101 
    102 
    103 
    104 define i32 @test4(i8 zeroext %c) nounwind ssp noredzone {
    105 entry:
    106   %cmp = icmp eq i8 %c, 62
    107   br i1 %cmp, label %lor.end, label %lor.lhs.false
    108 
    109 lor.lhs.false:                                    ; preds = %entry
    110   %cmp4 = icmp eq i8 %c, 34
    111   br i1 %cmp4, label %lor.end, label %lor.rhs
    112 
    113 lor.rhs:                                          ; preds = %lor.lhs.false
    114   %cmp8 = icmp eq i8 %c, 92
    115   br label %lor.end
    116 
    117 lor.end:                                          ; preds = %lor.rhs, %lor.lhs.false, %entry
    118   %0 = phi i1 [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp8, %lor.rhs ]
    119   %lor.ext = zext i1 %0 to i32
    120   ret i32 %lor.ext
    121 
    122 ; CHECK-LABEL: @test4(
    123 ; CHECK:  switch i8 %c, label %lor.rhs [
    124 ; CHECK:    i8 62, label %lor.end
    125 ; CHECK:    i8 34, label %lor.end
    126 ; CHECK:    i8 92, label %lor.end
    127 ; CHECK:  ]
    128 }
    129 
    130 define i32 @test5(i8 zeroext %c) nounwind ssp noredzone {
    131 entry:
    132   switch i8 %c, label %lor.rhs [
    133     i8 62, label %lor.end
    134     i8 34, label %lor.end
    135     i8 92, label %lor.end
    136   ]
    137 
    138 lor.rhs:                                          ; preds = %entry
    139   %V = icmp eq i8 %c, 92
    140   br label %lor.end
    141 
    142 lor.end:                                          ; preds = %entry, %entry, %entry, %lor.rhs
    143   %0 = phi i1 [ true, %entry ], [ %V, %lor.rhs ], [ true, %entry ], [ true, %entry ]
    144   %lor.ext = zext i1 %0 to i32
    145   ret i32 %lor.ext
    146 ; CHECK-LABEL: @test5(
    147 ; CHECK:  switch i8 %c, label %lor.rhs [
    148 ; CHECK:    i8 62, label %lor.end
    149 ; CHECK:    i8 34, label %lor.end
    150 ; CHECK:    i8 92, label %lor.end
    151 ; CHECK:  ]
    152 }
    153 
    154 
    155 define i1 @test6({ i32, i32 }* %I) {
    156 entry:
    157         %tmp.1.i = getelementptr { i32, i32 }, { i32, i32 }* %I, i64 0, i32 1         ; <i32*> [#uses=1]
    158         %tmp.2.i = load i32, i32* %tmp.1.i           ; <i32> [#uses=6]
    159         %tmp.2 = icmp eq i32 %tmp.2.i, 14               ; <i1> [#uses=1]
    160         br i1 %tmp.2, label %shortcirc_done.4, label %shortcirc_next.0
    161 shortcirc_next.0:               ; preds = %entry
    162         %tmp.6 = icmp eq i32 %tmp.2.i, 15               ; <i1> [#uses=1]
    163         br i1 %tmp.6, label %shortcirc_done.4, label %shortcirc_next.1
    164 shortcirc_next.1:               ; preds = %shortcirc_next.0
    165         %tmp.11 = icmp eq i32 %tmp.2.i, 16              ; <i1> [#uses=1]
    166         br i1 %tmp.11, label %shortcirc_done.4, label %shortcirc_next.2
    167 shortcirc_next.2:               ; preds = %shortcirc_next.1
    168         %tmp.16 = icmp eq i32 %tmp.2.i, 17              ; <i1> [#uses=1]
    169         br i1 %tmp.16, label %shortcirc_done.4, label %shortcirc_next.3
    170 shortcirc_next.3:               ; preds = %shortcirc_next.2
    171         %tmp.21 = icmp eq i32 %tmp.2.i, 18              ; <i1> [#uses=1]
    172         br i1 %tmp.21, label %shortcirc_done.4, label %shortcirc_next.4
    173 shortcirc_next.4:               ; preds = %shortcirc_next.3
    174         %tmp.26 = icmp eq i32 %tmp.2.i, 19              ; <i1> [#uses=1]
    175         br label %UnifiedReturnBlock
    176 shortcirc_done.4:               ; preds = %shortcirc_next.3, %shortcirc_next.2, %shortcirc_next.1, %shortcirc_next.0, %entry
    177         br label %UnifiedReturnBlock
    178 UnifiedReturnBlock:             ; preds = %shortcirc_done.4, %shortcirc_next.4
    179         %UnifiedRetVal = phi i1 [ %tmp.26, %shortcirc_next.4 ], [ true, %shortcirc_done.4 ]             ; <i1> [#uses=1]
    180         ret i1 %UnifiedRetVal
    181 
    182 ; CHECK-LABEL: @test6(
    183 ; CHECK: %tmp.2.i.off = add i32 %tmp.2.i, -14
    184 ; CHECK: %switch = icmp ult i32 %tmp.2.i.off, 6
    185 }
    186 
    187 define void @test7(i8 zeroext %c, i32 %x) nounwind ssp noredzone {
    188 entry:
    189   %cmp = icmp ult i32 %x, 32
    190   %cmp4 = icmp eq i8 %c, 97
    191   %or.cond = or i1 %cmp, %cmp4
    192   %cmp9 = icmp eq i8 %c, 99
    193   %or.cond11 = or i1 %or.cond, %cmp9
    194   br i1 %or.cond11, label %if.then, label %if.end
    195 
    196 if.then:                                          ; preds = %entry
    197   tail call void @foo1() nounwind noredzone
    198   ret void
    199 
    200 if.end:                                           ; preds = %entry
    201   ret void
    202 
    203 ; CHECK-LABEL: @test7(
    204 ; CHECK:   %cmp = icmp ult i32 %x, 32
    205 ; CHECK:   br i1 %cmp, label %if.then, label %switch.early.test
    206 ; CHECK: switch.early.test:
    207 ; CHECK:   switch i8 %c, label %if.end [
    208 ; CHECK:     i8 99, label %if.then
    209 ; CHECK:     i8 97, label %if.then
    210 ; CHECK:   ]
    211 }
    212 
    213 define i32 @test8(i8 zeroext %c, i32 %x, i1 %C) nounwind ssp noredzone {
    214 entry:
    215   br i1 %C, label %N, label %if.then
    216 N:
    217   %cmp = icmp ult i32 %x, 32
    218   %cmp4 = icmp eq i8 %c, 97
    219   %or.cond = or i1 %cmp, %cmp4
    220   %cmp9 = icmp eq i8 %c, 99
    221   %or.cond11 = or i1 %or.cond, %cmp9
    222   br i1 %or.cond11, label %if.then, label %if.end
    223 
    224 if.then:                                          ; preds = %entry
    225   %A = phi i32 [0, %entry], [42, %N]
    226   tail call void @foo1() nounwind noredzone
    227   ret i32 %A
    228 
    229 if.end:                                           ; preds = %entry
    230   ret i32 0
    231 
    232 ; CHECK-LABEL: @test8(
    233 ; CHECK: switch.early.test:
    234 ; CHECK:   switch i8 %c, label %if.end [
    235 ; CHECK:     i8 99, label %if.then
    236 ; CHECK:     i8 97, label %if.then
    237 ; CHECK:   ]
    238 ; CHECK:   %A = phi i32 [ 0, %entry ], [ 42, %switch.early.test ], [ 42, %N ], [ 42, %switch.early.test ]
    239 }
    240 
    241 ;; This is "Example 7" from http://blog.regehr.org/archives/320
    242 define i32 @test9(i8 zeroext %c) nounwind ssp noredzone {
    243 entry:
    244   %cmp = icmp ult i8 %c, 33
    245   br i1 %cmp, label %lor.end, label %lor.lhs.false
    246 
    247 lor.lhs.false:                                    ; preds = %entry
    248   %cmp4 = icmp eq i8 %c, 46
    249   br i1 %cmp4, label %lor.end, label %lor.lhs.false6
    250 
    251 lor.lhs.false6:                                   ; preds = %lor.lhs.false
    252   %cmp9 = icmp eq i8 %c, 44
    253   br i1 %cmp9, label %lor.end, label %lor.lhs.false11
    254 
    255 lor.lhs.false11:                                  ; preds = %lor.lhs.false6
    256   %cmp14 = icmp eq i8 %c, 58
    257   br i1 %cmp14, label %lor.end, label %lor.lhs.false16
    258 
    259 lor.lhs.false16:                                  ; preds = %lor.lhs.false11
    260   %cmp19 = icmp eq i8 %c, 59
    261   br i1 %cmp19, label %lor.end, label %lor.lhs.false21
    262 
    263 lor.lhs.false21:                                  ; preds = %lor.lhs.false16
    264   %cmp24 = icmp eq i8 %c, 60
    265   br i1 %cmp24, label %lor.end, label %lor.lhs.false26
    266 
    267 lor.lhs.false26:                                  ; preds = %lor.lhs.false21
    268   %cmp29 = icmp eq i8 %c, 62
    269   br i1 %cmp29, label %lor.end, label %lor.lhs.false31
    270 
    271 lor.lhs.false31:                                  ; preds = %lor.lhs.false26
    272   %cmp34 = icmp eq i8 %c, 34
    273   br i1 %cmp34, label %lor.end, label %lor.lhs.false36
    274 
    275 lor.lhs.false36:                                  ; preds = %lor.lhs.false31
    276   %cmp39 = icmp eq i8 %c, 92
    277   br i1 %cmp39, label %lor.end, label %lor.rhs
    278 
    279 lor.rhs:                                          ; preds = %lor.lhs.false36
    280   %cmp43 = icmp eq i8 %c, 39
    281   br label %lor.end
    282 
    283 lor.end:                                          ; preds = %lor.rhs, %lor.lhs.false36, %lor.lhs.false31, %lor.lhs.false26, %lor.lhs.false21, %lor.lhs.false16, %lor.lhs.false11, %lor.lhs.false6, %lor.lhs.false, %entry
    284   %0 = phi i1 [ true, %lor.lhs.false36 ], [ true, %lor.lhs.false31 ], [ true, %lor.lhs.false26 ], [ true, %lor.lhs.false21 ], [ true, %lor.lhs.false16 ], [ true, %lor.lhs.false11 ], [ true, %lor.lhs.false6 ], [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp43, %lor.rhs ]
    285   %conv46 = zext i1 %0 to i32
    286   ret i32 %conv46
    287 
    288 ; CHECK-LABEL: @test9(
    289 ; CHECK:   %cmp = icmp ult i8 %c, 33
    290 ; CHECK:   br i1 %cmp, label %lor.end, label %switch.early.test
    291 
    292 ; CHECK: switch.early.test:
    293 ; CHECK:   switch i8 %c, label %lor.rhs [
    294 ; CHECK:     i8 92, label %lor.end
    295 ; CHECK:     i8 62, label %lor.end
    296 ; CHECK:     i8 60, label %lor.end
    297 ; CHECK:     i8 59, label %lor.end
    298 ; CHECK:     i8 58, label %lor.end
    299 ; CHECK:     i8 46, label %lor.end
    300 ; CHECK:     i8 44, label %lor.end
    301 ; CHECK:     i8 34, label %lor.end
    302 ; CHECK:     i8 39, label %lor.end
    303 ; CHECK:   ]
    304 }
    305 
    306 define i32 @test10(i32 %mode, i1 %Cond) {
    307   %A = icmp ne i32 %mode, 0
    308   %B = icmp ne i32 %mode, 51
    309   %C = and i1 %A, %B
    310   %D = and i1 %C, %Cond
    311   br i1 %D, label %T, label %F
    312 T:
    313   ret i32 123
    314 F:
    315   ret i32 324
    316 
    317 ; CHECK-LABEL: @test10(
    318 ; CHECK:  br i1 %Cond, label %switch.early.test, label %F
    319 ; CHECK:switch.early.test:
    320 ; CHECK:  switch i32 %mode, label %T [
    321 ; CHECK:    i32 51, label %F
    322 ; CHECK:    i32 0, label %F
    323 ; CHECK:  ]
    324 }
    325 
    326 ; PR8780
    327 define i32 @test11(i32 %bar) nounwind {
    328 entry:
    329   %cmp = icmp eq i32 %bar, 4
    330   %cmp2 = icmp eq i32 %bar, 35
    331   %or.cond = or i1 %cmp, %cmp2
    332   %cmp5 = icmp eq i32 %bar, 53
    333   %or.cond1 = or i1 %or.cond, %cmp5
    334   %cmp8 = icmp eq i32 %bar, 24
    335   %or.cond2 = or i1 %or.cond1, %cmp8
    336   %cmp11 = icmp eq i32 %bar, 23
    337   %or.cond3 = or i1 %or.cond2, %cmp11
    338   %cmp14 = icmp eq i32 %bar, 55
    339   %or.cond4 = or i1 %or.cond3, %cmp14
    340   %cmp17 = icmp eq i32 %bar, 12
    341   %or.cond5 = or i1 %or.cond4, %cmp17
    342   %cmp20 = icmp eq i32 %bar, 35
    343   %or.cond6 = or i1 %or.cond5, %cmp20
    344   br i1 %or.cond6, label %if.then, label %if.end
    345 
    346 if.then:                                          ; preds = %entry
    347   br label %return
    348 
    349 if.end:                                           ; preds = %entry
    350   br label %return
    351 
    352 return:                                           ; preds = %if.end, %if.then
    353   %retval.0 = phi i32 [ 1, %if.then ], [ 0, %if.end ]
    354   ret i32 %retval.0
    355 
    356 ; CHECK-LABEL: @test11(
    357 ; CHECK: switch i32 %bar, label %if.end [
    358 ; CHECK:   i32 55, label %return
    359 ; CHECK:   i32 53, label %return
    360 ; CHECK:   i32 35, label %return
    361 ; CHECK:   i32 24, label %return
    362 ; CHECK:   i32 23, label %return
    363 ; CHECK:   i32 12, label %return
    364 ; CHECK:   i32 4, label %return
    365 ; CHECK: ]
    366 }
    367 
    368 define void @test12() nounwind {
    369 entry:
    370   br label %bb49.us.us
    371 
    372 bb49.us.us:
    373   %A = icmp eq i32 undef, undef
    374   br i1 %A, label %bb55.us.us, label %malformed
    375 
    376 bb48.us.us:
    377   %B = icmp ugt i32 undef, undef
    378   br i1 %B, label %bb55.us.us, label %bb49.us.us
    379 
    380 bb55.us.us:
    381   br label %bb48.us.us
    382 
    383 malformed:
    384   ret void
    385 ; CHECK-LABEL: @test12(
    386 
    387 }
    388 
    389 ; test13 - handle switch formation with ult.
    390 define void @test13(i32 %x) nounwind ssp noredzone {
    391 entry:
    392   %cmp = icmp ult i32 %x, 2
    393   br i1 %cmp, label %if.then, label %lor.lhs.false3
    394 
    395 lor.lhs.false3:                                   ; preds = %lor.lhs.false
    396   %cmp5 = icmp eq i32 %x, 3
    397   br i1 %cmp5, label %if.then, label %lor.lhs.false6
    398 
    399 lor.lhs.false6:                                   ; preds = %lor.lhs.false3
    400   %cmp8 = icmp eq i32 %x, 4
    401   br i1 %cmp8, label %if.then, label %lor.lhs.false9
    402 
    403 lor.lhs.false9:                                   ; preds = %lor.lhs.false6
    404   %cmp11 = icmp eq i32 %x, 6
    405   br i1 %cmp11, label %if.then, label %if.end
    406 
    407 if.then:                                          ; preds = %lor.lhs.false9, %lor.lhs.false6, %lor.lhs.false3, %lor.lhs.false, %entry
    408   call void @foo1() noredzone
    409   br label %if.end
    410 
    411 if.end:                                           ; preds = %if.then, %lor.lhs.false9
    412   ret void
    413 ; CHECK-LABEL: @test13(
    414 ; CHECK:  switch i32 %x, label %if.end [
    415 ; CHECK:     i32 6, label %if.then
    416 ; CHECK:     i32 4, label %if.then
    417 ; CHECK:     i32 3, label %if.then
    418 ; CHECK:     i32 1, label %if.then
    419 ; CHECK:     i32 0, label %if.then
    420 ; CHECK:   ]
    421 }
    422 
    423 ; test14 - handle switch formation with ult.
    424 define void @test14(i32 %x) nounwind ssp noredzone {
    425 entry:
    426   %cmp = icmp ugt i32 %x, 2
    427   br i1 %cmp, label %lor.lhs.false3, label %if.then
    428 
    429 lor.lhs.false3:                                   ; preds = %lor.lhs.false
    430   %cmp5 = icmp ne i32 %x, 3
    431   br i1 %cmp5, label %lor.lhs.false6, label %if.then
    432 
    433 lor.lhs.false6:                                   ; preds = %lor.lhs.false3
    434   %cmp8 = icmp ne i32 %x, 4
    435   br i1 %cmp8, label %lor.lhs.false9, label %if.then
    436 
    437 lor.lhs.false9:                                   ; preds = %lor.lhs.false6
    438   %cmp11 = icmp ne i32 %x, 6
    439   br i1 %cmp11, label %if.end, label %if.then
    440 
    441 if.then:                                          ; preds = %lor.lhs.false9, %lor.lhs.false6, %lor.lhs.false3, %lor.lhs.false, %entry
    442   call void @foo1() noredzone
    443   br label %if.end
    444 
    445 if.end:                                           ; preds = %if.then, %lor.lhs.false9
    446   ret void
    447 ; CHECK-LABEL: @test14(
    448 ; CHECK:  switch i32 %x, label %if.end [
    449 ; CHECK:     i32 6, label %if.then
    450 ; CHECK:     i32 4, label %if.then
    451 ; CHECK:     i32 3, label %if.then
    452 ; CHECK:     i32 1, label %if.then
    453 ; CHECK:     i32 0, label %if.then
    454 ; CHECK:   ]
    455 }
    456 
    457 ; Don't crash on ginormous ranges.
    458 define void @test15(i128 %x) nounwind {
    459   %cmp = icmp ugt i128 %x, 2
    460   br i1 %cmp, label %if.end, label %lor.false
    461 
    462 lor.false:
    463   %cmp2 = icmp ne i128 %x, 100000000000000000000
    464   br i1 %cmp2, label %if.end, label %if.then
    465 
    466 if.then:
    467   call void @foo1() noredzone
    468   br label %if.end
    469 
    470 if.end:
    471   ret void
    472 
    473 ; CHECK-LABEL: @test15(
    474 ; CHECK-NOT: switch
    475 ; CHECK: ret void
    476 }
    477 
    478 ; PR8675
    479 ; rdar://5134905
    480 define zeroext i1 @test16(i32 %x) nounwind {
    481 entry:
    482 ; CHECK-LABEL: @test16(
    483 ; CHECK: %x.off = add i32 %x, -1
    484 ; CHECK: %switch = icmp ult i32 %x.off, 3
    485   %cmp.i = icmp eq i32 %x, 1
    486   br i1 %cmp.i, label %lor.end, label %lor.lhs.false
    487 
    488 lor.lhs.false:
    489   %cmp.i2 = icmp eq i32 %x, 2
    490   br i1 %cmp.i2, label %lor.end, label %lor.rhs
    491 
    492 lor.rhs:
    493   %cmp.i1 = icmp eq i32 %x, 3
    494   br label %lor.end
    495 
    496 lor.end:
    497   %0 = phi i1 [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp.i1, %lor.rhs ]
    498   ret i1 %0
    499 }
    500 
    501 ; Check that we don't turn an icmp into a switch where it's not useful.
    502 define void @test17(i32 %x, i32 %y) {
    503   %cmp = icmp ult i32 %x, 3
    504   %switch = icmp ult i32 %y, 2
    505   %or.cond775 = or i1 %cmp, %switch
    506   br i1 %or.cond775, label %lor.lhs.false8, label %return
    507 
    508 lor.lhs.false8:
    509   tail call void @foo1()
    510   ret void
    511 
    512 return:
    513   ret void
    514 
    515 ; CHECK-LABEL: @test17(
    516 ; CHECK-NOT: switch.early.test
    517 ; CHECK-NOT: switch i32
    518 ; CHECK: ret void
    519 }
    520 
    521 define void @test18(i32 %arg) {
    522 bb:
    523   %tmp = and i32 %arg, -2
    524   %tmp1 = icmp eq i32 %tmp, 8
    525   %tmp2 = icmp eq i32 %arg, 10
    526   %tmp3 = or i1 %tmp1, %tmp2
    527   %tmp4 = icmp eq i32 %arg, 11
    528   %tmp5 = or i1 %tmp3, %tmp4
    529   %tmp6 = icmp eq i32 %arg, 12
    530   %tmp7 = or i1 %tmp5, %tmp6
    531   br i1 %tmp7, label %bb19, label %bb8
    532 
    533 bb8:                                              ; preds = %bb
    534   %tmp9 = add i32 %arg, -13
    535   %tmp10 = icmp ult i32 %tmp9, 2
    536   %tmp11 = icmp eq i32 %arg, 16
    537   %tmp12 = or i1 %tmp10, %tmp11
    538   %tmp13 = icmp eq i32 %arg, 17
    539   %tmp14 = or i1 %tmp12, %tmp13
    540   %tmp15 = icmp eq i32 %arg, 18
    541   %tmp16 = or i1 %tmp14, %tmp15
    542   %tmp17 = icmp eq i32 %arg, 15
    543   %tmp18 = or i1 %tmp16, %tmp17
    544   br i1 %tmp18, label %bb19, label %bb20
    545 
    546 bb19:                                             ; preds = %bb8, %bb
    547   tail call void @foo1()
    548   br label %bb20
    549 
    550 bb20:                                             ; preds = %bb19, %bb8
    551   ret void
    552 
    553 ; CHECK-LABEL: @test18(
    554 ; CHECK: %arg.off = add i32 %arg, -8
    555 ; CHECK: icmp ult i32 %arg.off, 11
    556 }
    557 
    558 define void @PR26323(i1 %tobool23, i32 %tmp3) {
    559 entry:
    560   %tobool5 = icmp ne i32 %tmp3, 0
    561   %neg14 = and i32 %tmp3, -2
    562   %cmp17 = icmp ne i32 %neg14, -1
    563   %or.cond = and i1 %tobool5, %tobool23
    564   %or.cond1 = and i1 %cmp17, %or.cond
    565   br i1 %or.cond1, label %if.end29, label %if.then27
    566 
    567 if.then27:                                        ; preds = %entry
    568   call void @foo1()
    569   unreachable
    570 
    571 if.end29:                                         ; preds = %entry
    572   ret void
    573 }
    574 
    575 ; CHECK-LABEL: define void @PR26323(
    576 ; CHECK:  %tobool5 = icmp ne i32 %tmp3, 0
    577 ; CHECK:  %neg14 = and i32 %tmp3, -2
    578 ; CHECK:  %cmp17 = icmp ne i32 %neg14, -1
    579 ; CHECK:  %or.cond = and i1 %tobool5, %tobool23
    580 ; CHECK:  %or.cond1 = and i1 %cmp17, %or.cond
    581 ; CHECK:  br i1 %or.cond1, label %if.end29, label %if.then27
    582 
    583 ; Form a switch when and'ing a negated power of two
    584 ; CHECK-LABEL: define void @test19
    585 ; CHECK: switch i32 %arg, label %else [
    586 ; CHECK: i32 32, label %if
    587 ; CHECK: i32 13, label %if
    588 ; CHECK: i32 12, label %if
    589 define void @test19(i32 %arg) {
    590   %and = and i32 %arg, -2
    591   %cmp1 = icmp eq i32 %and, 12
    592   %cmp2 = icmp eq i32 %arg, 32
    593   %pred = or i1 %cmp1, %cmp2
    594   br i1 %pred, label %if, label %else
    595 
    596 if:
    597   call void @foo1()
    598   ret void
    599 
    600 else:
    601   ret void
    602 }
    603 
    604 ; Since %cmp1 is always false, a switch is never formed
    605 ; CHECK-LABEL: define void @test20
    606 ; CHECK-NOT: switch
    607 ; CHECK: ret void
    608 define void @test20(i32 %arg) {
    609   %and = and i32 %arg, -2
    610   %cmp1 = icmp eq i32 %and, 13
    611   %cmp2 = icmp eq i32 %arg, 32
    612   %pred = or i1 %cmp1, %cmp2
    613   br i1 %pred, label %if, label %else
    614 
    615 if:
    616   call void @foo1()
    617   ret void
    618 
    619 else:
    620   ret void
    621 }
    622 
    623 ; Form a switch when or'ing a power of two
    624 ; CHECK-LABEL: define void @test21
    625 ; CHECK: i32 32, label %else
    626 ; CHECK: i32 13, label %else
    627 ; CHECK: i32 12, label %else
    628 define void @test21(i32 %arg) {
    629   %and = or i32 %arg, 1
    630   %cmp1 = icmp ne i32 %and, 13
    631   %cmp2 = icmp ne i32 %arg, 32
    632   %pred = and i1 %cmp1, %cmp2
    633   br i1 %pred, label %if, label %else
    634 
    635 if:
    636   call void @foo1()
    637   ret void
    638 
    639 else:
    640   ret void
    641 }
    642 
    643 ; Since %cmp1 is always false, a switch is never formed
    644 ; CHECK-LABEL: define void @test22
    645 ; CHECK-NOT: switch
    646 ; CHECK: ret void
    647 define void @test22(i32 %arg) {
    648   %and = or i32 %arg, 1
    649   %cmp1 = icmp ne i32 %and, 12
    650   %cmp2 = icmp ne i32 %arg, 32
    651   %pred = and i1 %cmp1, %cmp2
    652   br i1 %pred, label %if, label %else
    653 
    654 if:
    655   call void @foo1()
    656   ret void
    657 
    658 else:
    659   ret void
    660 }