aboutsummaryrefslogtreecommitdiff
path: root/gcc/expr.c
diff options
context:
space:
mode:
authorIlya Enkovich <enkovich.gnu@gmail.com>2016-01-11 10:27:17 +0000
committerIlya Enkovich <ienkovich@gcc.gnu.org>2016-01-11 10:27:17 +0000
commit83834c094206928ec7a3c43850f5e04889da640e (patch)
treeae811e2413e2fbf7ab3c5f0df5dc2cf1b1af4dc4 /gcc/expr.c
parentdcf89d578058ed538e3dcf4e151d99a267e81bf4 (diff)
downloadgcc-83834c094206928ec7a3c43850f5e04889da640e.zip
gcc-83834c094206928ec7a3c43850f5e04889da640e.tar.gz
gcc-83834c094206928ec7a3c43850f5e04889da640e.tar.bz2
re PR target/69010 (Boolean vector constant with a scalar mode is expanded incorrectly)
gcc/ PR target/69010 * expr.c (expand_expr_real_1): For boolean vector constants with a scalar mode use const_scalar_mask_from_tree. (const_scalar_mask_from_tree): New. * optabs.c (expand_vec_cond_mask_expr): Use mask mode assigned to a mask type to handle constants. gcc/testsuite/ PR target/69010 * gcc.target/i386/pr69010.c: New test. From-SVN: r232216
Diffstat (limited to 'gcc/expr.c')
-rw-r--r--gcc/expr.c36
1 files changed, 33 insertions, 3 deletions
diff --git a/gcc/expr.c b/gcc/expr.c
index 0c5dff8..0a1c425 100644
--- a/gcc/expr.c
+++ b/gcc/expr.c
@@ -137,6 +137,7 @@ static void emit_single_push_insn (machine_mode, rtx, tree);
#endif
static void do_tablejump (rtx, machine_mode, rtx, rtx, rtx, int);
static rtx const_vector_from_tree (tree);
+static rtx const_scalar_mask_from_tree (tree);
static tree tree_expr_size (const_tree);
static HOST_WIDE_INT int_expr_size (tree);
@@ -9742,9 +9743,15 @@ expand_expr_real_1 (tree exp, rtx target, machine_mode tmode,
return const_vector_from_tree (exp);
if (GET_MODE_CLASS (mode) == MODE_INT)
{
- tree type_for_mode = lang_hooks.types.type_for_mode (mode, 1);
- if (type_for_mode)
- tmp = fold_unary_loc (loc, VIEW_CONVERT_EXPR, type_for_mode, exp);
+ if (VECTOR_BOOLEAN_TYPE_P (TREE_TYPE (exp)))
+ return const_scalar_mask_from_tree (exp);
+ else
+ {
+ tree type_for_mode = lang_hooks.types.type_for_mode (mode, 1);
+ if (type_for_mode)
+ tmp = fold_unary_loc (loc, VIEW_CONVERT_EXPR,
+ type_for_mode, exp);
+ }
}
if (!tmp)
{
@@ -11455,6 +11462,29 @@ const_vector_mask_from_tree (tree exp)
return gen_rtx_CONST_VECTOR (mode, v);
}
+/* Return a CONST_INT rtx representing vector mask for
+ a VECTOR_CST of booleans. */
+static rtx
+const_scalar_mask_from_tree (tree exp)
+{
+ machine_mode mode = TYPE_MODE (TREE_TYPE (exp));
+ wide_int res = wi::zero (GET_MODE_PRECISION (mode));
+ tree elt;
+ unsigned i;
+
+ for (i = 0; i < VECTOR_CST_NELTS (exp); ++i)
+ {
+ elt = VECTOR_CST_ELT (exp, i);
+ gcc_assert (TREE_CODE (elt) == INTEGER_CST);
+ if (integer_all_onesp (elt))
+ res = wi::set_bit (res, i);
+ else
+ gcc_assert (integer_zerop (elt));
+ }
+
+ return immed_wide_int_const (res, mode);
+}
+
/* Return a CONST_VECTOR rtx for a VECTOR_CST tree. */
static rtx
const_vector_from_tree (tree exp)