On Thu, Nov 16, 2017 at 5:21 PM, Nathan Froyd <froy...@mozilla.com> wrote:
> Default-initialization of scalar arrays in C++ member initialization
> lists produced rather slow code, laboriously setting each element of the
> array to zero.  It would be much faster to block-initialize the array,
> and that's what this patch does.
>
> The patch works for me, but I'm not sure if it's the best way to
> accomplish this.  At least two other possibilities come to mind:
>
> 1) Detect this case in build_vec_init_expr and act as though the user
>    wrote 'member{0}', which the front-end already produces efficient
>    code for.
>
> 2) Detect this case in build_vec_init, but again, act as though the user
>    wrote 'member{0}' and let everything proceed as normal.
>    (Alternatively, handle this case prior to calling build_vec_init and
>    pass different arguments to build_vec_init.)
>
> Opinions as to the best way forward here?  I'm unsure of whether the
> code below is front-end friendly; I see in the gimple dumps that the
> solution below adds an extra CLOBBER on 'this' for 'member()', whereas
> 'member{0}' does not.  It's possible that I'm missing something.
>
> Bootstrapped on x86_64-unknown-linux-gnu, no regressions.
>
> OK for trunk?

Ok. This is certainly the form the middle-end likes most.

Richard.

> -Nathan
>
> gcc/cp/
>         PR c++/82888
>         * init.c (build_vec_init): Handle default-initialization of array
>         types.
>
> gcc/testsuite/
>         PR c++/82888
>         * g++.dg/init/pr82888.C: New.
>
> diff --git a/gcc/cp/init.c b/gcc/cp/init.c
> index c76460d..53d6133 100644
> --- a/gcc/cp/init.c
> +++ b/gcc/cp/init.c
> @@ -4038,6 +4038,15 @@ build_vec_init (tree base, tree maxindex, tree init,
>         }
>      }
>
> +  /* Default-initialize scalar arrays directly.  */
> +  if (TREE_CODE (atype) == ARRAY_TYPE
> +      && SCALAR_TYPE_P (TREE_TYPE (atype))
> +      && !init)
> +    {
> +      gcc_assert (!from_array);
> +      return build2 (MODIFY_EXPR, atype, base, build_constructor (atype, 
> NULL));
> +    }
> +
>    /* If we have a braced-init-list or string constant, make sure that the 
> array
>       is big enough for all the initializers.  */
>    bool length_check = (init
> diff --git a/gcc/testsuite/g++.dg/init/pr82888.C 
> b/gcc/testsuite/g++.dg/init/pr82888.C
> new file mode 100644
> index 0000000..9225e23
> --- /dev/null
> +++ b/gcc/testsuite/g++.dg/init/pr82888.C
> @@ -0,0 +1,18 @@
> +// { dg-do compile }
> +// { dg-options "-fdump-tree-gimple" }
> +
> +class A
> +{
> +public:
> +  A();
> +
> +private:
> +  unsigned char mStorage[4096];
> +};
> +
> +A::A()
> +  : mStorage()
> +{}
> +
> +// { dg-final { scan-tree-dump "this->mStorage = {}" "gimple" } }
> +// { dg-final { scan-tree-dump-not "&this->mStorage" "gimple" } }

Reply via email to