Home | History | Annotate | Download | only in NaryReassociate
      1 ; RUN: opt < %s -nary-reassociate -S | FileCheck %s
      2 ; RUN: opt < %s -passes='nary-reassociate' -S | FileCheck %s
      3 
      4 target datalayout = "e-i64:64-v16:16-v32:32-n16:32:64"
      5 
      6 declare void @foo(i32)
      7 
      8 ; foo(a + c);
      9 ; foo((a + (b + c));
     10 ;   =>
     11 ; t = a + c;
     12 ; foo(t);
     13 ; foo(t + b);
     14 define void @left_reassociate(i32 %a, i32 %b, i32 %c) {
     15 ; CHECK-LABEL: @left_reassociate(
     16   %1 = add i32 %a, %c
     17 ; CHECK: [[BASE:%[a-zA-Z0-9]+]] = add i32 %a, %c
     18   call void @foo(i32 %1)
     19   %2 = add i32 %b, %c
     20   %3 = add i32 %a, %2
     21 ; CHECK: [[RESULT:%[a-zA-Z0-9]+]] = add i32 [[BASE]], %b
     22   call void @foo(i32 %3)
     23 ; CHECK-NEXT: call void @foo(i32 [[RESULT]])
     24   ret void
     25 }
     26 
     27 ; foo(a + c);
     28 ; foo((a + b) + c);
     29 ;   =>
     30 ; t = a + c;
     31 ; foo(t);
     32 ; foo(t + b);
     33 define void @right_reassociate(i32 %a, i32 %b, i32 %c) {
     34 ; CHECK-LABEL: @right_reassociate(
     35   %1 = add i32 %a, %c
     36 ; CHECK: [[BASE:%[a-zA-Z0-9]+]] = add i32 %a, %c
     37   call void @foo(i32 %1)
     38   %2 = add i32 %a, %b
     39   %3 = add i32 %2, %c
     40 ; CHECK: [[RESULT:%[a-zA-Z0-9]+]] = add i32 [[BASE]], %b
     41   call void @foo(i32 %3)
     42 ; CHECK-NEXT: call void @foo(i32 [[RESULT]])
     43   ret void
     44 }
     45 
     46 ; t1 = a + c;
     47 ; foo(t1);
     48 ; t2 = a + b;
     49 ; foo(t2);
     50 ; t3 = t2 + c;
     51 ; foo(t3);
     52 ;
     53 ; Do not rewrite t3 into t1 + b because t2 is used elsewhere and is likely free.
     54 define void @no_reassociate(i32 %a, i32 %b, i32 %c) {
     55 ; CHECK-LABEL: @no_reassociate(
     56   %1 = add i32 %a, %c
     57 ; CHECK: add i32 %a, %c
     58   call void @foo(i32 %1)
     59   %2 = add i32 %a, %b
     60 ; CHECK: add i32 %a, %b
     61   call void @foo(i32 %2)
     62   %3 = add i32 %2, %c
     63 ; CHECK: add i32 %2, %c
     64   call void @foo(i32 %3)
     65   ret void
     66 }
     67 
     68 ; if (p1)
     69 ;   foo(a + c);
     70 ; if (p2)
     71 ;   foo(a + c);
     72 ; if (p3)
     73 ;   foo((a + b) + c);
     74 ;
     75 ; No action because (a + c) does not dominate ((a + b) + c).
     76 define void @conditional(i1 %p1, i1 %p2, i1 %p3, i32 %a, i32 %b, i32 %c) {
     77 ; CHECK-LABEL: @conditional(
     78 entry:
     79   br i1 %p1, label %then1, label %branch1
     80 
     81 then1:
     82   %0 = add i32 %a, %c
     83 ; CHECK: add i32 %a, %c
     84   call void @foo(i32 %0)
     85   br label %branch1
     86 
     87 branch1:
     88   br i1 %p2, label %then2, label %branch2
     89 
     90 then2:
     91   %1 = add i32 %a, %c
     92 ; CHECK: add i32 %a, %c
     93   call void @foo(i32 %1)
     94   br label %branch2
     95 
     96 branch2:
     97   br i1 %p3, label %then3, label %return
     98 
     99 then3:
    100   %2 = add i32 %a, %b
    101 ; CHECK: %2 = add i32 %a, %b
    102   %3 = add i32 %2, %c
    103 ; CHECK: add i32 %2, %c
    104   call void @foo(i32 %3)
    105   br label %return
    106 
    107 return:
    108   ret void
    109 }
    110 
    111 ; This test involves more conditional reassociation candidates. It exercises
    112 ; the stack optimization in tryReassociatedAdd that pops the candidates that
    113 ; do not dominate the current instruction.
    114 ;
    115 ;       def1
    116 ;      cond1
    117 ;      /  \
    118 ;     /    \
    119 ;   cond2  use2
    120 ;   /  \
    121 ;  /    \
    122 ; def2  def3
    123 ;      cond3
    124 ;       /  \
    125 ;      /    \
    126 ;    def4   use1
    127 ;
    128 ; NaryReassociate should match use1 with def3, and use2 with def1.
    129 define void @conditional2(i32 %a, i32 %b, i32 %c, i1 %cond1, i1 %cond2, i1 %cond3) {
    130 entry:
    131   %def1 = add i32 %a, %b
    132   br i1 %cond1, label %bb1, label %bb6
    133 bb1:
    134   br i1 %cond2, label %bb2, label %bb3
    135 bb2:
    136   %def2 = add i32 %a, %b
    137   call void @foo(i32 %def2)
    138   ret void
    139 bb3:
    140   %def3 = add i32 %a, %b
    141   br i1 %cond3, label %bb4, label %bb5
    142 bb4:
    143   %def4 = add i32 %a, %b
    144   call void @foo(i32 %def4)
    145   ret void
    146 bb5:
    147   %0 = add i32 %a, %c
    148   %1 = add i32 %0, %b
    149 ; CHECK: [[t1:%[0-9]+]] = add i32 %def3, %c
    150   call void @foo(i32 %1) ; foo((a + c) + b);
    151 ; CHECK-NEXT: call void @foo(i32 [[t1]])
    152   ret void
    153 bb6:
    154   %2 = add i32 %a, %c
    155   %3 = add i32 %2, %b
    156 ; CHECK: [[t2:%[0-9]+]] = add i32 %def1, %c
    157   call void @foo(i32 %3) ; foo((a + c) + b);
    158 ; CHECK-NEXT: call void @foo(i32 [[t2]])
    159   ret void
    160 }
    161 
    162 ; foo((a + b) + c)
    163 ; foo(((a + d) + b) + c)
    164 ;   =>
    165 ; t = (a + b) + c;
    166 ; foo(t);
    167 ; foo(t + d);
    168 define void @quaternary(i32 %a, i32 %b, i32 %c, i32 %d) {
    169 ; CHECK-LABEL: @quaternary(
    170   %1 = add i32 %a, %b
    171   %2 = add i32 %1, %c
    172   call void @foo(i32 %2)
    173 ; CHECK: call void @foo(i32 [[TMP1:%[a-zA-Z0-9]]])
    174   %3 = add i32 %a, %d
    175   %4 = add i32 %3, %b
    176   %5 = add i32 %4, %c
    177 ; CHECK: [[TMP2:%[a-zA-Z0-9]]] = add i32 [[TMP1]], %d
    178   call void @foo(i32 %5)
    179 ; CHECK: call void @foo(i32 [[TMP2]]
    180   ret void
    181 }
    182 
    183 define void @iterative(i32 %a, i32 %b, i32 %c) {
    184   %ab = add i32 %a, %b
    185   %abc = add i32 %ab, %c
    186   call void @foo(i32 %abc)
    187 
    188   %ab2 = add i32 %ab, %b
    189   %ab2c = add i32 %ab2, %c
    190 ; CHECK: %ab2c = add i32 %abc, %b
    191   call void @foo(i32 %ab2c)
    192 ; CHECK-NEXT: call void @foo(i32 %ab2c)
    193 
    194   %ab3 = add i32 %ab2, %b
    195   %ab3c = add i32 %ab3, %c
    196 ; CHECK-NEXT: %ab3c = add i32 %ab2c, %b
    197   call void @foo(i32 %ab3c)
    198 ; CHECK-NEXT: call void @foo(i32 %ab3c)
    199 
    200   ret void
    201 }
    202 
    203 define void @avoid_infinite_loop(i32 %a, i32 %b) {
    204 ; CHECK-LABEL: @avoid_infinite_loop
    205   %ab = add i32 %a, %b
    206 ; CHECK-NEXT: %ab
    207   %ab2 = add i32 %ab, %b
    208 ; CHECK-NEXT: %ab2
    209   call void @foo(i32 %ab2)
    210 ; CHECK-NEXT: @foo(i32 %ab2)
    211   ret void
    212 }
    213