From: Eric Botcazou Date: Wed, 5 Jun 2019 14:14:40 +0000 (+0000) Subject: fold-const.c (extract_muldiv_1): Do not distribute a multiplication by a power-of... X-Git-Url: https://git.libre-soc.org/?a=commitdiff_plain;h=c613138b6ce48448fa7f62dd03513d28919361d6;p=gcc.git fold-const.c (extract_muldiv_1): Do not distribute a multiplication by a power-of-two value. * fold-const.c (extract_muldiv_1) : Do not distribute a multiplication by a power-of-two value. (fold_plusminus_mult_expr): Use pow2p_hwi to spot a power-of-two value and turn the modulo operation into a masking operation. From-SVN: r271963 --- diff --git a/gcc/ChangeLog b/gcc/ChangeLog index 7c1bd52945f..2be354c9035 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,10 @@ +2019-06-05 Eric Botcazou + + * fold-const.c (extract_muldiv_1) : Do not distribute a + multiplication by a power-of-two value. + (fold_plusminus_mult_expr): Use pow2p_hwi to spot a power-of-two value + and turn the modulo operation into a masking operation. + 2019-06-05 Jakub Jelinek PR debug/90733 diff --git a/gcc/fold-const.c b/gcc/fold-const.c index 3e066a2ecc4..8252cc6bb6b 100644 --- a/gcc/fold-const.c +++ b/gcc/fold-const.c @@ -6475,8 +6475,12 @@ extract_muldiv_1 (tree t, tree c, enum tree_code code, tree wide_type, apply the distributive law to commute the multiply and addition if the multiplication of the constants doesn't overflow and overflow is defined. With undefined overflow - op0 * c might overflow, while (op0 + orig_op1) * c doesn't. */ - if (code == MULT_EXPR && TYPE_OVERFLOW_WRAPS (ctype)) + op0 * c might overflow, while (op0 + orig_op1) * c doesn't. + But fold_plusminus_mult_expr would factor back any power-of-two + value so do not distribute in the first place in this case. */ + if (code == MULT_EXPR + && TYPE_OVERFLOW_WRAPS (ctype) + && !(tree_fits_shwi_p (c) && pow2p_hwi (absu_hwi (tree_to_shwi (c))))) return fold_build2 (tcode, ctype, fold_build2 (code, ctype, fold_convert (ctype, op0), @@ -7124,14 +7128,13 @@ fold_plusminus_mult_expr (location_t loc, enum tree_code code, tree type, /* No identical multiplicands; see if we can find a common power-of-two factor in non-power-of-two multiplies. This can help in multi-dimensional array access. */ - else if (tree_fits_shwi_p (arg01) - && tree_fits_shwi_p (arg11)) + else if (tree_fits_shwi_p (arg01) && tree_fits_shwi_p (arg11)) { - HOST_WIDE_INT int01, int11, tmp; + HOST_WIDE_INT int01 = tree_to_shwi (arg01); + HOST_WIDE_INT int11 = tree_to_shwi (arg11); + HOST_WIDE_INT tmp; bool swap = false; tree maybe_same; - int01 = tree_to_shwi (arg01); - int11 = tree_to_shwi (arg11); /* Move min of absolute values to int11. */ if (absu_hwi (int01) < absu_hwi (int11)) @@ -7144,7 +7147,10 @@ fold_plusminus_mult_expr (location_t loc, enum tree_code code, tree type, else maybe_same = arg11; - if (exact_log2 (absu_hwi (int11)) > 0 && int01 % int11 == 0 + const unsigned HOST_WIDE_INT factor = absu_hwi (int11); + if (factor > 1 + && pow2p_hwi (factor) + && (int01 & (factor - 1)) == 0 /* The remainder should not be a constant, otherwise we end up folding i * 4 + 2 to (i * 2 + 1) * 2 which has increased the number of multiplications necessary. */ diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog index 0d120c422a2..0a88fed79de 100644 --- a/gcc/testsuite/ChangeLog +++ b/gcc/testsuite/ChangeLog @@ -1,3 +1,7 @@ +2019-06-05 Eric Botcazou + + * gnat.dg/specs/discr6.ads: New test. + 2019-06-05 Sam Tebbs * gcc.target/aarch64/return_address_sign_b_1.c: New file. diff --git a/gcc/testsuite/gnat.dg/specs/discr6.ads b/gcc/testsuite/gnat.dg/specs/discr6.ads new file mode 100644 index 00000000000..af16553694a --- /dev/null +++ b/gcc/testsuite/gnat.dg/specs/discr6.ads @@ -0,0 +1,21 @@ +-- { dg-do compile } + +package Discr6 is + + subtype Index_T is Integer range 0 .. 15; + + type Arr is array (Index_T range <> ) of Long_Long_Integer; + + type Rec2 (Size : Index_T := 2) is record + A : Arr (2 .. Size); + end record; + + type Rec3 (D : Boolean := False) is record + R : Rec2; + case D is + when False=> null; + when True => I : Integer; + end case; + end record; + +end Discr6;