1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
|
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py UTC_ARGS: --version 3
; Test 64-bit addition in which the second operand is constant and in which
; three-operand forms are available.
;
; RUN: llc < %s -mtriple=s390x-linux-gnu -mcpu=z196 | FileCheck %s
declare i64 @foo()
; Check subtraction of 1.
define zeroext i1 @f1(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f1:
; CHECK: # %bb.0:
; CHECK-NEXT: alghsik %r0, %r3, -1
; CHECK-NEXT: stg %r0, 0(%r4)
; CHECK-NEXT: ipm %r1
; CHECK-NEXT: afi %r1, -536870912
; CHECK-NEXT: risbg %r2, %r1, 63, 191, 33
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Check the high end of the ALGHSIK range.
define zeroext i1 @f2(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f2:
; CHECK: # %bb.0:
; CHECK-NEXT: alghsik %r0, %r3, -32768
; CHECK-NEXT: stg %r0, 0(%r4)
; CHECK-NEXT: ipm %r1
; CHECK-NEXT: afi %r1, -536870912
; CHECK-NEXT: risbg %r2, %r1, 63, 191, 33
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 32768)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Check the next value up, which must use SLGFI instead.
define zeroext i1 @f3(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f3:
; CHECK: # %bb.0:
; CHECK-NEXT: slgfi %r3, 32769
; CHECK-NEXT: stg %r3, 0(%r4)
; CHECK-NEXT: ipm %r0
; CHECK-NEXT: afi %r0, -536870912
; CHECK-NEXT: risbg %r2, %r0, 63, 191, 33
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 32769)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Check the high end of the negative ALGHSIK range.
define zeroext i1 @f4(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f4:
; CHECK: # %bb.0:
; CHECK-NEXT: alghsik %r0, %r3, 1
; CHECK-NEXT: stg %r0, 0(%r4)
; CHECK-NEXT: ipm %r1
; CHECK-NEXT: afi %r1, -536870912
; CHECK-NEXT: risbg %r2, %r1, 63, 191, 33
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 -1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Check the low end of the ALGHSIK range.
define zeroext i1 @f5(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f5:
; CHECK: # %bb.0:
; CHECK-NEXT: alghsik %r0, %r3, 32767
; CHECK-NEXT: stg %r0, 0(%r4)
; CHECK-NEXT: ipm %r1
; CHECK-NEXT: afi %r1, -536870912
; CHECK-NEXT: risbg %r2, %r1, 63, 191, 33
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 -32767)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Test the next value down, which cannot use either ALGHSIK or SLGFI.
define zeroext i1 @f6(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f6:
; CHECK: # %bb.0:
; CHECK-NEXT: lghi %r0, -32768
; CHECK-NEXT: slgr %r3, %r0
; CHECK-NEXT: ipm %r0
; CHECK-NEXT: afi %r0, -536870912
; CHECK-NEXT: risbg %r2, %r0, 63, 191, 33
; CHECK-NEXT: stg %r3, 0(%r4)
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 -32768)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Check using the overflow result for a branch.
define void @f7(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f7:
; CHECK: # %bb.0:
; CHECK-NEXT: alghsik %r0, %r3, -1
; CHECK-NEXT: stg %r0, 0(%r4)
; CHECK-NEXT: jgle foo@PLT
; CHECK-NEXT: .LBB6_1: # %exit
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
br i1 %obit, label %call, label %exit
call:
tail call i64 @foo()
br label %exit
exit:
ret void
}
; ... and the same with the inverted direction.
define void @f8(i64 %dummy, i64 %a, ptr %res) {
; CHECK-LABEL: f8:
; CHECK: # %bb.0:
; CHECK-NEXT: alghsik %r0, %r3, -1
; CHECK-NEXT: stg %r0, 0(%r4)
; CHECK-NEXT: jgnle foo@PLT
; CHECK-NEXT: .LBB7_1: # %exit
; CHECK-NEXT: br %r14
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
br i1 %obit, label %exit, label %call
call:
tail call i64 @foo()
br label %exit
exit:
ret void
}
declare {i64, i1} @llvm.usub.with.overflow.i64(i64, i64) nounwind readnone
|