Home | History | Annotate | Download | only in SystemZ
      1 ; Test 64-bit comparison in which the second operand is a zero-extended i32.
      2 ;
      3 ; RUN: llc < %s -mtriple=s390x-linux-gnu | FileCheck %s
      4 
      5 declare i64 @foo()
      6 
      7 ; Check unsigned register comparison.
      8 define double @f1(double %a, double %b, i64 %i1, i32 %unext) {
      9 ; CHECK-LABEL: f1:
     10 ; CHECK: clgfr %r2, %r3
     11 ; CHECK-NEXT: jl
     12 ; CHECK: ldr %f0, %f2
     13 ; CHECK: br %r14
     14   %i2 = zext i32 %unext to i64
     15   %cond = icmp ult i64 %i1, %i2
     16   %res = select i1 %cond, double %a, double %b
     17   ret double %res
     18 }
     19 
     20 ; ...and again with a different representation.
     21 define double @f2(double %a, double %b, i64 %i1, i64 %unext) {
     22 ; CHECK-LABEL: f2:
     23 ; CHECK: clgfr %r2, %r3
     24 ; CHECK-NEXT: jl
     25 ; CHECK: ldr %f0, %f2
     26 ; CHECK: br %r14
     27   %i2 = and i64 %unext, 4294967295
     28   %cond = icmp ult i64 %i1, %i2
     29   %res = select i1 %cond, double %a, double %b
     30   ret double %res
     31 }
     32 
     33 ; Check signed register comparison, which can't use CLGFR.
     34 define double @f3(double %a, double %b, i64 %i1, i32 %unext) {
     35 ; CHECK-LABEL: f3:
     36 ; CHECK-NOT: clgfr
     37 ; CHECK: br %r14
     38   %i2 = zext i32 %unext to i64
     39   %cond = icmp slt i64 %i1, %i2
     40   %res = select i1 %cond, double %a, double %b
     41   ret double %res
     42 }
     43 
     44 ; ...and again with a different representation
     45 define double @f4(double %a, double %b, i64 %i1, i64 %unext) {
     46 ; CHECK-LABEL: f4:
     47 ; CHECK-NOT: clgfr
     48 ; CHECK: br %r14
     49   %i2 = and i64 %unext, 4294967295
     50   %cond = icmp slt i64 %i1, %i2
     51   %res = select i1 %cond, double %a, double %b
     52   ret double %res
     53 }
     54 
     55 ; Check register equality.
     56 define double @f5(double %a, double %b, i64 %i1, i32 %unext) {
     57 ; CHECK-LABEL: f5:
     58 ; CHECK: clgfr %r2, %r3
     59 ; CHECK-NEXT: je
     60 ; CHECK: ldr %f0, %f2
     61 ; CHECK: br %r14
     62   %i2 = zext i32 %unext to i64
     63   %cond = icmp eq i64 %i1, %i2
     64   %res = select i1 %cond, double %a, double %b
     65   ret double %res
     66 }
     67 
     68 ; ...and again with a different representation
     69 define double @f6(double %a, double %b, i64 %i1, i64 %unext) {
     70 ; CHECK-LABEL: f6:
     71 ; CHECK: clgfr %r2, %r3
     72 ; CHECK-NEXT: je
     73 ; CHECK: ldr %f0, %f2
     74 ; CHECK: br %r14
     75   %i2 = and i64 %unext, 4294967295
     76   %cond = icmp eq i64 %i1, %i2
     77   %res = select i1 %cond, double %a, double %b
     78   ret double %res
     79 }
     80 
     81 ; Check register inequality.
     82 define double @f7(double %a, double %b, i64 %i1, i32 %unext) {
     83 ; CHECK-LABEL: f7:
     84 ; CHECK: clgfr %r2, %r3
     85 ; CHECK-NEXT: jlh
     86 ; CHECK: ldr %f0, %f2
     87 ; CHECK: br %r14
     88   %i2 = zext i32 %unext to i64
     89   %cond = icmp ne i64 %i1, %i2
     90   %res = select i1 %cond, double %a, double %b
     91   ret double %res
     92 }
     93 
     94 ; ...and again with a different representation
     95 define double @f8(double %a, double %b, i64 %i1, i64 %unext) {
     96 ; CHECK-LABEL: f8:
     97 ; CHECK: clgfr %r2, %r3
     98 ; CHECK-NEXT: jlh
     99 ; CHECK: ldr %f0, %f2
    100 ; CHECK: br %r14
    101   %i2 = and i64 %unext, 4294967295
    102   %cond = icmp ne i64 %i1, %i2
    103   %res = select i1 %cond, double %a, double %b
    104   ret double %res
    105 }
    106 
    107 ; Check unsigned comparison with memory.
    108 define double @f9(double %a, double %b, i64 %i1, i32 *%ptr) {
    109 ; CHECK-LABEL: f9:
    110 ; CHECK: clgf %r2, 0(%r3)
    111 ; CHECK-NEXT: jl
    112 ; CHECK: ldr %f0, %f2
    113 ; CHECK: br %r14
    114   %unext = load i32 *%ptr
    115   %i2 = zext i32 %unext to i64
    116   %cond = icmp ult i64 %i1, %i2
    117   %res = select i1 %cond, double %a, double %b
    118   ret double %res
    119 }
    120 
    121 ; Check signed comparison with memory.
    122 define double @f10(double %a, double %b, i64 %i1, i32 *%ptr) {
    123 ; CHECK-LABEL: f10:
    124 ; CHECK-NOT: clgf
    125 ; CHECK: br %r14
    126   %unext = load i32 *%ptr
    127   %i2 = zext i32 %unext to i64
    128   %cond = icmp slt i64 %i1, %i2
    129   %res = select i1 %cond, double %a, double %b
    130   ret double %res
    131 }
    132 
    133 ; Check memory equality.
    134 define double @f11(double %a, double %b, i64 %i1, i32 *%ptr) {
    135 ; CHECK-LABEL: f11:
    136 ; CHECK: clgf %r2, 0(%r3)
    137 ; CHECK-NEXT: je
    138 ; CHECK: ldr %f0, %f2
    139 ; CHECK: br %r14
    140   %unext = load i32 *%ptr
    141   %i2 = zext i32 %unext to i64
    142   %cond = icmp eq i64 %i1, %i2
    143   %res = select i1 %cond, double %a, double %b
    144   ret double %res
    145 }
    146 
    147 ; Check memory inequality.
    148 define double @f12(double %a, double %b, i64 %i1, i32 *%ptr) {
    149 ; CHECK-LABEL: f12:
    150 ; CHECK: clgf %r2, 0(%r3)
    151 ; CHECK-NEXT: jlh
    152 ; CHECK: ldr %f0, %f2
    153 ; CHECK: br %r14
    154   %unext = load i32 *%ptr
    155   %i2 = zext i32 %unext to i64
    156   %cond = icmp ne i64 %i1, %i2
    157   %res = select i1 %cond, double %a, double %b
    158   ret double %res
    159 }
    160 
    161 ; Check the high end of the aligned CLGF range.
    162 define double @f13(double %a, double %b, i64 %i1, i32 *%base) {
    163 ; CHECK-LABEL: f13:
    164 ; CHECK: clgf %r2, 524284(%r3)
    165 ; CHECK-NEXT: jl
    166 ; CHECK: ldr %f0, %f2
    167 ; CHECK: br %r14
    168   %ptr = getelementptr i32 *%base, i64 131071
    169   %unext = load i32 *%ptr
    170   %i2 = zext i32 %unext to i64
    171   %cond = icmp ult i64 %i1, %i2
    172   %res = select i1 %cond, double %a, double %b
    173   ret double %res
    174 }
    175 
    176 ; Check the next word up, which needs separate address logic.
    177 ; Other sequences besides this one would be OK.
    178 define double @f14(double %a, double %b, i64 %i1, i32 *%base) {
    179 ; CHECK-LABEL: f14:
    180 ; CHECK: agfi %r3, 524288
    181 ; CHECK: clgf %r2, 0(%r3)
    182 ; CHECK-NEXT: jl
    183 ; CHECK: ldr %f0, %f2
    184 ; CHECK: br %r14
    185   %ptr = getelementptr i32 *%base, i64 131072
    186   %unext = load i32 *%ptr
    187   %i2 = zext i32 %unext to i64
    188   %cond = icmp ult i64 %i1, %i2
    189   %res = select i1 %cond, double %a, double %b
    190   ret double %res
    191 }
    192 
    193 ; Check the high end of the negative aligned CLGF range.
    194 define double @f15(double %a, double %b, i64 %i1, i32 *%base) {
    195 ; CHECK-LABEL: f15:
    196 ; CHECK: clgf %r2, -4(%r3)
    197 ; CHECK-NEXT: jl
    198 ; CHECK: ldr %f0, %f2
    199 ; CHECK: br %r14
    200   %ptr = getelementptr i32 *%base, i64 -1
    201   %unext = load i32 *%ptr
    202   %i2 = zext i32 %unext to i64
    203   %cond = icmp ult i64 %i1, %i2
    204   %res = select i1 %cond, double %a, double %b
    205   ret double %res
    206 }
    207 
    208 ; Check the low end of the CLGF range.
    209 define double @f16(double %a, double %b, i64 %i1, i32 *%base) {
    210 ; CHECK-LABEL: f16:
    211 ; CHECK: clgf %r2, -524288(%r3)
    212 ; CHECK-NEXT: jl
    213 ; CHECK: ldr %f0, %f2
    214 ; CHECK: br %r14
    215   %ptr = getelementptr i32 *%base, i64 -131072
    216   %unext = load i32 *%ptr
    217   %i2 = zext i32 %unext to i64
    218   %cond = icmp ult i64 %i1, %i2
    219   %res = select i1 %cond, double %a, double %b
    220   ret double %res
    221 }
    222 
    223 ; Check the next word down, which needs separate address logic.
    224 ; Other sequences besides this one would be OK.
    225 define double @f17(double %a, double %b, i64 %i1, i32 *%base) {
    226 ; CHECK-LABEL: f17:
    227 ; CHECK: agfi %r3, -524292
    228 ; CHECK: clgf %r2, 0(%r3)
    229 ; CHECK-NEXT: jl
    230 ; CHECK: ldr %f0, %f2
    231 ; CHECK: br %r14
    232   %ptr = getelementptr i32 *%base, i64 -131073
    233   %unext = load i32 *%ptr
    234   %i2 = zext i32 %unext to i64
    235   %cond = icmp ult i64 %i1, %i2
    236   %res = select i1 %cond, double %a, double %b
    237   ret double %res
    238 }
    239 
    240 ; Check that CLGF allows an index.
    241 define double @f18(double %a, double %b, i64 %i1, i64 %base, i64 %index) {
    242 ; CHECK-LABEL: f18:
    243 ; CHECK: clgf %r2, 524284({{%r4,%r3|%r3,%r4}})
    244 ; CHECK-NEXT: jl
    245 ; CHECK: ldr %f0, %f2
    246 ; CHECK: br %r14
    247   %add1 = add i64 %base, %index
    248   %add2 = add i64 %add1, 524284
    249   %ptr = inttoptr i64 %add2 to i32 *
    250   %unext = load i32 *%ptr
    251   %i2 = zext i32 %unext to i64
    252   %cond = icmp ult i64 %i1, %i2
    253   %res = select i1 %cond, double %a, double %b
    254   ret double %res
    255 }
    256 
    257 ; Check that comparisons of spilled values can use CLGF rather than CLGFR.
    258 define i64 @f19(i32 *%ptr0) {
    259 ; CHECK-LABEL: f19:
    260 ; CHECK: brasl %r14, foo@PLT
    261 ; CHECK: clgf {{%r[0-9]+}}, 16{{[04]}}(%r15)
    262 ; CHECK: br %r14
    263   %ptr1 = getelementptr i32 *%ptr0, i64 2
    264   %ptr2 = getelementptr i32 *%ptr0, i64 4
    265   %ptr3 = getelementptr i32 *%ptr0, i64 6
    266   %ptr4 = getelementptr i32 *%ptr0, i64 8
    267   %ptr5 = getelementptr i32 *%ptr0, i64 10
    268   %ptr6 = getelementptr i32 *%ptr0, i64 12
    269   %ptr7 = getelementptr i32 *%ptr0, i64 14
    270   %ptr8 = getelementptr i32 *%ptr0, i64 16
    271   %ptr9 = getelementptr i32 *%ptr0, i64 18
    272 
    273   %val0 = load i32 *%ptr0
    274   %val1 = load i32 *%ptr1
    275   %val2 = load i32 *%ptr2
    276   %val3 = load i32 *%ptr3
    277   %val4 = load i32 *%ptr4
    278   %val5 = load i32 *%ptr5
    279   %val6 = load i32 *%ptr6
    280   %val7 = load i32 *%ptr7
    281   %val8 = load i32 *%ptr8
    282   %val9 = load i32 *%ptr9
    283 
    284   %frob0 = add i32 %val0, 100
    285   %frob1 = add i32 %val1, 100
    286   %frob2 = add i32 %val2, 100
    287   %frob3 = add i32 %val3, 100
    288   %frob4 = add i32 %val4, 100
    289   %frob5 = add i32 %val5, 100
    290   %frob6 = add i32 %val6, 100
    291   %frob7 = add i32 %val7, 100
    292   %frob8 = add i32 %val8, 100
    293   %frob9 = add i32 %val9, 100
    294 
    295   store i32 %frob0, i32 *%ptr0
    296   store i32 %frob1, i32 *%ptr1
    297   store i32 %frob2, i32 *%ptr2
    298   store i32 %frob3, i32 *%ptr3
    299   store i32 %frob4, i32 *%ptr4
    300   store i32 %frob5, i32 *%ptr5
    301   store i32 %frob6, i32 *%ptr6
    302   store i32 %frob7, i32 *%ptr7
    303   store i32 %frob8, i32 *%ptr8
    304   store i32 %frob9, i32 *%ptr9
    305 
    306   %ret = call i64 @foo()
    307 
    308   %ext0 = zext i32 %frob0 to i64
    309   %ext1 = zext i32 %frob1 to i64
    310   %ext2 = zext i32 %frob2 to i64
    311   %ext3 = zext i32 %frob3 to i64
    312   %ext4 = zext i32 %frob4 to i64
    313   %ext5 = zext i32 %frob5 to i64
    314   %ext6 = zext i32 %frob6 to i64
    315   %ext7 = zext i32 %frob7 to i64
    316   %ext8 = zext i32 %frob8 to i64
    317   %ext9 = zext i32 %frob9 to i64
    318 
    319   %cmp0 = icmp ult i64 %ret, %ext0
    320   %cmp1 = icmp ult i64 %ret, %ext1
    321   %cmp2 = icmp ult i64 %ret, %ext2
    322   %cmp3 = icmp ult i64 %ret, %ext3
    323   %cmp4 = icmp ult i64 %ret, %ext4
    324   %cmp5 = icmp ult i64 %ret, %ext5
    325   %cmp6 = icmp ult i64 %ret, %ext6
    326   %cmp7 = icmp ult i64 %ret, %ext7
    327   %cmp8 = icmp ult i64 %ret, %ext8
    328   %cmp9 = icmp ult i64 %ret, %ext9
    329 
    330   %sel0 = select i1 %cmp0, i64 %ret, i64 0
    331   %sel1 = select i1 %cmp1, i64 %sel0, i64 1
    332   %sel2 = select i1 %cmp2, i64 %sel1, i64 2
    333   %sel3 = select i1 %cmp3, i64 %sel2, i64 3
    334   %sel4 = select i1 %cmp4, i64 %sel3, i64 4
    335   %sel5 = select i1 %cmp5, i64 %sel4, i64 5
    336   %sel6 = select i1 %cmp6, i64 %sel5, i64 6
    337   %sel7 = select i1 %cmp7, i64 %sel6, i64 7
    338   %sel8 = select i1 %cmp8, i64 %sel7, i64 8
    339   %sel9 = select i1 %cmp9, i64 %sel8, i64 9
    340 
    341   ret i64 %sel9
    342 }
    343 
    344 ; Check the comparison can be reversed if that allows CLGFR to be used.
    345 define double @f20(double %a, double %b, i64 %i1, i32 %unext) {
    346 ; CHECK-LABEL: f20:
    347 ; CHECK: clgfr %r2, %r3
    348 ; CHECK-NEXT: jh
    349 ; CHECK: ldr %f0, %f2
    350 ; CHECK: br %r14
    351   %i2 = zext i32 %unext to i64
    352   %cond = icmp ult i64 %i2, %i1
    353   %res = select i1 %cond, double %a, double %b
    354   ret double %res
    355 }
    356 
    357 ; ...and again with the AND representation.
    358 define double @f21(double %a, double %b, i64 %i1, i64 %unext) {
    359 ; CHECK-LABEL: f21:
    360 ; CHECK: clgfr %r2, %r3
    361 ; CHECK-NEXT: jh
    362 ; CHECK: ldr %f0, %f2
    363 ; CHECK: br %r14
    364   %i2 = and i64 %unext, 4294967295
    365   %cond = icmp ult i64 %i2, %i1
    366   %res = select i1 %cond, double %a, double %b
    367   ret double %res
    368 }
    369 
    370 ; Check the comparison can be reversed if that allows CLGF to be used.
    371 define double @f22(double %a, double %b, i64 %i2, i32 *%ptr) {
    372 ; CHECK-LABEL: f22:
    373 ; CHECK: clgf %r2, 0(%r3)
    374 ; CHECK-NEXT: jh {{\.L.*}}
    375 ; CHECK: ldr %f0, %f2
    376 ; CHECK: br %r14
    377   %unext = load i32 *%ptr
    378   %i1 = zext i32 %unext to i64
    379   %cond = icmp ult i64 %i1, %i2
    380   %res = select i1 %cond, double %a, double %b
    381   ret double %res
    382 }
    383