1; Test 64-bit unsigned comparison in which the second operand is a variable. 2; 3; RUN: llc < %s -mtriple=s390x-linux-gnu | FileCheck %s 4 5; Check CLGR. 6define double @f1(double %a, double %b, i64 %i1, i64 %i2) { 7; CHECK-LABEL: f1: 8; CHECK: clgrbl %r2, %r3, 0(%r14) 9; CHECK: ldr %f0, %f2 10; CHECK: br %r14 11 %cond = icmp ult i64 %i1, %i2 12 %res = select i1 %cond, double %a, double %b 13 ret double %res 14} 15 16; Check CLG with no displacement. 17define double @f2(double %a, double %b, i64 %i1, i64 *%ptr) { 18; CHECK-LABEL: f2: 19; CHECK: clg %r2, 0(%r3) 20; CHECK-NEXT: blr %r14 21; CHECK: ldr %f0, %f2 22; CHECK: br %r14 23 %i2 = load i64 , i64 *%ptr 24 %cond = icmp ult i64 %i1, %i2 25 %res = select i1 %cond, double %a, double %b 26 ret double %res 27} 28 29; Check the high end of the aligned CLG range. 30define double @f3(double %a, double %b, i64 %i1, i64 *%base) { 31; CHECK-LABEL: f3: 32; CHECK: clg %r2, 524280(%r3) 33; CHECK-NEXT: blr %r14 34; CHECK: ldr %f0, %f2 35; CHECK: br %r14 36 %ptr = getelementptr i64, i64 *%base, i64 65535 37 %i2 = load i64 , i64 *%ptr 38 %cond = icmp ult i64 %i1, %i2 39 %res = select i1 %cond, double %a, double %b 40 ret double %res 41} 42 43; Check the next doubleword up, which needs separate address logic. 44; Other sequences besides this one would be OK. 45define double @f4(double %a, double %b, i64 %i1, i64 *%base) { 46; CHECK-LABEL: f4: 47; CHECK: agfi %r3, 524288 48; CHECK: clg %r2, 0(%r3) 49; CHECK-NEXT: blr %r14 50; CHECK: ldr %f0, %f2 51; CHECK: br %r14 52 %ptr = getelementptr i64, i64 *%base, i64 65536 53 %i2 = load i64 , i64 *%ptr 54 %cond = icmp ult i64 %i1, %i2 55 %res = select i1 %cond, double %a, double %b 56 ret double %res 57} 58 59; Check the high end of the negative aligned CLG range. 60define double @f5(double %a, double %b, i64 %i1, i64 *%base) { 61; CHECK-LABEL: f5: 62; CHECK: clg %r2, -8(%r3) 63; CHECK-NEXT: blr %r14 64; CHECK: ldr %f0, %f2 65; CHECK: br %r14 66 %ptr = getelementptr i64, i64 *%base, i64 -1 67 %i2 = load i64 , i64 *%ptr 68 %cond = icmp ult i64 %i1, %i2 69 %res = select i1 %cond, double %a, double %b 70 ret double %res 71} 72 73; Check the low end of the CLG range. 74define double @f6(double %a, double %b, i64 %i1, i64 *%base) { 75; CHECK-LABEL: f6: 76; CHECK: clg %r2, -524288(%r3) 77; CHECK-NEXT: blr %r14 78; CHECK: ldr %f0, %f2 79; CHECK: br %r14 80 %ptr = getelementptr i64, i64 *%base, i64 -65536 81 %i2 = load i64 , i64 *%ptr 82 %cond = icmp ult i64 %i1, %i2 83 %res = select i1 %cond, double %a, double %b 84 ret double %res 85} 86 87; Check the next doubleword down, which needs separate address logic. 88; Other sequences besides this one would be OK. 89define double @f7(double %a, double %b, i64 %i1, i64 *%base) { 90; CHECK-LABEL: f7: 91; CHECK: agfi %r3, -524296 92; CHECK: clg %r2, 0(%r3) 93; CHECK-NEXT: blr %r14 94; CHECK: ldr %f0, %f2 95; CHECK: br %r14 96 %ptr = getelementptr i64, i64 *%base, i64 -65537 97 %i2 = load i64 , i64 *%ptr 98 %cond = icmp ult i64 %i1, %i2 99 %res = select i1 %cond, double %a, double %b 100 ret double %res 101} 102 103; Check that CLG allows an index. 104define double @f8(double %a, double %b, i64 %i1, i64 %base, i64 %index) { 105; CHECK-LABEL: f8: 106; CHECK: clg %r2, 524280({{%r4,%r3|%r3,%r4}}) 107; CHECK-NEXT: blr %r14 108; CHECK: ldr %f0, %f2 109; CHECK: br %r14 110 %add1 = add i64 %base, %index 111 %add2 = add i64 %add1, 524280 112 %ptr = inttoptr i64 %add2 to i64 * 113 %i2 = load i64 , i64 *%ptr 114 %cond = icmp ult i64 %i1, %i2 115 %res = select i1 %cond, double %a, double %b 116 ret double %res 117} 118 119; Check the comparison can be reversed if that allows CLG to be used. 120define double @f9(double %a, double %b, i64 %i2, i64 *%ptr) { 121; CHECK-LABEL: f9: 122; CHECK: clg %r2, 0(%r3) 123; CHECK-NEXT: bhr %r14 124; CHECK: ldr %f0, %f2 125; CHECK: br %r14 126 %i1 = load i64 , i64 *%ptr 127 %cond = icmp ult i64 %i1, %i2 128 %res = select i1 %cond, double %a, double %b 129 ret double %res 130} 131