From b617fc71010a3806aa03dfc3bb5cf581b2b9d95c Mon Sep 17 00:00:00 2001 From: Julian Brown Date: Thu, 26 Jul 2007 13:59:57 +0000 Subject: [PATCH] gcc/ * config/arm/arm-protos.h (neon_lane_bounds, neon_const_bounds) (neon_element_bits): Add prototypes. * config/arm/arm.c (bounds_check, neon_lane_bounds) (neon_const_bounds, neon_element_bits): New functions. * config/arm/neon.md (neon_vget_lane, neon_vget_lanedi) (neon_vget_lanev2di, neon_vset_lane, neon_vset_lanedi) (neon_vset_lanev2di, neon_vdup_lane, neon_vdup_lanedi) (neon_vdup_lanev2di, neon_vcvt_n, neon_vmul_lane) (neon_vmull_lane, neon_vqdmull_lane) (neon_vqdmulh_lane, neon_vmla_lane) (neon_vmlal_lane, neon_vqdmlal_lane) (neon_vmls_lane, neon_vmlsl_lane) (neon_vqdmlsl_lane, neon_vext, neon_vshr_n) (neon_vshrn_n, neon_vqshrn_n, neon_vqshrun_n) (neon_vshl_n, neon_vshl_n, neon_vqshl_n) (neon_vqshlu_n, neon_vshll_n, neon_vsra_n) (neon_vsri_n, neon_vsli_n): Add bounds checks. From-SVN: r126958 --- gcc/ChangeLog | 20 ++++ gcc/config/arm/arm-protos.h | 3 + gcc/config/arm/arm.c | 43 +++++++++ gcc/config/arm/neon.md | 184 +++++++++++++++++++++++++++++------- 4 files changed, 215 insertions(+), 35 deletions(-) diff --git a/gcc/ChangeLog b/gcc/ChangeLog index f78914868199..017187c3b187 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,23 @@ +2007-07-26 Julian Brown + + * config/arm/arm-protos.h (neon_lane_bounds, neon_const_bounds) + (neon_element_bits): Add prototypes. + * config/arm/arm.c (bounds_check, neon_lane_bounds) + (neon_const_bounds, neon_element_bits): New functions. + * config/arm/neon.md (neon_vget_lane, neon_vget_lanedi) + (neon_vget_lanev2di, neon_vset_lane, neon_vset_lanedi) + (neon_vset_lanev2di, neon_vdup_lane, neon_vdup_lanedi) + (neon_vdup_lanev2di, neon_vcvt_n, neon_vmul_lane) + (neon_vmull_lane, neon_vqdmull_lane) + (neon_vqdmulh_lane, neon_vmla_lane) + (neon_vmlal_lane, neon_vqdmlal_lane) + (neon_vmls_lane, neon_vmlsl_lane) + (neon_vqdmlsl_lane, neon_vext, neon_vshr_n) + (neon_vshrn_n, neon_vqshrn_n, neon_vqshrun_n) + (neon_vshl_n, neon_vshl_n, neon_vqshl_n) + (neon_vqshlu_n, neon_vshll_n, neon_vsra_n) + (neon_vsri_n, neon_vsli_n): Add bounds checks. + 2007-07-26 Nathan Froyd * config/vxworks.h (VXWORKS_LINK_SPEC): Fix typo. diff --git a/gcc/config/arm/arm-protos.h b/gcc/config/arm/arm-protos.h index f2380264eee6..f7439f24da6a 100644 --- a/gcc/config/arm/arm-protos.h +++ b/gcc/config/arm/arm-protos.h @@ -76,6 +76,9 @@ extern char *neon_output_logic_immediate (const char *, rtx *, extern void neon_pairwise_reduce (rtx, rtx, enum machine_mode, rtx (*) (rtx, rtx, rtx)); extern void neon_expand_vector_init (rtx, rtx); +extern void neon_lane_bounds (rtx, HOST_WIDE_INT, HOST_WIDE_INT); +extern void neon_const_bounds (rtx, HOST_WIDE_INT, HOST_WIDE_INT); +extern HOST_WIDE_INT neon_element_bits (enum machine_mode); extern void neon_reinterpret (rtx, rtx); extern void neon_emit_pair_result_insn (enum machine_mode, rtx (*) (rtx, rtx, rtx, rtx), diff --git a/gcc/config/arm/arm.c b/gcc/config/arm/arm.c index de0fb41c3094..47f339450364 100644 --- a/gcc/config/arm/arm.c +++ b/gcc/config/arm/arm.c @@ -6234,6 +6234,49 @@ neon_expand_vector_init (rtx target, rtx vals) emit_move_insn (target, mem); } +/* Ensure OPERAND lies between LOW (inclusive) and HIGH (exclusive). Raise + ERR if it doesn't. FIXME: NEON bounds checks occur late in compilation, so + reported source locations are bogus. */ + +static void +bounds_check (rtx operand, HOST_WIDE_INT low, HOST_WIDE_INT high, + const char *err) +{ + HOST_WIDE_INT lane; + + gcc_assert (GET_CODE (operand) == CONST_INT); + + lane = INTVAL (operand); + + if (lane < low || lane >= high) + error (err); +} + +/* Bounds-check lanes. */ + +void +neon_lane_bounds (rtx operand, HOST_WIDE_INT low, HOST_WIDE_INT high) +{ + bounds_check (operand, low, high, "lane out of range"); +} + +/* Bounds-check constants. */ + +void +neon_const_bounds (rtx operand, HOST_WIDE_INT low, HOST_WIDE_INT high) +{ + bounds_check (operand, low, high, "constant out of range"); +} + +HOST_WIDE_INT +neon_element_bits (enum machine_mode mode) +{ + if (mode == DImode) + return GET_MODE_BITSIZE (mode); + else + return GET_MODE_BITSIZE (GET_MODE_INNER (mode)); +} + /* Predicates for `match_operand' and `match_operator'. */ diff --git a/gcc/config/arm/neon.md b/gcc/config/arm/neon.md index c62ffc3cff67..06b9b3c762ea 100644 --- a/gcc/config/arm/neon.md +++ b/gcc/config/arm/neon.md @@ -2395,7 +2395,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VGET_LANE))] "TARGET_NEON" - "vmov%?.%t3%#\t%0, %P1[%c2]" +{ + neon_lane_bounds (operands[2], 0, GET_MODE_NUNITS (mode)); + return "vmov%?.%t3%#\t%0, %P1[%c2]"; +} [(set_attr "predicable" "yes") (set_attr "neon_type" "neon_bp_simple")] ) @@ -2411,7 +2414,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VGET_LANE))] "TARGET_NEON" - "vmov%?\t%Q0, %R0, %P1 @ di" +{ + neon_lane_bounds (operands[2], 0, 1); + return "vmov%?\t%Q0, %R0, %P1 @ di"; +} [(set_attr "predicable" "yes") (set_attr "neon_type" "neon_bp_simple")] ) @@ -2429,6 +2435,8 @@ unsigned int halfelts = GET_MODE_NUNITS (mode) / 2; unsigned int elt = INTVAL (operands[2]); + neon_lane_bounds (operands[2], 0, halfelts * 2); + ops[0] = operands[0]; ops[1] = gen_rtx_REG (mode, regno + 2 * (elt / halfelts)); ops[2] = GEN_INT (elt % halfelts); @@ -2453,6 +2461,8 @@ unsigned int regno = REGNO (operands[1]); unsigned int elt = INTVAL (operands[2]); + neon_lane_bounds (operands[2], 0, 2); + ops[0] = operands[0]; ops[1] = gen_rtx_REG (DImode, regno + 2 * elt); output_asm_insn ("vmov%?\t%Q0, %R0, %P1 @ v2di", ops); @@ -2470,7 +2480,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSET_LANE))] "TARGET_NEON" - "vmov%?.\t%P0[%c3], %1" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vmov%?.\t%P0[%c3], %1"; +} [(set_attr "predicable" "yes") (set_attr "neon_type" "neon_bp_simple")] ) @@ -2484,7 +2497,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSET_LANE))] "TARGET_NEON" - "vmov%?\t%P0, %Q1, %R1 @ di" +{ + neon_lane_bounds (operands[3], 0, 1); + return "vmov%?\t%P0, %Q1, %R1 @ di"; +} [(set_attr "predicable" "yes") (set_attr "neon_type" "neon_bp_simple")] ) @@ -2502,6 +2518,8 @@ unsigned int halfelts = GET_MODE_NUNITS (mode) / 2; unsigned int elt = INTVAL (operands[3]); + neon_lane_bounds (operands[3], 0, halfelts * 2); + ops[0] = gen_rtx_REG (mode, regno + 2 * (elt / halfelts)); ops[1] = operands[1]; ops[2] = GEN_INT (elt % halfelts); @@ -2525,6 +2543,8 @@ unsigned int regno = REGNO (operands[0]); unsigned int elt = INTVAL (operands[3]); + neon_lane_bounds (operands[3], 0, 2); + ops[0] = gen_rtx_REG (DImode, regno + 2 * elt); ops[1] = operands[1]; output_asm_insn ("vmov%?\t%P0, %Q1, %R1 @ v2di", ops); @@ -2583,7 +2603,10 @@ (match_operand:SI 2 "immediate_operand" "i")] UNSPEC_VDUP_LANE))] "TARGET_NEON" - "vdup.\t%P0, %P1[%c2]" +{ + neon_lane_bounds (operands[2], 0, GET_MODE_NUNITS (mode)); + return "vdup.\t%P0, %P1[%c2]"; +} ;; Assume this schedules like vmov. [(set_attr "neon_type" "neon_bp_simple")] ) @@ -2594,7 +2617,10 @@ (match_operand:SI 2 "immediate_operand" "i")] UNSPEC_VDUP_LANE))] "TARGET_NEON" - "vdup.\t%q0, %P1[%c2]" +{ + neon_lane_bounds (operands[2], 0, GET_MODE_NUNITS (mode)); + return "vdup.\t%q0, %P1[%c2]"; +} ;; Assume this schedules like vmov. [(set_attr "neon_type" "neon_bp_simple")] ) @@ -2607,6 +2633,7 @@ UNSPEC_VDUP_LANE))] "TARGET_NEON" { + neon_lane_bounds (operands[2], 0, 1); emit_move_insn (operands[0], operands[1]); DONE; }) @@ -2618,7 +2645,10 @@ (match_operand:SI 2 "immediate_operand" "i")] UNSPEC_VDUP_LANE))] "TARGET_NEON" - "vmov\t%e0, %P1\;vmov\t%f0, %P1" +{ + neon_lane_bounds (operands[2], 0, 1); + return "vmov\t%e0, %P1\;vmov\t%f0, %P1"; +} [(set_attr "length" "8") (set_attr "neon_type" "neon_bp_simple")] ) @@ -2741,7 +2771,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VCVT_N))] "TARGET_NEON" - "vcvt.%T3%#32.f32\t%0, %1, %2" +{ + neon_const_bounds (operands[2], 1, 33); + return "vcvt.%T3%#32.f32\t%0, %1, %2"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vadd_ddd_vabs_dd") @@ -2755,7 +2788,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VCVT_N))] "TARGET_NEON" - "vcvt.f32.%T3%#32\t%0, %1, %2" +{ + neon_const_bounds (operands[2], 1, 33); + return "vcvt.f32.%T3%#32\t%0, %1, %2"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vadd_ddd_vabs_dd") @@ -2811,7 +2847,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VMUL_LANE))] "TARGET_NEON" - "vmul.\t%P0, %P1, %P2[%c3]" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vmul.\t%P0, %P1, %P2[%c3]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vmul_ddd") @@ -2829,7 +2868,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VMUL_LANE))] "TARGET_NEON" - "vmul.\t%q0, %q1, %P2[%c3]" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vmul.\t%q0, %q1, %P2[%c3]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vmul_qqd") @@ -2847,7 +2889,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VMULL_LANE))] "TARGET_NEON" - "vmull.%T4%#\t%q0, %P1, %P2[%c3]" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vmull.%T4%#\t%q0, %P1, %P2[%c3]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mul_ddd_16_scalar_32_16_long_scalar") @@ -2863,7 +2908,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VQDMULL_LANE))] "TARGET_NEON" - "vqdmull.\t%q0, %P1, %P2[%c3]" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vqdmull.\t%q0, %P1, %P2[%c3]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mul_ddd_16_scalar_32_16_long_scalar") @@ -2879,7 +2927,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VQDMULH_LANE))] "TARGET_NEON" - "vq%O4dmulh.%T4%#\t%q0, %q1, %P2[%c3]" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vq%O4dmulh.%T4%#\t%q0, %q1, %P2[%c3]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mul_qdd_64_32_long_qqd_16_ddd_32_scalar_64_32_long_scalar") @@ -2895,7 +2946,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VQDMULH_LANE))] "TARGET_NEON" - "vq%O4dmulh.%T4%#\t%P0, %P1, %P2[%c3]" +{ + neon_lane_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vq%O4dmulh.%T4%#\t%P0, %P1, %P2[%c3]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mul_ddd_16_scalar_32_16_long_scalar") @@ -2912,7 +2966,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VMLA_LANE))] "TARGET_NEON" - "vmla.\t%P0, %P2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vmla.\t%P0, %P2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vmla_ddd_scalar") @@ -2931,7 +2988,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VMLA_LANE))] "TARGET_NEON" - "vmla.\t%q0, %q2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vmla.\t%q0, %q2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vmla_qqq_scalar") @@ -2950,7 +3010,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VMLAL_LANE))] "TARGET_NEON" - "vmlal.%T5%#\t%q0, %P2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vmlal.%T5%#\t%q0, %P2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mla_ddd_16_scalar_qdd_32_16_long_scalar") @@ -2967,7 +3030,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VQDMLAL_LANE))] "TARGET_NEON" - "vqdmlal.\t%q0, %P2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vqdmlal.\t%q0, %P2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mla_ddd_16_scalar_qdd_32_16_long_scalar") @@ -2984,7 +3050,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VMLS_LANE))] "TARGET_NEON" - "vmls.\t%P0, %P2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vmls.\t%P0, %P2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vmla_ddd_scalar") @@ -3003,7 +3072,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VMLS_LANE))] "TARGET_NEON" - "vmls.\t%q0, %q2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vmls.\t%q0, %q2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_fp_vmla_qqq_scalar") @@ -3022,7 +3094,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VMLSL_LANE))] "TARGET_NEON" - "vmlsl.%T5%#\t%q0, %P2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vmlsl.%T5%#\t%q0, %P2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mla_ddd_16_scalar_qdd_32_16_long_scalar") @@ -3039,7 +3114,10 @@ (match_operand:SI 5 "immediate_operand" "i")] UNSPEC_VQDMLSL_LANE))] "TARGET_NEON" - "vqdmlsl.\t%q0, %P2, %P3[%c4]" +{ + neon_lane_bounds (operands[4], 0, GET_MODE_NUNITS (mode)); + return "vqdmlsl.\t%q0, %P2, %P3[%c4]"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_mla_ddd_16_scalar_qdd_32_16_long_scalar") @@ -3264,7 +3342,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VEXT))] "TARGET_NEON" - "vext.\t%0, %1, %2, %3" +{ + neon_const_bounds (operands[3], 0, GET_MODE_NUNITS (mode)); + return "vext.\t%0, %1, %2, %3"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_bp_simple") @@ -3370,7 +3451,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSHR_N))] "TARGET_NEON" - "v%O3shr.%T3%#\t%0, %1, %2" +{ + neon_const_bounds (operands[2], 1, neon_element_bits (mode) + 1); + return "v%O3shr.%T3%#\t%0, %1, %2"; +} [(set_attr "neon_type" "neon_shift_1")] ) @@ -3381,7 +3465,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSHRN_N))] "TARGET_NEON" - "v%O3shrn.\t%P0, %q1, %2" +{ + neon_const_bounds (operands[2], 1, neon_element_bits (mode) / 2 + 1); + return "v%O3shrn.\t%P0, %q1, %2"; +} [(set_attr "neon_type" "neon_shift_1")] ) @@ -3392,7 +3479,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VQSHRN_N))] "TARGET_NEON" - "vq%O3shrn.%T3%#\t%P0, %q1, %2" +{ + neon_const_bounds (operands[2], 1, neon_element_bits (mode) / 2 + 1); + return "vq%O3shrn.%T3%#\t%P0, %q1, %2"; +} [(set_attr "neon_type" "neon_shift_2")] ) @@ -3403,7 +3493,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VQSHRUN_N))] "TARGET_NEON" - "vq%O3shrun.%T3%#\t%P0, %q1, %2" +{ + neon_const_bounds (operands[2], 1, neon_element_bits (mode) / 2 + 1); + return "vq%O3shrun.%T3%#\t%P0, %q1, %2"; +} [(set_attr "neon_type" "neon_shift_2")] ) @@ -3414,7 +3507,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSHL_N))] "TARGET_NEON" - "vshl.\t%0, %1, %2" +{ + neon_const_bounds (operands[2], 0, neon_element_bits (mode)); + return "vshl.\t%0, %1, %2"; +} [(set_attr "neon_type" "neon_shift_1")] ) @@ -3425,7 +3521,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VQSHL_N))] "TARGET_NEON" - "vqshl.%T3%#\t%0, %1, %2" +{ + neon_const_bounds (operands[2], 0, neon_element_bits (mode)); + return "vqshl.%T3%#\t%0, %1, %2"; +} [(set_attr "neon_type" "neon_shift_2")] ) @@ -3436,7 +3535,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VQSHLU_N))] "TARGET_NEON" - "vqshlu.%T3%#\t%0, %1, %2" +{ + neon_const_bounds (operands[2], 0, neon_element_bits (mode)); + return "vqshlu.%T3%#\t%0, %1, %2"; +} [(set_attr "neon_type" "neon_shift_2")] ) @@ -3447,7 +3549,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSHLL_N))] "TARGET_NEON" - "vshll.%T3%#\t%q0, %P1, %2" +{ + neon_const_bounds (operands[2], 0, neon_element_bits (mode)); + return "vshll.%T3%#\t%q0, %P1, %2"; +} [(set_attr "neon_type" "neon_shift_1")] ) @@ -3459,7 +3564,10 @@ (match_operand:SI 4 "immediate_operand" "i")] UNSPEC_VSRA_N))] "TARGET_NEON" - "v%O4sra.%T4%#\t%0, %2, %3" +{ + neon_const_bounds (operands[3], 1, neon_element_bits (mode) + 1); + return "v%O4sra.%T4%#\t%0, %2, %3"; +} [(set_attr "neon_type" "neon_vsra_vrsra")] ) @@ -3470,7 +3578,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSRI))] "TARGET_NEON" - "vsri.\t%0, %2, %3" +{ + neon_const_bounds (operands[3], 1, neon_element_bits (mode) + 1); + return "vsri.\t%0, %2, %3"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_shift_1") @@ -3484,7 +3595,10 @@ (match_operand:SI 3 "immediate_operand" "i")] UNSPEC_VSLI))] "TARGET_NEON" - "vsli.\t%0, %2, %3" +{ + neon_const_bounds (operands[3], 0, neon_element_bits (mode)); + return "vsli.\t%0, %2, %3"; +} [(set (attr "neon_type") (if_then_else (ne (symbol_ref "") (const_int 0)) (const_string "neon_shift_1")