mirror of
https://github.com/RPCSX/llvm.git
synced 2024-12-15 16:09:02 +00:00
1c6611db44
The current peephole optimizing for compare inst assumes an instr that uses CPSR has an MO for ARM Cond code.However, for VSEL instructions (vseqeq, vselgt, vselgt, vselvs), there is no such operand nor do they support the modification of Cond Code. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@196588 91177308-0d34-0410-b5e6-96231b3b80d8
147 lines
3.3 KiB
LLVM
147 lines
3.3 KiB
LLVM
; RUN: llc < %s -mtriple=arm-apple-darwin | FileCheck %s
|
|
; RUN: llc < %s -mtriple=arm-apple-darwin | FileCheck %s --check-prefix=V7
|
|
; RUN: llc < %s -mtriple=armv8-none-linux-gnueabi | FileCheck %s -check-prefix=V8
|
|
|
|
|
|
define i32 @f(i32 %a, i32 %b) nounwind ssp {
|
|
entry:
|
|
; CHECK-LABEL: f:
|
|
; CHECK: subs
|
|
; CHECK-NOT: cmp
|
|
%cmp = icmp sgt i32 %a, %b
|
|
%sub = sub nsw i32 %a, %b
|
|
%sub. = select i1 %cmp, i32 %sub, i32 0
|
|
ret i32 %sub.
|
|
}
|
|
|
|
define i32 @g(i32 %a, i32 %b) nounwind ssp {
|
|
entry:
|
|
; CHECK-LABEL: g:
|
|
; CHECK: subs
|
|
; CHECK-NOT: cmp
|
|
%cmp = icmp slt i32 %a, %b
|
|
%sub = sub nsw i32 %b, %a
|
|
%sub. = select i1 %cmp, i32 %sub, i32 0
|
|
ret i32 %sub.
|
|
}
|
|
|
|
define i32 @h(i32 %a, i32 %b) nounwind ssp {
|
|
entry:
|
|
; CHECK-LABEL: h:
|
|
; CHECK: subs
|
|
; CHECK-NOT: cmp
|
|
%cmp = icmp sgt i32 %a, 3
|
|
%sub = sub nsw i32 %a, 3
|
|
%sub. = select i1 %cmp, i32 %sub, i32 %b
|
|
ret i32 %sub.
|
|
}
|
|
|
|
; rdar://11725965
|
|
define i32 @i(i32 %a, i32 %b) nounwind readnone ssp {
|
|
entry:
|
|
; CHECK-LABEL: i:
|
|
; CHECK: subs
|
|
; CHECK-NOT: cmp
|
|
%cmp = icmp ult i32 %a, %b
|
|
%sub = sub i32 %b, %a
|
|
%sub. = select i1 %cmp, i32 %sub, i32 0
|
|
ret i32 %sub.
|
|
}
|
|
; If CPSR is live-out, we can't remove cmp if there exists
|
|
; a swapped sub.
|
|
define i32 @j(i32 %a, i32 %b) nounwind {
|
|
entry:
|
|
; CHECK-LABEL: j:
|
|
; CHECK: sub
|
|
; CHECK: cmp
|
|
%cmp = icmp eq i32 %b, %a
|
|
%sub = sub nsw i32 %a, %b
|
|
br i1 %cmp, label %if.then, label %if.else
|
|
|
|
if.then:
|
|
%cmp2 = icmp sgt i32 %b, %a
|
|
%sel = select i1 %cmp2, i32 %sub, i32 %a
|
|
ret i32 %sel
|
|
|
|
if.else:
|
|
ret i32 %sub
|
|
}
|
|
|
|
; If the sub/rsb instruction is predicated, we can't use the flags.
|
|
; <rdar://problem/12263428>
|
|
; Test case from MultiSource/Benchmarks/Ptrdist/bc/number.s
|
|
; CHECK: bc_raise
|
|
; CHECK: rsbeq
|
|
; CHECK: cmp
|
|
define i32 @bc_raise() nounwind ssp {
|
|
entry:
|
|
%val.2.i = select i1 undef, i32 0, i32 undef
|
|
%sub.i = sub nsw i32 0, %val.2.i
|
|
%retval.0.i = select i1 undef, i32 %val.2.i, i32 %sub.i
|
|
%cmp1 = icmp eq i32 %retval.0.i, 0
|
|
br i1 %cmp1, label %land.lhs.true, label %if.end11
|
|
|
|
land.lhs.true: ; preds = %num2long.exit
|
|
ret i32 17
|
|
|
|
if.end11: ; preds = %num2long.exit
|
|
ret i32 23
|
|
}
|
|
|
|
define float @float_sel(i32 %a, i32 %b, float %x, float %y) {
|
|
entry:
|
|
; CHECK-LABEL: float_sel:
|
|
; CHECK-NOT: cmp
|
|
; V8-LABEL: float_sel:
|
|
; V8-NOT: cmp
|
|
; V8: vseleq.f32
|
|
%sub = sub i32 %a, %b
|
|
%cmp = icmp eq i32 %sub, 0
|
|
%ret = select i1 %cmp, float %x, float %y
|
|
ret float %ret
|
|
}
|
|
|
|
define double @double_sel(i32 %a, i32 %b, double %x, double %y) {
|
|
entry:
|
|
; CHECK-LABEL: double_sel:
|
|
; CHECK-NOT: cmp
|
|
; V8-LABEL: double_sel:
|
|
; V8-NOT: cmp
|
|
; V8: vseleq.f64
|
|
%sub = sub i32 %a, %b
|
|
%cmp = icmp eq i32 %sub, 0
|
|
%ret = select i1 %cmp, double %x, double %y
|
|
ret double %ret
|
|
}
|
|
|
|
@t = common global i32 0
|
|
define double @double_sub(i32 %a, i32 %b, double %x, double %y) {
|
|
entry:
|
|
; CHECK-LABEL: double_sub:
|
|
; CHECK: subs
|
|
; CHECK-NOT: cmp
|
|
; V8-LABEL: double_sub:
|
|
; V8: vsel
|
|
%cmp = icmp sgt i32 %a, %b
|
|
%sub = sub i32 %a, %b
|
|
store i32 %sub, i32* @t
|
|
%ret = select i1 %cmp, double %x, double %y
|
|
ret double %ret
|
|
}
|
|
|
|
define double @double_sub_swap(i32 %a, i32 %b, double %x, double %y) {
|
|
entry:
|
|
; V7-LABEL: double_sub_swap:
|
|
; V7-NOT: cmp
|
|
; V7: subs
|
|
; V8-LABEL: double_sub_swap:
|
|
; V8-NOT: subs
|
|
; V8: cmp
|
|
; V8: vsel
|
|
%cmp = icmp sgt i32 %a, %b
|
|
%sub = sub i32 %b, %a
|
|
%ret = select i1 %cmp, double %x, double %y
|
|
store i32 %sub, i32* @t
|
|
ret double %ret
|
|
}
|