Files
clang-p2996/llvm/test/CodeGen/X86/xaluo.ll
Craig Topper cca49663a5 [FastISel][X86] Use getTypeForExtReturn in GetReturnInfo. (#80803)
The comment and code here seems to match getTypeForExtReturn. The
history shows that at the time this code was added, similar code existed
in SelectionDAGBuilder. SelectionDAGBuiler code has since been
refactored into getTypeForExtReturn.

This patch makes FastISel match SelectionDAGBuilder.

The test changes are because X86 has customization of
getTypeForExtReturn. So now we only extend returns to i8.

Stumbled onto this difference by accident.
2024-02-06 09:38:25 -08:00

1068 lines
30 KiB
LLVM

; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
; RUN: llc -mtriple=x86_64-darwin-unknown < %s | FileCheck %s --check-prefixes=CHECK,SDAG,GENERIC
; RUN: llc -mtriple=x86_64-darwin-unknown -fast-isel -fast-isel-abort=1 < %s | FileCheck %s --check-prefixes=CHECK,FAST
; RUN: llc -mtriple=x86_64-darwin-unknown -mcpu=knl < %s | FileCheck %s --check-prefixes=CHECK,SDAG,KNL
;
; Get the actual value of the overflow bit.
;
; SADDO reg, reg
define zeroext i1 @saddoi8(i8 signext %v1, i8 signext %v2, ptr %res) {
; SDAG-LABEL: saddoi8:
; SDAG: ## %bb.0:
; SDAG-NEXT: addb %sil, %dil
; SDAG-NEXT: seto %al
; SDAG-NEXT: movb %dil, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi8:
; FAST: ## %bb.0:
; FAST-NEXT: addb %sil, %dil
; FAST-NEXT: seto %al
; FAST-NEXT: movb %dil, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i8, i1} @llvm.sadd.with.overflow.i8(i8 %v1, i8 %v2)
%val = extractvalue {i8, i1} %t, 0
%obit = extractvalue {i8, i1} %t, 1
store i8 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoi16(i16 %v1, i16 %v2, ptr %res) {
; SDAG-LABEL: saddoi16:
; SDAG: ## %bb.0:
; SDAG-NEXT: addw %si, %di
; SDAG-NEXT: seto %al
; SDAG-NEXT: movw %di, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi16:
; FAST: ## %bb.0:
; FAST-NEXT: addw %si, %di
; FAST-NEXT: seto %al
; FAST-NEXT: movw %di, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i16, i1} @llvm.sadd.with.overflow.i16(i16 %v1, i16 %v2)
%val = extractvalue {i16, i1} %t, 0
%obit = extractvalue {i16, i1} %t, 1
store i16 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoi32(i32 %v1, i32 %v2, ptr %res) {
; SDAG-LABEL: saddoi32:
; SDAG: ## %bb.0:
; SDAG-NEXT: addl %esi, %edi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movl %edi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi32:
; FAST: ## %bb.0:
; FAST-NEXT: addl %esi, %edi
; FAST-NEXT: seto %al
; FAST-NEXT: movl %edi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.sadd.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
store i32 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoi64(i64 %v1, i64 %v2, ptr %res) {
; SDAG-LABEL: saddoi64:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq %rsi, %rdi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi64:
; FAST: ## %bb.0:
; FAST-NEXT: addq %rsi, %rdi
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rdi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; SADDO reg, 1 | INC
define zeroext i1 @saddoinci8(i8 %v1, ptr %res) {
; SDAG-LABEL: saddoinci8:
; SDAG: ## %bb.0:
; SDAG-NEXT: incb %dil
; SDAG-NEXT: seto %al
; SDAG-NEXT: movb %dil, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoinci8:
; FAST: ## %bb.0:
; FAST-NEXT: incb %dil
; FAST-NEXT: seto %al
; FAST-NEXT: movb %dil, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i8, i1} @llvm.sadd.with.overflow.i8(i8 %v1, i8 1)
%val = extractvalue {i8, i1} %t, 0
%obit = extractvalue {i8, i1} %t, 1
store i8 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoinci16(i16 %v1, ptr %res) {
; SDAG-LABEL: saddoinci16:
; SDAG: ## %bb.0:
; SDAG-NEXT: incw %di
; SDAG-NEXT: seto %al
; SDAG-NEXT: movw %di, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoinci16:
; FAST: ## %bb.0:
; FAST-NEXT: incw %di
; FAST-NEXT: seto %al
; FAST-NEXT: movw %di, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i16, i1} @llvm.sadd.with.overflow.i16(i16 %v1, i16 1)
%val = extractvalue {i16, i1} %t, 0
%obit = extractvalue {i16, i1} %t, 1
store i16 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoinci32(i32 %v1, ptr %res) {
; SDAG-LABEL: saddoinci32:
; SDAG: ## %bb.0:
; SDAG-NEXT: incl %edi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movl %edi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoinci32:
; FAST: ## %bb.0:
; FAST-NEXT: incl %edi
; FAST-NEXT: seto %al
; FAST-NEXT: movl %edi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.sadd.with.overflow.i32(i32 %v1, i32 1)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
store i32 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoinci64(i64 %v1, ptr %res) {
; SDAG-LABEL: saddoinci64:
; SDAG: ## %bb.0:
; SDAG-NEXT: incq %rdi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoinci64:
; FAST: ## %bb.0:
; FAST-NEXT: incq %rdi
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rdi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; SADDO reg, imm | imm, reg
define zeroext i1 @saddoi64imm1(i64 %v1, ptr %res) {
; SDAG-LABEL: saddoi64imm1:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq $2, %rdi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi64imm1:
; FAST: ## %bb.0:
; FAST-NEXT: addq $2, %rdi
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rdi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 2, i64 %v1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; Check boundary conditions for large immediates.
define zeroext i1 @saddoi64imm2(i64 %v1, ptr %res) {
; SDAG-LABEL: saddoi64imm2:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq $-2147483648, %rdi ## imm = 0x80000000
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi64imm2:
; FAST: ## %bb.0:
; FAST-NEXT: addq $-2147483648, %rdi ## imm = 0x80000000
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rdi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 -2147483648)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoi64imm3(i64 %v1, ptr %res) {
; SDAG-LABEL: saddoi64imm3:
; SDAG: ## %bb.0:
; SDAG-NEXT: movabsq $-21474836489, %rcx ## imm = 0xFFFFFFFAFFFFFFF7
; SDAG-NEXT: addq %rdi, %rcx
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rcx, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi64imm3:
; FAST: ## %bb.0:
; FAST-NEXT: movabsq $-21474836489, %rcx ## imm = 0xFFFFFFFAFFFFFFF7
; FAST-NEXT: addq %rdi, %rcx
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rcx, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 -21474836489)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoi64imm4(i64 %v1, ptr %res) {
; SDAG-LABEL: saddoi64imm4:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq $2147483647, %rdi ## imm = 0x7FFFFFFF
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi64imm4:
; FAST: ## %bb.0:
; FAST-NEXT: addq $2147483647, %rdi ## imm = 0x7FFFFFFF
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rdi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 2147483647)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @saddoi64imm5(i64 %v1, ptr %res) {
; SDAG-LABEL: saddoi64imm5:
; SDAG: ## %bb.0:
; SDAG-NEXT: subq $-2147483648, %rdi ## imm = 0x80000000
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: saddoi64imm5:
; FAST: ## %bb.0:
; FAST-NEXT: movl $2147483648, %ecx ## imm = 0x80000000
; FAST-NEXT: addq %rdi, %rcx
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rcx, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 2147483648)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; UADDO
define zeroext i1 @uaddoi32(i32 %v1, i32 %v2, ptr %res) {
; SDAG-LABEL: uaddoi32:
; SDAG: ## %bb.0:
; SDAG-NEXT: addl %esi, %edi
; SDAG-NEXT: setb %al
; SDAG-NEXT: movl %edi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddoi32:
; FAST: ## %bb.0:
; FAST-NEXT: addl %esi, %edi
; FAST-NEXT: setb %al
; FAST-NEXT: movl %edi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.uadd.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
store i32 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @uaddoi64(i64 %v1, i64 %v2, ptr %res) {
; SDAG-LABEL: uaddoi64:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq %rsi, %rdi
; SDAG-NEXT: setb %al
; SDAG-NEXT: movq %rdi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddoi64:
; FAST: ## %bb.0:
; FAST-NEXT: addq %rsi, %rdi
; FAST-NEXT: setb %al
; FAST-NEXT: movq %rdi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.uadd.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; UADDO reg, 1 | NOT INC
define zeroext i1 @uaddoinci8(i8 %v1, ptr %res) {
; SDAG-LABEL: uaddoinci8:
; SDAG: ## %bb.0:
; SDAG-NEXT: incb %dil
; SDAG-NEXT: sete %al
; SDAG-NEXT: movb %dil, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddoinci8:
; FAST: ## %bb.0:
; FAST-NEXT: addb $1, %dil
; FAST-NEXT: setb %al
; FAST-NEXT: movb %dil, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i8, i1} @llvm.uadd.with.overflow.i8(i8 %v1, i8 1)
%val = extractvalue {i8, i1} %t, 0
%obit = extractvalue {i8, i1} %t, 1
store i8 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @uaddoinci16(i16 %v1, ptr %res) {
; SDAG-LABEL: uaddoinci16:
; SDAG: ## %bb.0:
; SDAG-NEXT: incw %di
; SDAG-NEXT: sete %al
; SDAG-NEXT: movw %di, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddoinci16:
; FAST: ## %bb.0:
; FAST-NEXT: addw $1, %di
; FAST-NEXT: setb %al
; FAST-NEXT: movw %di, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i16, i1} @llvm.uadd.with.overflow.i16(i16 %v1, i16 1)
%val = extractvalue {i16, i1} %t, 0
%obit = extractvalue {i16, i1} %t, 1
store i16 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @uaddoinci32(i32 %v1, ptr %res) {
; SDAG-LABEL: uaddoinci32:
; SDAG: ## %bb.0:
; SDAG-NEXT: incl %edi
; SDAG-NEXT: sete %al
; SDAG-NEXT: movl %edi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddoinci32:
; FAST: ## %bb.0:
; FAST-NEXT: addl $1, %edi
; FAST-NEXT: setb %al
; FAST-NEXT: movl %edi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.uadd.with.overflow.i32(i32 %v1, i32 1)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
store i32 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @uaddoinci64(i64 %v1, ptr %res) {
; SDAG-LABEL: uaddoinci64:
; SDAG: ## %bb.0:
; SDAG-NEXT: incq %rdi
; SDAG-NEXT: sete %al
; SDAG-NEXT: movq %rdi, (%rsi)
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddoinci64:
; FAST: ## %bb.0:
; FAST-NEXT: addq $1, %rdi
; FAST-NEXT: setb %al
; FAST-NEXT: movq %rdi, (%rsi)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.uadd.with.overflow.i64(i64 %v1, i64 1)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; SSUBO
define zeroext i1 @ssuboi32(i32 %v1, i32 %v2, ptr %res) {
; SDAG-LABEL: ssuboi32:
; SDAG: ## %bb.0:
; SDAG-NEXT: subl %esi, %edi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movl %edi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: ssuboi32:
; FAST: ## %bb.0:
; FAST-NEXT: subl %esi, %edi
; FAST-NEXT: seto %al
; FAST-NEXT: movl %edi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.ssub.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
store i32 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @ssuboi64(i64 %v1, i64 %v2, ptr %res) {
; SDAG-LABEL: ssuboi64:
; SDAG: ## %bb.0:
; SDAG-NEXT: subq %rsi, %rdi
; SDAG-NEXT: seto %al
; SDAG-NEXT: movq %rdi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: ssuboi64:
; FAST: ## %bb.0:
; FAST-NEXT: subq %rsi, %rdi
; FAST-NEXT: seto %al
; FAST-NEXT: movq %rdi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.ssub.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
; USUBO
define zeroext i1 @usuboi32(i32 %v1, i32 %v2, ptr %res) {
; SDAG-LABEL: usuboi32:
; SDAG: ## %bb.0:
; SDAG-NEXT: subl %esi, %edi
; SDAG-NEXT: setb %al
; SDAG-NEXT: movl %edi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: usuboi32:
; FAST: ## %bb.0:
; FAST-NEXT: subl %esi, %edi
; FAST-NEXT: setb %al
; FAST-NEXT: movl %edi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.usub.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
store i32 %val, ptr %res
ret i1 %obit
}
define zeroext i1 @usuboi64(i64 %v1, i64 %v2, ptr %res) {
; SDAG-LABEL: usuboi64:
; SDAG: ## %bb.0:
; SDAG-NEXT: subq %rsi, %rdi
; SDAG-NEXT: setb %al
; SDAG-NEXT: movq %rdi, (%rdx)
; SDAG-NEXT: retq
;
; FAST-LABEL: usuboi64:
; FAST: ## %bb.0:
; FAST-NEXT: subq %rsi, %rdi
; FAST-NEXT: setb %al
; FAST-NEXT: movq %rdi, (%rdx)
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
store i64 %val, ptr %res
ret i1 %obit
}
;
; Check the use of the overflow bit in combination with a select instruction.
;
define i32 @saddoselecti32(i32 %v1, i32 %v2) {
; CHECK-LABEL: saddoselecti32:
; CHECK: ## %bb.0:
; CHECK-NEXT: movl %esi, %eax
; CHECK-NEXT: movl %edi, %ecx
; CHECK-NEXT: addl %esi, %ecx
; CHECK-NEXT: cmovol %edi, %eax
; CHECK-NEXT: retq
%t = call {i32, i1} @llvm.sadd.with.overflow.i32(i32 %v1, i32 %v2)
%obit = extractvalue {i32, i1} %t, 1
%ret = select i1 %obit, i32 %v1, i32 %v2
ret i32 %ret
}
define i64 @saddoselecti64(i64 %v1, i64 %v2) {
; CHECK-LABEL: saddoselecti64:
; CHECK: ## %bb.0:
; CHECK-NEXT: movq %rsi, %rax
; CHECK-NEXT: movq %rdi, %rcx
; CHECK-NEXT: addq %rsi, %rcx
; CHECK-NEXT: cmovoq %rdi, %rax
; CHECK-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 %v2)
%obit = extractvalue {i64, i1} %t, 1
%ret = select i1 %obit, i64 %v1, i64 %v2
ret i64 %ret
}
define i32 @uaddoselecti32(i32 %v1, i32 %v2) {
; CHECK-LABEL: uaddoselecti32:
; CHECK: ## %bb.0:
; CHECK-NEXT: movl %esi, %eax
; CHECK-NEXT: movl %edi, %ecx
; CHECK-NEXT: addl %esi, %ecx
; CHECK-NEXT: cmovbl %edi, %eax
; CHECK-NEXT: retq
%t = call {i32, i1} @llvm.uadd.with.overflow.i32(i32 %v1, i32 %v2)
%obit = extractvalue {i32, i1} %t, 1
%ret = select i1 %obit, i32 %v1, i32 %v2
ret i32 %ret
}
define i64 @uaddoselecti64(i64 %v1, i64 %v2) {
; CHECK-LABEL: uaddoselecti64:
; CHECK: ## %bb.0:
; CHECK-NEXT: movq %rsi, %rax
; CHECK-NEXT: movq %rdi, %rcx
; CHECK-NEXT: addq %rsi, %rcx
; CHECK-NEXT: cmovbq %rdi, %rax
; CHECK-NEXT: retq
%t = call {i64, i1} @llvm.uadd.with.overflow.i64(i64 %v1, i64 %v2)
%obit = extractvalue {i64, i1} %t, 1
%ret = select i1 %obit, i64 %v1, i64 %v2
ret i64 %ret
}
define i32 @ssuboselecti32(i32 %v1, i32 %v2) {
; CHECK-LABEL: ssuboselecti32:
; CHECK: ## %bb.0:
; CHECK-NEXT: movl %esi, %eax
; CHECK-NEXT: cmpl %esi, %edi
; CHECK-NEXT: cmovol %edi, %eax
; CHECK-NEXT: retq
%t = call {i32, i1} @llvm.ssub.with.overflow.i32(i32 %v1, i32 %v2)
%obit = extractvalue {i32, i1} %t, 1
%ret = select i1 %obit, i32 %v1, i32 %v2
ret i32 %ret
}
define i64 @ssuboselecti64(i64 %v1, i64 %v2) {
; CHECK-LABEL: ssuboselecti64:
; CHECK: ## %bb.0:
; CHECK-NEXT: movq %rsi, %rax
; CHECK-NEXT: cmpq %rsi, %rdi
; CHECK-NEXT: cmovoq %rdi, %rax
; CHECK-NEXT: retq
%t = call {i64, i1} @llvm.ssub.with.overflow.i64(i64 %v1, i64 %v2)
%obit = extractvalue {i64, i1} %t, 1
%ret = select i1 %obit, i64 %v1, i64 %v2
ret i64 %ret
}
define i32 @usuboselecti32(i32 %v1, i32 %v2) {
; CHECK-LABEL: usuboselecti32:
; CHECK: ## %bb.0:
; CHECK-NEXT: movl %esi, %eax
; CHECK-NEXT: cmpl %esi, %edi
; CHECK-NEXT: cmovbl %edi, %eax
; CHECK-NEXT: retq
%t = call {i32, i1} @llvm.usub.with.overflow.i32(i32 %v1, i32 %v2)
%obit = extractvalue {i32, i1} %t, 1
%ret = select i1 %obit, i32 %v1, i32 %v2
ret i32 %ret
}
define i64 @usuboselecti64(i64 %v1, i64 %v2) {
; CHECK-LABEL: usuboselecti64:
; CHECK: ## %bb.0:
; CHECK-NEXT: movq %rsi, %rax
; CHECK-NEXT: cmpq %rsi, %rdi
; CHECK-NEXT: cmovbq %rdi, %rax
; CHECK-NEXT: retq
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %v1, i64 %v2)
%obit = extractvalue {i64, i1} %t, 1
%ret = select i1 %obit, i64 %v1, i64 %v2
ret i64 %ret
}
;
; Check the use of the overflow bit in combination with a branch instruction.
;
define zeroext i1 @saddobri32(i32 %v1, i32 %v2) {
; SDAG-LABEL: saddobri32:
; SDAG: ## %bb.0:
; SDAG-NEXT: addl %esi, %edi
; SDAG-NEXT: jo LBB31_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB31_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: saddobri32:
; FAST: ## %bb.0:
; FAST-NEXT: addl %esi, %edi
; FAST-NEXT: jo LBB31_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB31_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.sadd.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @saddobri64(i64 %v1, i64 %v2) {
; SDAG-LABEL: saddobri64:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq %rsi, %rdi
; SDAG-NEXT: jo LBB32_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB32_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: saddobri64:
; FAST: ## %bb.0:
; FAST-NEXT: addq %rsi, %rdi
; FAST-NEXT: jo LBB32_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB32_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @uaddobri32(i32 %v1, i32 %v2) {
; SDAG-LABEL: uaddobri32:
; SDAG: ## %bb.0:
; SDAG-NEXT: addl %esi, %edi
; SDAG-NEXT: jb LBB33_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB33_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddobri32:
; FAST: ## %bb.0:
; FAST-NEXT: addl %esi, %edi
; FAST-NEXT: jb LBB33_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB33_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.uadd.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @uaddobri64(i64 %v1, i64 %v2) {
; SDAG-LABEL: uaddobri64:
; SDAG: ## %bb.0:
; SDAG-NEXT: addq %rsi, %rdi
; SDAG-NEXT: jb LBB34_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB34_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: uaddobri64:
; FAST: ## %bb.0:
; FAST-NEXT: addq %rsi, %rdi
; FAST-NEXT: jb LBB34_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB34_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.uadd.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @ssubobri32(i32 %v1, i32 %v2) {
; SDAG-LABEL: ssubobri32:
; SDAG: ## %bb.0:
; SDAG-NEXT: cmpl %esi, %edi
; SDAG-NEXT: jo LBB35_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB35_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: ssubobri32:
; FAST: ## %bb.0:
; FAST-NEXT: cmpl %esi, %edi
; FAST-NEXT: jo LBB35_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB35_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.ssub.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @ssubobri64(i64 %v1, i64 %v2) {
; SDAG-LABEL: ssubobri64:
; SDAG: ## %bb.0:
; SDAG-NEXT: cmpq %rsi, %rdi
; SDAG-NEXT: jo LBB36_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB36_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: ssubobri64:
; FAST: ## %bb.0:
; FAST-NEXT: cmpq %rsi, %rdi
; FAST-NEXT: jo LBB36_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB36_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.ssub.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @usubobri32(i32 %v1, i32 %v2) {
; SDAG-LABEL: usubobri32:
; SDAG: ## %bb.0:
; SDAG-NEXT: cmpl %esi, %edi
; SDAG-NEXT: jb LBB37_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB37_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: usubobri32:
; FAST: ## %bb.0:
; FAST-NEXT: cmpl %esi, %edi
; FAST-NEXT: jb LBB37_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB37_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i32, i1} @llvm.usub.with.overflow.i32(i32 %v1, i32 %v2)
%val = extractvalue {i32, i1} %t, 0
%obit = extractvalue {i32, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define zeroext i1 @usubobri64(i64 %v1, i64 %v2) {
; SDAG-LABEL: usubobri64:
; SDAG: ## %bb.0:
; SDAG-NEXT: cmpq %rsi, %rdi
; SDAG-NEXT: jb LBB38_1
; SDAG-NEXT: ## %bb.2: ## %continue
; SDAG-NEXT: movb $1, %al
; SDAG-NEXT: retq
; SDAG-NEXT: LBB38_1: ## %overflow
; SDAG-NEXT: xorl %eax, %eax
; SDAG-NEXT: retq
;
; FAST-LABEL: usubobri64:
; FAST: ## %bb.0:
; FAST-NEXT: cmpq %rsi, %rdi
; FAST-NEXT: jb LBB38_1
; FAST-NEXT: ## %bb.2: ## %continue
; FAST-NEXT: movb $1, %al
; FAST-NEXT: andb $1, %al
; FAST-NEXT: retq
; FAST-NEXT: LBB38_1: ## %overflow
; FAST-NEXT: xorl %eax, %eax
; FAST-NEXT: andb $1, %al
; FAST-NEXT: ## kill: def $al killed $al killed $eax
; FAST-NEXT: retq
%t = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %v1, i64 %v2)
%val = extractvalue {i64, i1} %t, 0
%obit = extractvalue {i64, i1} %t, 1
br i1 %obit, label %overflow, label %continue, !prof !0
overflow:
ret i1 false
continue:
ret i1 true
}
define {i64, i1} @saddoovf(i64 %a, i64 %b) {
; CHECK-LABEL: saddoovf:
; CHECK: ## %bb.0:
; CHECK-NEXT: sarq $17, %rdi
; CHECK-NEXT: shrq $31, %rsi
; CHECK-NEXT: leaq (%rsi,%rdi), %rax
; CHECK-NEXT: xorl %edx, %edx
; CHECK-NEXT: retq
%1 = ashr i64 %a, 17
%2 = lshr i64 %b, 31
%t = call {i64, i1} @llvm.sadd.with.overflow.i64(i64 %1, i64 %2)
ret {i64, i1} %t
}
define {i64, i1} @ssuboovf(i64 %a, i64 %b) {
; CHECK-LABEL: ssuboovf:
; CHECK: ## %bb.0:
; CHECK-NEXT: movzwl %di, %eax
; CHECK-NEXT: shrq $22, %rsi
; CHECK-NEXT: subq %rsi, %rax
; CHECK-NEXT: xorl %edx, %edx
; CHECK-NEXT: retq
%1 = and i64 %a, 65535
%2 = lshr i64 %b, 22
%t = call {i64, i1} @llvm.ssub.with.overflow.i64(i64 %1, i64 %2)
ret {i64, i1} %t
}
define {i64, i1} @uaddoovf(i64 %a, i64 %b) {
; CHECK-LABEL: uaddoovf:
; CHECK: ## %bb.0:
; CHECK-NEXT: movzbl %dil, %ecx
; CHECK-NEXT: movzbl %sil, %eax
; CHECK-NEXT: addq %rcx, %rax
; CHECK-NEXT: xorl %edx, %edx
; CHECK-NEXT: retq
%1 = and i64 %a, 255
%2 = and i64 %b, 255
%t = call {i64, i1} @llvm.uadd.with.overflow.i64(i64 %1, i64 %2)
ret {i64, i1} %t
}
define {i64, i1} @usuboovf(i64 %a, i64 %b) {
; CHECK-LABEL: usuboovf:
; CHECK: ## %bb.0:
; CHECK-NEXT: movq %rsi, %rax
; CHECK-NEXT: notq %rax
; CHECK-NEXT: xorl %edx, %edx
; CHECK-NEXT: retq
%t0 = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %a, i64 %a)
%v0 = extractvalue {i64, i1} %t0, 0
%o0 = extractvalue {i64, i1} %t0, 1
%t1 = call {i64, i1} @llvm.usub.with.overflow.i64(i64 -1, i64 %b)
%v1 = extractvalue {i64, i1} %t1, 0
%o1 = extractvalue {i64, i1} %t1, 1
%oo = or i1 %o0, %o1
%t2 = call {i64, i1} @llvm.usub.with.overflow.i64(i64 %v1, i64 %v0)
%v2 = extractvalue {i64, i1} %t2, 0
%o2 = extractvalue {i64, i1} %t2, 1
%ooo = or i1 %oo, %o2
%t = insertvalue {i64, i1} %t2, i1 %ooo, 1
ret {i64, i1} %t
}
; Make sure we select an INC for both the data use and the flag use.
define i32 @incovfselectstore(i32 %v1, i32 %v2, ptr %x) {
; CHECK-LABEL: incovfselectstore:
; CHECK: ## %bb.0:
; CHECK-NEXT: movl %esi, %eax
; CHECK-NEXT: movl %edi, %ecx
; CHECK-NEXT: incl %ecx
; CHECK-NEXT: cmovol %edi, %eax
; CHECK-NEXT: movl %ecx, (%rdx)
; CHECK-NEXT: retq
%t = call {i32, i1} @llvm.sadd.with.overflow.i32(i32 %v1, i32 1)
%obit = extractvalue {i32, i1} %t, 1
%ret = select i1 %obit, i32 %v1, i32 %v2
%val = extractvalue {i32, i1} %t, 0
store i32 %val, ptr %x
ret i32 %ret
}
; Make sure we select a DEC for both the data use and the flag use.
define i32 @decovfselectstore(i32 %v1, i32 %v2, ptr %x) {
; GENERIC-LABEL: decovfselectstore:
; GENERIC: ## %bb.0:
; GENERIC-NEXT: movl %esi, %eax
; GENERIC-NEXT: movl %edi, %ecx
; GENERIC-NEXT: decl %ecx
; GENERIC-NEXT: cmovol %edi, %eax
; GENERIC-NEXT: movl %ecx, (%rdx)
; GENERIC-NEXT: retq
;
; FAST-LABEL: decovfselectstore:
; FAST: ## %bb.0:
; FAST-NEXT: movl %esi, %eax
; FAST-NEXT: movl %edi, %ecx
; FAST-NEXT: decl %ecx
; FAST-NEXT: cmovol %edi, %eax
; FAST-NEXT: movl %ecx, (%rdx)
; FAST-NEXT: retq
;
; KNL-LABEL: decovfselectstore:
; KNL: ## %bb.0:
; KNL-NEXT: movl %esi, %eax
; KNL-NEXT: movl %edi, %ecx
; KNL-NEXT: addl $-1, %ecx
; KNL-NEXT: cmovol %edi, %eax
; KNL-NEXT: movl %ecx, (%rdx)
; KNL-NEXT: retq
%t = call {i32, i1} @llvm.ssub.with.overflow.i32(i32 %v1, i32 1)
%obit = extractvalue {i32, i1} %t, 1
%ret = select i1 %obit, i32 %v1, i32 %v2
%val = extractvalue {i32, i1} %t, 0
store i32 %val, ptr %x
ret i32 %ret
}
declare {i8, i1} @llvm.sadd.with.overflow.i8 (i8, i8 ) nounwind readnone
declare {i16, i1} @llvm.sadd.with.overflow.i16(i16, i16) nounwind readnone
declare {i32, i1} @llvm.sadd.with.overflow.i32(i32, i32) nounwind readnone
declare {i64, i1} @llvm.sadd.with.overflow.i64(i64, i64) nounwind readnone
declare {i8, i1} @llvm.uadd.with.overflow.i8 (i8, i8 ) nounwind readnone
declare {i16, i1} @llvm.uadd.with.overflow.i16(i16, i16) nounwind readnone
declare {i32, i1} @llvm.uadd.with.overflow.i32(i32, i32) nounwind readnone
declare {i64, i1} @llvm.uadd.with.overflow.i64(i64, i64) nounwind readnone
declare {i32, i1} @llvm.ssub.with.overflow.i32(i32, i32) nounwind readnone
declare {i64, i1} @llvm.ssub.with.overflow.i64(i64, i64) nounwind readnone
declare {i32, i1} @llvm.usub.with.overflow.i32(i32, i32) nounwind readnone
declare {i64, i1} @llvm.usub.with.overflow.i64(i64, i64) nounwind readnone
!0 = !{!"branch_weights", i32 0, i32 2147483647}