2017-04-11 18:52:34 +08:00
|
|
|
; RUN: llc -mtriple arm-linux-gnueabihf -mattr=+vfp2 -float-abi=hard -global-isel %s -o - | FileCheck %s -check-prefix CHECK -check-prefix HARD
|
|
|
|
; RUN: llc -mtriple arm-linux-gnueabi -mattr=+vfp2,+soft-float -float-abi=soft -global-isel %s -o - | FileCheck %s -check-prefix CHECK -check-prefix SOFT-AEABI
|
|
|
|
; RUN: llc -mtriple arm-linux-gnu- -mattr=+vfp2,+soft-float -float-abi=soft -global-isel %s -o - | FileCheck %s -check-prefix CHECK -check-prefix SOFT-DEFAULT
|
2017-04-07 17:41:39 +08:00
|
|
|
|
|
|
|
define arm_aapcscc float @test_frem_float(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: test_frem_float:
|
2017-10-25 19:42:40 +08:00
|
|
|
; CHECK: bl fmodf
|
2017-04-07 17:41:39 +08:00
|
|
|
%r = frem float %x, %y
|
|
|
|
ret float %r
|
|
|
|
}
|
|
|
|
|
2017-04-07 18:50:02 +08:00
|
|
|
define arm_aapcscc double @test_frem_double(double %x, double %y) {
|
|
|
|
; CHECK-LABEL: test_frem_double:
|
2017-10-25 19:42:40 +08:00
|
|
|
; CHECK: bl fmod
|
2017-04-07 18:50:02 +08:00
|
|
|
%r = frem double %x, %y
|
|
|
|
ret double %r
|
|
|
|
}
|
2017-04-10 17:27:39 +08:00
|
|
|
|
|
|
|
declare float @llvm.pow.f32(float %x, float %y)
|
|
|
|
define arm_aapcscc float @test_fpow_float(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: test_fpow_float:
|
2017-10-25 19:42:40 +08:00
|
|
|
; CHECK: bl powf
|
2017-04-10 17:27:39 +08:00
|
|
|
%r = call float @llvm.pow.f32(float %x, float %y)
|
|
|
|
ret float %r
|
|
|
|
}
|
|
|
|
|
|
|
|
declare double @llvm.pow.f64(double %x, double %y)
|
|
|
|
define arm_aapcscc double @test_fpow_double(double %x, double %y) {
|
|
|
|
; CHECK-LABEL: test_fpow_double:
|
2017-10-25 19:42:40 +08:00
|
|
|
; CHECK: bl pow
|
2017-04-10 17:27:39 +08:00
|
|
|
%r = call double @llvm.pow.f64(double %x, double %y)
|
|
|
|
ret double %r
|
|
|
|
}
|
2017-04-11 18:52:34 +08:00
|
|
|
|
|
|
|
define arm_aapcscc float @test_add_float(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: test_add_float:
|
|
|
|
; HARD: vadd.f32
|
2017-10-25 19:42:40 +08:00
|
|
|
; SOFT-AEABI: bl __aeabi_fadd
|
|
|
|
; SOFT-DEFAULT: bl __addsf3
|
2017-04-11 18:52:34 +08:00
|
|
|
%r = fadd float %x, %y
|
|
|
|
ret float %r
|
|
|
|
}
|
|
|
|
|
|
|
|
define arm_aapcscc double @test_add_double(double %x, double %y) {
|
|
|
|
; CHECK-LABEL: test_add_double:
|
|
|
|
; HARD: vadd.f64
|
2017-10-25 19:42:40 +08:00
|
|
|
; SOFT-AEABI: bl __aeabi_dadd
|
|
|
|
; SOFT-DEFAULT: bl __adddf3
|
2017-04-11 18:52:34 +08:00
|
|
|
%r = fadd double %x, %y
|
|
|
|
ret double %r
|
|
|
|
}
|
[ARM] GlobalISel: Select hard G_FCMP for s32
We lower to a sequence consisting of:
- MOVi 0 into a register
- VCMPS to do the actual comparison and set the VFP flags
- FMSTAT to move the flags out of the VFP unit
- MOVCCi to either use the "zero register" that we have previously set
with the MOVi, or move 1 into the result register, based on the values
of the flags
As was the case with soft-float, for some predicates (one, ueq) we
actually need two comparisons instead of just one. When that happens, we
generate two VCMPS-FMSTAT-MOVCCi sequences and chain them by means of
using the result of the first MOVCCi as the "zero register" for the
second one. This is a bit overkill, since one comparison followed by
two non-flag-setting conditional moves should be enough. In any case,
the backend manages to CSE one of the comparisons away so it doesn't
matter much.
Note that unlike SelectionDAG and FastISel, we always use VCMPS, and not
VCMPES. This makes the code a lot simpler, and it also seems correct
since the LLVM Lang Ref defines simple true/false returns if the
operands are QNaN's. For SNaN's, even VCMPS throws an Invalid Operand
exception, so they won't be slipping through unnoticed.
Implementation-wise, this introduces a template so we can share the same
code that we use for handling integer comparisons, since the only
differences are in the details (exact opcodes to be used etc). Hopefully
this will be easy to extend to s64 G_FCMP.
llvm-svn: 307365
2017-07-07 16:39:04 +08:00
|
|
|
|
|
|
|
define arm_aapcs_vfpcc i32 @test_cmp_float_ogt(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: test_cmp_float_ogt
|
|
|
|
; HARD: vcmp.f32
|
|
|
|
; HARD: vmrs APSR_nzcv, fpscr
|
|
|
|
; HARD-NEXT: movgt
|
2017-10-25 19:42:40 +08:00
|
|
|
; SOFT-AEABI: bl __aeabi_fcmpgt
|
|
|
|
; SOFT-DEFAULT: bl __gtsf2
|
[ARM] GlobalISel: Select hard G_FCMP for s32
We lower to a sequence consisting of:
- MOVi 0 into a register
- VCMPS to do the actual comparison and set the VFP flags
- FMSTAT to move the flags out of the VFP unit
- MOVCCi to either use the "zero register" that we have previously set
with the MOVi, or move 1 into the result register, based on the values
of the flags
As was the case with soft-float, for some predicates (one, ueq) we
actually need two comparisons instead of just one. When that happens, we
generate two VCMPS-FMSTAT-MOVCCi sequences and chain them by means of
using the result of the first MOVCCi as the "zero register" for the
second one. This is a bit overkill, since one comparison followed by
two non-flag-setting conditional moves should be enough. In any case,
the backend manages to CSE one of the comparisons away so it doesn't
matter much.
Note that unlike SelectionDAG and FastISel, we always use VCMPS, and not
VCMPES. This makes the code a lot simpler, and it also seems correct
since the LLVM Lang Ref defines simple true/false returns if the
operands are QNaN's. For SNaN's, even VCMPS throws an Invalid Operand
exception, so they won't be slipping through unnoticed.
Implementation-wise, this introduces a template so we can share the same
code that we use for handling integer comparisons, since the only
differences are in the details (exact opcodes to be used etc). Hopefully
this will be easy to extend to s64 G_FCMP.
llvm-svn: 307365
2017-07-07 16:39:04 +08:00
|
|
|
entry:
|
|
|
|
%v = fcmp ogt float %x, %y
|
|
|
|
%r = zext i1 %v to i32
|
|
|
|
ret i32 %r
|
|
|
|
}
|
|
|
|
|
|
|
|
define arm_aapcs_vfpcc i32 @test_cmp_float_one(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: test_cmp_float_one
|
|
|
|
; HARD: vcmp.f32
|
|
|
|
; HARD: vmrs APSR_nzcv, fpscr
|
|
|
|
; HARD: movgt
|
|
|
|
; HARD-NOT: vcmp
|
|
|
|
; HARD: movmi
|
2017-10-25 19:42:40 +08:00
|
|
|
; SOFT-AEABI-DAG: bl __aeabi_fcmpgt
|
|
|
|
; SOFT-AEABI-DAG: bl __aeabi_fcmplt
|
|
|
|
; SOFT-DEFAULT-DAG: bl __gtsf2
|
|
|
|
; SOFT-DEFAULT-DAG: bl __ltsf2
|
[ARM] GlobalISel: Select hard G_FCMP for s32
We lower to a sequence consisting of:
- MOVi 0 into a register
- VCMPS to do the actual comparison and set the VFP flags
- FMSTAT to move the flags out of the VFP unit
- MOVCCi to either use the "zero register" that we have previously set
with the MOVi, or move 1 into the result register, based on the values
of the flags
As was the case with soft-float, for some predicates (one, ueq) we
actually need two comparisons instead of just one. When that happens, we
generate two VCMPS-FMSTAT-MOVCCi sequences and chain them by means of
using the result of the first MOVCCi as the "zero register" for the
second one. This is a bit overkill, since one comparison followed by
two non-flag-setting conditional moves should be enough. In any case,
the backend manages to CSE one of the comparisons away so it doesn't
matter much.
Note that unlike SelectionDAG and FastISel, we always use VCMPS, and not
VCMPES. This makes the code a lot simpler, and it also seems correct
since the LLVM Lang Ref defines simple true/false returns if the
operands are QNaN's. For SNaN's, even VCMPS throws an Invalid Operand
exception, so they won't be slipping through unnoticed.
Implementation-wise, this introduces a template so we can share the same
code that we use for handling integer comparisons, since the only
differences are in the details (exact opcodes to be used etc). Hopefully
this will be easy to extend to s64 G_FCMP.
llvm-svn: 307365
2017-07-07 16:39:04 +08:00
|
|
|
entry:
|
|
|
|
%v = fcmp one float %x, %y
|
|
|
|
%r = zext i1 %v to i32
|
|
|
|
ret i32 %r
|
|
|
|
}
|