On Tue, Oct 2, 2012 at 3:01 PM, Jakub Jelinek <ja...@redhat.com> wrote:
> Hi!
>
> As discussed in the PR and on IRC, this patch verifies that vector
> CONSTRUCTOR in GIMPLE is either empty CONSTRUCTOR, or contains scalar
> elements of type compatible with vector element type (then the verification
> is less strict, allows less than TYPE_VECTOR_SUBPARTS elements and allows
> non-NULL indexes if they are consecutive (no holes); this is because
> from FEs often CONSTRUCTORs with those properties leak into the IL, and
> a change in the gimplifier to canonicalize them wasn't enough, they keep
> leaking even from non-gimplified DECL_INITIAL values etc.), or
> contains vector elements (element types must be compatible, the vector
> elements must be of the same type and their number must fill the whole
> wider vector - these are created/used by tree-vect-generic lowering if
> HW supports only shorter vectors than what is requested in source).
>
> Bootstrapped/regtested on x86_64-linux and i686-linux, ok for trunk?

Ok with ...

> 2012-10-02  Jakub Jelinek  <ja...@redhat.com>
>
>         PR tree-optimization/54713
>         * expr.c (categorize_ctor_elements_1): Don't assume purpose is
>         non-NULL.
>         * tree-cfg.c (verify_gimple_assign_single): Add verification of
>         vector CONSTRUCTORs.
>         * tree-ssa-sccvn.c (vn_reference_lookup_3): For VECTOR_TYPE
>         CONSTRUCTORs, don't do anything if element type is VECTOR_TYPE,
>         and don't check index.
>         * tree-vect-slp.c (vect_get_constant_vectors): VIEW_CONVERT_EXPR
>         ctor elements first if their type isn't compatible with vector
>         element type.
>
> --- gcc/expr.c.jj       2012-09-27 12:45:53.000000000 +0200
> +++ gcc/expr.c  2012-10-01 18:21:40.885122833 +0200
> @@ -5491,7 +5491,7 @@ categorize_ctor_elements_1 (const_tree c
>      {
>        HOST_WIDE_INT mult = 1;
>
> -      if (TREE_CODE (purpose) == RANGE_EXPR)
> +      if (purpose && TREE_CODE (purpose) == RANGE_EXPR)
>         {
>           tree lo_index = TREE_OPERAND (purpose, 0);
>           tree hi_index = TREE_OPERAND (purpose, 1);
> --- gcc/tree-cfg.c.jj   2012-10-01 17:28:17.469921927 +0200
> +++ gcc/tree-cfg.c      2012-10-02 11:24:11.686155889 +0200
> @@ -4000,6 +4000,80 @@ verify_gimple_assign_single (gimple stmt
>        return res;
>
>      case CONSTRUCTOR:
> +      if (TREE_CODE (rhs1_type) == VECTOR_TYPE)
> +       {
> +         unsigned int i;
> +         tree elt_i, elt_v, elt_t = NULL_TREE;
> +
> +         if (CONSTRUCTOR_NELTS (rhs1) == 0)
> +           return res;
> +         /* For vector CONSTRUCTORs we require that either it is empty
> +            CONSTRUCTOR, or it is a CONSTRUCTOR of smaller vector elements
> +            (then the element count must be correct to cover the whole
> +            outer vector and index must be NULL on all elements, or it is
> +            a CONSTRUCTOR of scalar elements, where we as an exception allow
> +            smaller number of elements (assuming zero filling) and
> +            consecutive indexes as compared to NULL indexes (such
> +            CONSTRUCTORs can appear in the IL from FEs).  */
> +         FOR_EACH_CONSTRUCTOR_ELT (CONSTRUCTOR_ELTS (rhs1), i, elt_i, elt_v)
> +           {
> +             if (elt_t == NULL_TREE)
> +               {
> +                 elt_t = TREE_TYPE (elt_v);
> +                 if (TREE_CODE (elt_t) == VECTOR_TYPE)
> +                   {
> +                     tree elt_t = TREE_TYPE (elt_v);
> +                     if (!useless_type_conversion_p (TREE_TYPE (rhs1_type),
> +                                                     TREE_TYPE (elt_t)))
> +                       {
> +                         error ("incorrect type of vector CONSTRUCTOR"
> +                                " elements");
> +                         debug_generic_stmt (rhs1);
> +                         return true;
> +                       }
> +                     else if (CONSTRUCTOR_NELTS (rhs1)
> +                              * TYPE_VECTOR_SUBPARTS (elt_t)
> +                              != TYPE_VECTOR_SUBPARTS (rhs1_type))
> +                       {
> +                         error ("incorrect number of vector CONSTRUCTOR"
> +                                " elements");
> +                         debug_generic_stmt (rhs1);
> +                         return true;
> +                       }
> +                   }
> +                 else if (!useless_type_conversion_p (TREE_TYPE (rhs1_type),
> +                                                      elt_t))
> +                   {
> +                     error ("incorrect type of vector CONSTRUCTOR elements");
> +                     debug_generic_stmt (rhs1);
> +                     return true;
> +                   }
> +                 else if (CONSTRUCTOR_NELTS (rhs1)
> +                          > TYPE_VECTOR_SUBPARTS (rhs1_type))
> +                   {
> +                     error ("incorrect number of vector CONSTRUCTOR 
> elements");
> +                     debug_generic_stmt (rhs1);
> +                     return true;
> +                   }
> +               }
> +             else if (!useless_type_conversion_p (elt_t, TREE_TYPE (elt_v)))
> +               {
> +                 error ("incorrect type of vector CONSTRUCTOR elements");
> +                 debug_generic_stmt (rhs1);
> +                 return true;
> +               }
> +             if (elt_i != NULL_TREE
> +                 && (TREE_CODE (elt_t) == VECTOR_TYPE
> +                     || TREE_CODE (elt_i) != INTEGER_CST
> +                     || compare_tree_int (elt_i, i) != 0))
> +               {
> +                 error ("vector CONSTRUCTOR with non-NULL element index");
> +                 debug_generic_stmt (rhs1);
> +                 return true;
> +               }
> +           }
> +       }
> +      return res;
>      case OBJ_TYPE_REF:
>      case ASSERT_EXPR:
>      case WITH_SIZE_EXPR:
> --- gcc/tree-ssa-sccvn.c.jj     2012-09-25 11:59:43.000000000 +0200
> +++ gcc/tree-ssa-sccvn.c        2012-10-02 11:26:34.935326468 +0200
> @@ -1639,8 +1639,17 @@ vn_reference_lookup_3 (ao_ref *ref, tree
>                   if (i < CONSTRUCTOR_NELTS (ctor))
>                     {
>                       constructor_elt *elt = CONSTRUCTOR_ELT (ctor, i);
> -                     if (compare_tree_int (elt->index, i) == 0)
> -                       val = elt->value;
> +                     if (TREE_CODE (TREE_TYPE (rhs1)) == VECTOR_TYPE)
> +                       {
> +                         if (TREE_CODE (TREE_TYPE (elt->value))
> +                             != VECTOR_TYPE)
> +                           val = elt->value;
> +                       }
> +                     else if (elt->index)

This case removed.  We are only interested in VECTOR constructors
(the only constructors allowed in gimple).

Thanks,
Richard.

> +                       {
> +                         if (compare_tree_int (elt->index, i) == 0)
> +                           val = elt->value;
> +                       }
>                     }
>                 }
>               if (val)
> --- gcc/tree-vect-slp.c.jj      2012-10-01 17:28:17.215923346 +0200
> +++ gcc/tree-vect-slp.c 2012-10-01 17:59:49.423421284 +0200
> @@ -2345,6 +2345,7 @@ vect_get_constant_vectors (tree op, slp_
>    enum tree_code code = gimple_expr_code (stmt);
>    gimple def_stmt;
>    struct loop *loop;
> +  gimple_seq ctor_seq = NULL;
>
>    if (STMT_VINFO_DEF_TYPE (stmt_vinfo) == vect_reduction_def
>        && reduc_index != -1)
> @@ -2503,11 +2504,27 @@ vect_get_constant_vectors (tree op, slp_
>
>            /* Create 'vect_ = {op0,op1,...,opn}'.  */
>            number_of_places_left_in_vector--;
> -         if (constant_p
> -             && !types_compatible_p (TREE_TYPE (vector_type), TREE_TYPE 
> (op)))
> +         if (!types_compatible_p (TREE_TYPE (vector_type), TREE_TYPE (op)))
>             {
> -             op = fold_unary (VIEW_CONVERT_EXPR, TREE_TYPE (vector_type), 
> op);
> -             gcc_assert (op && CONSTANT_CLASS_P (op));
> +             if (constant_p)
> +               {
> +                 op = fold_unary (VIEW_CONVERT_EXPR,
> +                                  TREE_TYPE (vector_type), op);
> +                 gcc_assert (op && CONSTANT_CLASS_P (op));
> +               }
> +             else
> +               {
> +                 tree new_temp
> +                   = make_ssa_name (TREE_TYPE (vector_type), NULL);
> +                 gimple init_stmt;
> +                 op = build1 (VIEW_CONVERT_EXPR, TREE_TYPE (vector_type),
> +                              op);
> +                 init_stmt
> +                   = gimple_build_assign_with_ops (VIEW_CONVERT_EXPR,
> +                                                   new_temp, op, NULL_TREE);
> +                 gimple_seq_add_stmt (&ctor_seq, init_stmt);
> +                 op = new_temp;
> +               }
>             }
>           elts[number_of_places_left_in_vector] = op;
>
> @@ -2529,6 +2546,15 @@ vect_get_constant_vectors (tree op, slp_
>                VEC_quick_push (tree, voprnds,
>                                vect_init_vector (stmt, vec_cst,
>                                                 vector_type, NULL));
> +             if (ctor_seq != NULL)
> +               {
> +                 gimple init_stmt
> +                   = SSA_NAME_DEF_STMT (VEC_last (tree, voprnds));
> +                 gimple_stmt_iterator gsi = gsi_for_stmt (init_stmt);
> +                 gsi_insert_seq_before_without_update (&gsi, ctor_seq,
> +                                                       GSI_SAME_STMT);
> +                 ctor_seq = NULL;
> +               }
>              }
>          }
>      }
>
>         Jakub

Reply via email to