Index: llvm/lib/Target/ARM/ARMInstrMVE.td =================================================================== --- llvm/lib/Target/ARM/ARMInstrMVE.td +++ llvm/lib/Target/ARM/ARMInstrMVE.td @@ -3621,21 +3621,24 @@ let validForTailPredication = 1; } -multiclass MVE_VMULT_fp_m { +multiclass MVE_VMULT_fp_m { def "" : MVE_VMUL_fp; defvar Inst = !cast(NAME); let Predicates = [HasMVEFloat] in { - defm : MVE_TwoOpPattern(NAME)>; + defm : MVE_TwoOpPattern(NAME), IdentityVec>; } } -multiclass MVE_VMUL_fp_m - : MVE_VMULT_fp_m<"vmul", VTI, fmul, int_arm_mve_mul_predicated>; +multiclass MVE_VMUL_fp_m + : MVE_VMULT_fp_m<"vmul", VTI, fmul, int_arm_mve_mul_predicated, IdentityVec>; + +def ARMimmOneF: PatLeaf<(bitconvert (v4f32 (ARMvmovFPImm (i32 112))))>; // 1.0 float +def ARMimmOneH: PatLeaf<(bitconvert (v8i16 (ARMvmovImm (i32 2620))))>; // 1.0 half -defm MVE_VMULf32 : MVE_VMUL_fp_m; -defm MVE_VMULf16 : MVE_VMUL_fp_m; +defm MVE_VMULf32 : MVE_VMUL_fp_m; +defm MVE_VMULf16 : MVE_VMUL_fp_m; class MVE_VCMLA size> : MVEFloatArithNeon<"vcmla", suffix, size{1}, (outs MQPR:$Qd), @@ -3747,27 +3750,30 @@ defm MVE_VFMSf16 : MVE_VFMA_fp_multi<"vfms", 1, MVE_v8f16>; multiclass MVE_VADDSUB_fp_m { + SDNode Op, Intrinsic PredInt, SDPatternOperator IdentityVec> { def "" : MVE_VADDSUBFMA_fp { let validForTailPredication = 1; } defvar Inst = !cast(NAME); let Predicates = [HasMVEFloat] in { - defm : MVE_TwoOpPattern(NAME)>; + defm : MVE_TwoOpPattern(NAME), IdentityVec>; } } -multiclass MVE_VADD_fp_m - : MVE_VADDSUB_fp_m<"vadd", 0, VTI, fadd, int_arm_mve_add_predicated>; -multiclass MVE_VSUB_fp_m - : MVE_VADDSUB_fp_m<"vsub", 1, VTI, fsub, int_arm_mve_sub_predicated>; +multiclass MVE_VADD_fp_m + : MVE_VADDSUB_fp_m<"vadd", 0, VTI, fadd, int_arm_mve_add_predicated, IdentityVec>; +multiclass MVE_VSUB_fp_m + : MVE_VADDSUB_fp_m<"vsub", 1, VTI, fsub, int_arm_mve_sub_predicated, IdentityVec>; -defm MVE_VADDf32 : MVE_VADD_fp_m; -defm MVE_VADDf16 : MVE_VADD_fp_m; +def ARMimmMinusZeroF: PatLeaf<(bitconvert (v4i32 (ARMvmovImm (i32 1664))))>; // -0.0 float +def ARMimmMinusZeroH: PatLeaf<(bitconvert (v8i16 (ARMvmovImm (i32 2688))))>; // -0.0 half -defm MVE_VSUBf32 : MVE_VSUB_fp_m; -defm MVE_VSUBf16 : MVE_VSUB_fp_m; +defm MVE_VADDf32 : MVE_VADD_fp_m; +defm MVE_VADDf16 : MVE_VADD_fp_m; + +defm MVE_VSUBf32 : MVE_VSUB_fp_m; +defm MVE_VSUBf16 : MVE_VSUB_fp_m; class MVE_VCADD size, string cstr=""> : MVEFloatArithNeon<"vcadd", suffix, size{1}, (outs MQPR:$Qd), @@ -5373,22 +5379,22 @@ defm MVE_VHSUB_qr_u32 : MVE_VHSUB_qr_m; multiclass MVE_VADDSUB_qr_f { + SDNode Op, Intrinsic PredInt, SDPatternOperator IdentityVec> { def "" : MVE_VxADDSUB_qr; defm : MVE_TwoOpPatternDup(NAME)>; + !cast(NAME), IdentityVec>; } let Predicates = [HasMVEFloat] in { defm MVE_VADD_qr_f32 : MVE_VADDSUB_qr_f<"vadd", MVE_v4f32, 0b0, fadd, - int_arm_mve_add_predicated>; + int_arm_mve_add_predicated, ARMimmMinusZeroF>; defm MVE_VADD_qr_f16 : MVE_VADDSUB_qr_f<"vadd", MVE_v8f16, 0b0, fadd, - int_arm_mve_add_predicated>; + int_arm_mve_add_predicated, ARMimmMinusZeroH>; defm MVE_VSUB_qr_f32 : MVE_VADDSUB_qr_f<"vsub", MVE_v4f32, 0b1, fsub, - int_arm_mve_sub_predicated>; + int_arm_mve_sub_predicated, ARMimmAllZerosV>; defm MVE_VSUB_qr_f16 : MVE_VADDSUB_qr_f<"vsub", MVE_v8f16, 0b1, fsub, - int_arm_mve_sub_predicated>; + int_arm_mve_sub_predicated, ARMimmAllZerosV>; } class MVE_VxSHL_qr size, @@ -5567,16 +5573,16 @@ defm MVE_VQRDMULH_qr_s16 : MVE_VQRDMULH_qr_m; defm MVE_VQRDMULH_qr_s32 : MVE_VQRDMULH_qr_m; -multiclass MVE_VxxMUL_qr_f_m { +multiclass MVE_VxxMUL_qr_f_m { let validForTailPredication = 1 in def "" : MVE_VxxMUL_qr<"vmul", VTI.Suffix, VTI.Size{0}, 0b11, VTI.Size>; defm : MVE_TwoOpPatternDup(NAME)>; + !cast(NAME), IdentityVec>; } let Predicates = [HasMVEFloat] in { - defm MVE_VMUL_qr_f16 : MVE_VxxMUL_qr_f_m; - defm MVE_VMUL_qr_f32 : MVE_VxxMUL_qr_f_m; + defm MVE_VMUL_qr_f16 : MVE_VxxMUL_qr_f_m; + defm MVE_VMUL_qr_f32 : MVE_VxxMUL_qr_f_m; } class MVE_VFMAMLA_qr @fadd_v4f32_x(<4 x float> %x, <4 x float> %y, i32 %n) { ; CHECK-LABEL: fadd_v4f32_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i32 q2, #0x80000000 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q1 -; CHECK-NEXT: vadd.f32 q0, q2, q0 +; CHECK-NEXT: vaddt.f32 q0, q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -368,11 +366,9 @@ define arm_aapcs_vfpcc <8 x half> @fadd_v8f16_x(<8 x half> %x, <8 x half> %y, i32 %n) { ; CHECK-LABEL: fadd_v8f16_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i16 q2, #0x8000 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q1 -; CHECK-NEXT: vadd.f16 q0, q2, q0 +; CHECK-NEXT: vaddt.f16 q0, q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -384,11 +380,9 @@ define arm_aapcs_vfpcc <4 x float> @fsub_v4f32_x(<4 x float> %x, <4 x float> %y, i32 %n) { ; CHECK-LABEL: fsub_v4f32_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i32 q2, #0x0 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q1 -; CHECK-NEXT: vsub.f32 q0, q0, q2 +; CHECK-NEXT: vsubt.f32 q0, q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -400,11 +394,9 @@ define arm_aapcs_vfpcc <8 x half> @fsub_v8f16_x(<8 x half> %x, <8 x half> %y, i32 %n) { ; CHECK-LABEL: fsub_v8f16_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i32 q2, #0x0 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q1 -; CHECK-NEXT: vsub.f16 q0, q0, q2 +; CHECK-NEXT: vsubt.f16 q0, q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -416,11 +408,9 @@ define arm_aapcs_vfpcc <4 x float> @fmul_v4f32_x(<4 x float> %x, <4 x float> %y, i32 %n) { ; CHECK-LABEL: fmul_v4f32_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.f32 q2, #1.000000e+00 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q1 -; CHECK-NEXT: vmul.f32 q0, q2, q0 +; CHECK-NEXT: vmult.f32 q0, q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -432,11 +422,9 @@ define arm_aapcs_vfpcc <8 x half> @fmul_v8f16_x(<8 x half> %x, <8 x half> %y, i32 %n) { ; CHECK-LABEL: fmul_v8f16_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i16 q2, #0x3c00 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q1 -; CHECK-NEXT: vmul.f16 q0, q2, q0 +; CHECK-NEXT: vmult.f16 q0, q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -1000,12 +988,10 @@ define arm_aapcs_vfpcc <4 x float> @faddqr_v4f32_x(<4 x float> %x, float %y, i32 %n) { ; CHECK-LABEL: faddqr_v4f32_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i32 q2, #0x80000000 ; CHECK-NEXT: vmov r1, s4 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vdupt.32 q2, r1 -; CHECK-NEXT: vadd.f32 q0, q2, q0 +; CHECK-NEXT: vaddt.f32 q0, q0, r1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -1020,11 +1006,9 @@ ; CHECK-LABEL: faddqr_v8f16_x: ; CHECK: @ %bb.0: @ %entry ; CHECK-NEXT: vmov.f16 r1, s4 -; CHECK-NEXT: vmov.i16 q1, #0x8000 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vdupt.16 q1, r1 -; CHECK-NEXT: vadd.f16 q0, q1, q0 +; CHECK-NEXT: vaddt.f16 q0, q0, r1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -1038,12 +1022,10 @@ define arm_aapcs_vfpcc <4 x float> @fsubqr_v4f32_x(<4 x float> %x, float %y, i32 %n) { ; CHECK-LABEL: fsubqr_v4f32_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i32 q2, #0x0 ; CHECK-NEXT: vmov r1, s4 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vdupt.32 q2, r1 -; CHECK-NEXT: vsub.f32 q0, q0, q2 +; CHECK-NEXT: vsubt.f32 q0, q0, r1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -1058,11 +1040,9 @@ ; CHECK-LABEL: fsubqr_v8f16_x: ; CHECK: @ %bb.0: @ %entry ; CHECK-NEXT: vmov.f16 r1, s4 -; CHECK-NEXT: vmov.i32 q1, #0x0 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vdupt.16 q1, r1 -; CHECK-NEXT: vsub.f16 q0, q0, q1 +; CHECK-NEXT: vsubt.f16 q0, q0, r1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -1076,12 +1056,10 @@ define arm_aapcs_vfpcc <4 x float> @fmulqr_v4f32_x(<4 x float> %x, float %y, i32 %n) { ; CHECK-LABEL: fmulqr_v4f32_x: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.f32 q2, #1.000000e+00 ; CHECK-NEXT: vmov r1, s4 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vdupt.32 q2, r1 -; CHECK-NEXT: vmul.f32 q0, q2, q0 +; CHECK-NEXT: vmult.f32 q0, q0, r1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -1096,11 +1074,9 @@ ; CHECK-LABEL: fmulqr_v8f16_x: ; CHECK: @ %bb.0: @ %entry ; CHECK-NEXT: vmov.f16 r1, s4 -; CHECK-NEXT: vmov.i16 q1, #0x3c00 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vdupt.16 q1, r1 -; CHECK-NEXT: vmul.f16 q0, q1, q0 +; CHECK-NEXT: vmult.f16 q0, q0, r1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -1672,11 +1648,10 @@ define arm_aapcs_vfpcc <4 x float> @fadd_v4f32_y(<4 x float> %x, <4 x float> %y, i32 %n) { ; CHECK-LABEL: fadd_v4f32_y: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i32 q2, #0x80000000 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vadd.f32 q0, q2, q1 +; CHECK-NEXT: vaddt.f32 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -1688,11 +1663,10 @@ define arm_aapcs_vfpcc <8 x half> @fadd_v8f16_y(<8 x half> %x, <8 x half> %y, i32 %n) { ; CHECK-LABEL: fadd_v8f16_y: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i16 q2, #0x8000 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vadd.f16 q0, q2, q1 +; CHECK-NEXT: vaddt.f16 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -1734,11 +1708,10 @@ define arm_aapcs_vfpcc <4 x float> @fmul_v4f32_y(<4 x float> %x, <4 x float> %y, i32 %n) { ; CHECK-LABEL: fmul_v4f32_y: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.f32 q2, #1.000000e+00 ; CHECK-NEXT: vctp.32 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vmul.f32 q0, q2, q1 +; CHECK-NEXT: vmult.f32 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -1750,11 +1723,10 @@ define arm_aapcs_vfpcc <8 x half> @fmul_v8f16_y(<8 x half> %x, <8 x half> %y, i32 %n) { ; CHECK-LABEL: fmul_v8f16_y: ; CHECK: @ %bb.0: @ %entry -; CHECK-NEXT: vmov.i16 q2, #0x3c00 ; CHECK-NEXT: vctp.16 r0 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vmul.f16 q0, q2, q1 +; CHECK-NEXT: vmult.f16 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -2380,12 +2352,12 @@ define arm_aapcs_vfpcc <4 x float> @faddqr_v4f32_y(<4 x float> %x, float %y, i32 %n) { ; CHECK-LABEL: faddqr_v4f32_y: ; CHECK: @ %bb.0: @ %entry +; CHECK-NEXT: vmov r1, s4 ; CHECK-NEXT: vctp.32 r0 -; CHECK-NEXT: vmov.i32 q2, #0x80000000 +; CHECK-NEXT: vdup.32 q1, r1 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vmov r0, s4 -; CHECK-NEXT: vadd.f32 q0, q2, r0 +; CHECK-NEXT: vaddt.f32 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -2399,12 +2371,12 @@ define arm_aapcs_vfpcc <8 x half> @faddqr_v8f16_y(<8 x half> %x, half %y, i32 %n) { ; CHECK-LABEL: faddqr_v8f16_y: ; CHECK: @ %bb.0: @ %entry +; CHECK-NEXT: vmov.f16 r1, s4 ; CHECK-NEXT: vctp.16 r0 -; CHECK-NEXT: vmov.i16 q2, #0x8000 +; CHECK-NEXT: vdup.16 q1, r1 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vmov.f16 r0, s4 -; CHECK-NEXT: vadd.f16 q0, q2, r0 +; CHECK-NEXT: vaddt.f16 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n) @@ -2456,12 +2428,12 @@ define arm_aapcs_vfpcc <4 x float> @fmulqr_v4f32_y(<4 x float> %x, float %y, i32 %n) { ; CHECK-LABEL: fmulqr_v4f32_y: ; CHECK: @ %bb.0: @ %entry +; CHECK-NEXT: vmov r1, s4 ; CHECK-NEXT: vctp.32 r0 -; CHECK-NEXT: vmov.f32 q2, #1.000000e+00 +; CHECK-NEXT: vdup.32 q1, r1 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vmov r0, s4 -; CHECK-NEXT: vmul.f32 q0, q2, r0 +; CHECK-NEXT: vmult.f32 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <4 x i1> @llvm.arm.mve.vctp32(i32 %n) @@ -2475,12 +2447,12 @@ define arm_aapcs_vfpcc <8 x half> @fmulqr_v8f16_y(<8 x half> %x, half %y, i32 %n) { ; CHECK-LABEL: fmulqr_v8f16_y: ; CHECK: @ %bb.0: @ %entry +; CHECK-NEXT: vmov.f16 r1, s4 ; CHECK-NEXT: vctp.16 r0 -; CHECK-NEXT: vmov.i16 q2, #0x3c00 +; CHECK-NEXT: vdup.16 q1, r1 ; CHECK-NEXT: vpst -; CHECK-NEXT: vmovt q2, q0 -; CHECK-NEXT: vmov.f16 r0, s4 -; CHECK-NEXT: vmul.f16 q0, q2, r0 +; CHECK-NEXT: vmult.f16 q1, q1, q0 +; CHECK-NEXT: vmov q0, q1 ; CHECK-NEXT: bx lr entry: %c = call <8 x i1> @llvm.arm.mve.vctp16(i32 %n)