mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-26 11:25:27 +00:00
[ARM] MVE minnm and maxnm instructions
This adds the patterns for minnm and maxnm from the fminnum and fmaxnum nodes, similar to scalar types. Original patch by Simon Tatham Differential Revision: https://reviews.llvm.org/D63870 llvm-svn: 366002
This commit is contained in:
parent
2a7f520460
commit
ec8af0db6c
@ -281,6 +281,9 @@ void ARMTargetLowering::addMVEVectorTypes(bool HasMVEFP) {
|
||||
setOperationAction(ISD::SCALAR_TO_VECTOR, VT, Legal);
|
||||
|
||||
if (HasMVEFP) {
|
||||
setOperationAction(ISD::FMINNUM, VT, Legal);
|
||||
setOperationAction(ISD::FMAXNUM, VT, Legal);
|
||||
|
||||
// No native support for these.
|
||||
setOperationAction(ISD::FDIV, VT, Expand);
|
||||
setOperationAction(ISD::FREM, VT, Expand);
|
||||
@ -1254,10 +1257,12 @@ ARMTargetLowering::ARMTargetLowering(const TargetMachine &TM,
|
||||
setOperationAction(ISD::FRINT, MVT::f32, Legal);
|
||||
setOperationAction(ISD::FMINNUM, MVT::f32, Legal);
|
||||
setOperationAction(ISD::FMAXNUM, MVT::f32, Legal);
|
||||
setOperationAction(ISD::FMINNUM, MVT::v2f32, Legal);
|
||||
setOperationAction(ISD::FMAXNUM, MVT::v2f32, Legal);
|
||||
setOperationAction(ISD::FMINNUM, MVT::v4f32, Legal);
|
||||
setOperationAction(ISD::FMAXNUM, MVT::v4f32, Legal);
|
||||
if (Subtarget->hasNEON()) {
|
||||
setOperationAction(ISD::FMINNUM, MVT::v2f32, Legal);
|
||||
setOperationAction(ISD::FMAXNUM, MVT::v2f32, Legal);
|
||||
setOperationAction(ISD::FMINNUM, MVT::v4f32, Legal);
|
||||
setOperationAction(ISD::FMAXNUM, MVT::v4f32, Legal);
|
||||
}
|
||||
|
||||
if (Subtarget->hasFP64()) {
|
||||
setOperationAction(ISD::FFLOOR, MVT::f64, Legal);
|
||||
|
@ -863,9 +863,24 @@ class MVE_VMINMAXNM<string iname, string suffix, bit sz, bit bit_21,
|
||||
def MVE_VMAXNMf32 : MVE_VMINMAXNM<"vmaxnm", "f32", 0b0, 0b0>;
|
||||
def MVE_VMAXNMf16 : MVE_VMINMAXNM<"vmaxnm", "f16", 0b1, 0b0>;
|
||||
|
||||
let Predicates = [HasMVEFloat] in {
|
||||
def : Pat<(v4f32 (fmaxnum (v4f32 MQPR:$val1), (v4f32 MQPR:$val2))),
|
||||
(v4f32 (MVE_VMAXNMf32 (v4f32 MQPR:$val1), (v4f32 MQPR:$val2)))>;
|
||||
def : Pat<(v8f16 (fmaxnum (v8f16 MQPR:$val1), (v8f16 MQPR:$val2))),
|
||||
(v8f16 (MVE_VMAXNMf16 (v8f16 MQPR:$val1), (v8f16 MQPR:$val2)))>;
|
||||
}
|
||||
|
||||
def MVE_VMINNMf32 : MVE_VMINMAXNM<"vminnm", "f32", 0b0, 0b1>;
|
||||
def MVE_VMINNMf16 : MVE_VMINMAXNM<"vminnm", "f16", 0b1, 0b1>;
|
||||
|
||||
let Predicates = [HasMVEFloat] in {
|
||||
def : Pat<(v4f32 (fminnum (v4f32 MQPR:$val1), (v4f32 MQPR:$val2))),
|
||||
(v4f32 (MVE_VMINNMf32 (v4f32 MQPR:$val1), (v4f32 MQPR:$val2)))>;
|
||||
def : Pat<(v8f16 (fminnum (v8f16 MQPR:$val1), (v8f16 MQPR:$val2))),
|
||||
(v8f16 (MVE_VMINNMf16 (v8f16 MQPR:$val1), (v8f16 MQPR:$val2)))>;
|
||||
}
|
||||
|
||||
|
||||
class MVE_VMINMAX<string iname, string suffix, bit U, bits<2> size,
|
||||
bit bit_4, list<dag> pattern=[]>
|
||||
: MVE_comp<NoItinerary, iname, suffix, "", pattern> {
|
||||
|
95
llvm/test/CodeGen/Thumb2/mve-minmax.ll
Normal file
95
llvm/test/CodeGen/Thumb2/mve-minmax.ll
Normal file
@ -0,0 +1,95 @@
|
||||
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
|
||||
; RUN: llc -mtriple=thumbv8.1m.main-arm-none-eabi -mattr=+mve,+fullfp16 -verify-machineinstrs %s -o - | FileCheck %s --check-prefix=CHECK --check-prefix=CHECK-MVE
|
||||
; RUN: llc -mtriple=thumbv8.1m.main-arm-none-eabi -mattr=+mve.fp -verify-machineinstrs %s -o - | FileCheck %s --check-prefix=CHECK --check-prefix=CHECK-MVEFP
|
||||
|
||||
define arm_aapcs_vfpcc <4 x float> @maxnm_float32_t(<4 x float> %src1, <4 x float> %src2) {
|
||||
; CHECK-MVE-LABEL: maxnm_float32_t:
|
||||
; CHECK-MVE: @ %bb.0: @ %entry
|
||||
; CHECK-MVE-NEXT: vmaxnm.f32 s11, s7, s3
|
||||
; CHECK-MVE-NEXT: vmaxnm.f32 s10, s6, s2
|
||||
; CHECK-MVE-NEXT: vmaxnm.f32 s9, s5, s1
|
||||
; CHECK-MVE-NEXT: vmaxnm.f32 s8, s4, s0
|
||||
; CHECK-MVE-NEXT: vmov q0, q2
|
||||
; CHECK-MVE-NEXT: bx lr
|
||||
;
|
||||
; CHECK-MVEFP-LABEL: maxnm_float32_t:
|
||||
; CHECK-MVEFP: @ %bb.0: @ %entry
|
||||
; CHECK-MVEFP-NEXT: vmaxnm.f32 q0, q1, q0
|
||||
; CHECK-MVEFP-NEXT: bx lr
|
||||
entry:
|
||||
%cmp = fcmp fast ogt <4 x float> %src2, %src1
|
||||
%0 = select <4 x i1> %cmp, <4 x float> %src2, <4 x float> %src1
|
||||
ret <4 x float> %0
|
||||
}
|
||||
|
||||
define arm_aapcs_vfpcc <8 x half> @minnm_float16_t(<8 x half> %src1, <8 x half> %src2) {
|
||||
; CHECK-MVE-LABEL: minnm_float16_t:
|
||||
; CHECK-MVE: @ %bb.0: @ %entry
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[0]
|
||||
; CHECK-MVE-NEXT: vmov.u16 r1, q0[1]
|
||||
; CHECK-MVE-NEXT: vmov s8, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[0]
|
||||
; CHECK-MVE-NEXT: vmov s10, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r2, q1[1]
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s8, s10, s8
|
||||
; CHECK-MVE-NEXT: vmov s10, r2
|
||||
; CHECK-MVE-NEXT: vmov r0, s8
|
||||
; CHECK-MVE-NEXT: vmov s8, r1
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s8, s10, s8
|
||||
; CHECK-MVE-NEXT: vmov r1, s8
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[0], r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[2]
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[1], r1
|
||||
; CHECK-MVE-NEXT: vmov s12, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[2]
|
||||
; CHECK-MVE-NEXT: vmov s14, r0
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s12, s14, s12
|
||||
; CHECK-MVE-NEXT: vmov r0, s12
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[2], r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[3]
|
||||
; CHECK-MVE-NEXT: vmov s12, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[3]
|
||||
; CHECK-MVE-NEXT: vmov s14, r0
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s12, s14, s12
|
||||
; CHECK-MVE-NEXT: vmov r0, s12
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[3], r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[4]
|
||||
; CHECK-MVE-NEXT: vmov s12, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[4]
|
||||
; CHECK-MVE-NEXT: vmov s14, r0
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s12, s14, s12
|
||||
; CHECK-MVE-NEXT: vmov r0, s12
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[4], r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[5]
|
||||
; CHECK-MVE-NEXT: vmov s12, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[5]
|
||||
; CHECK-MVE-NEXT: vmov s14, r0
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s12, s14, s12
|
||||
; CHECK-MVE-NEXT: vmov r0, s12
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[5], r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[6]
|
||||
; CHECK-MVE-NEXT: vmov s12, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[6]
|
||||
; CHECK-MVE-NEXT: vmov s14, r0
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s12, s14, s12
|
||||
; CHECK-MVE-NEXT: vmov r0, s12
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[6], r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q0[7]
|
||||
; CHECK-MVE-NEXT: vmov s0, r0
|
||||
; CHECK-MVE-NEXT: vmov.u16 r0, q1[7]
|
||||
; CHECK-MVE-NEXT: vmov s2, r0
|
||||
; CHECK-MVE-NEXT: vminnm.f16 s0, s2, s0
|
||||
; CHECK-MVE-NEXT: vmov r0, s0
|
||||
; CHECK-MVE-NEXT: vmov.16 q2[7], r0
|
||||
; CHECK-MVE-NEXT: vmov q0, q2
|
||||
; CHECK-MVE-NEXT: bx lr
|
||||
;
|
||||
; CHECK-MVEFP-LABEL: minnm_float16_t:
|
||||
; CHECK-MVEFP: @ %bb.0: @ %entry
|
||||
; CHECK-MVEFP-NEXT: vminnm.f16 q0, q1, q0
|
||||
; CHECK-MVEFP-NEXT: bx lr
|
||||
entry:
|
||||
%cmp = fcmp fast ogt <8 x half> %src2, %src1
|
||||
%0 = select <8 x i1> %cmp, <8 x half> %src1, <8 x half> %src2
|
||||
ret <8 x half> %0
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user