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