c++: constexpr empty subobject confusion [PR110197]

Message ID 20230726165735.2058945-1-ppalka@redhat.com
State Accepted
Headers
Series c++: constexpr empty subobject confusion [PR110197] |

Checks

Context Check Description
snail/gcc-patch-check success Github commit url

Commit Message

Patrick Palka July 26, 2023, 4:57 p.m. UTC
  Bootstrapped and regtested on x86_64-pc-linux-gnu, does this look OK for
trunk/13 (later)?

-- >8 --

Now that init_subob_ctx no longer sets new_ctx.ctor for a subobject of
empty type, it seems we need to ensure its callers cxx_eval_bare_aggregate
and cxx_eval_vec_init_1 consistently omit entries for such subobjects in
the parent ctx->ctor.  We also need to allow cxx_eval_array_reference
to synthesize an empty element object even if the array CONSTRUCTOR
has CONSTRUCTOR_NO_CLEARING set.

	PR c++/110197

gcc/cp/ChangeLog:

	* constexpr.cc (cxx_eval_array_reference): Return a synthesized
	empty subobject even if CONSTRUCTOR_NO_CLEARING is set.
	(cxx_eval_bare_aggregate): Set 'no_slot' to true more generally
	whenever new_ctx.ctor is empty, i.e. for any subobject of empty
	type.
	(cxx_eval_vec_init_1): Define 'no_slot' as above and use it
	accordingly.

gcc/testsuite/ChangeLog:

	* g++.dg/cpp0x/constexpr-empty18.C: New test.
	* g++.dg/cpp0x/constexpr-empty19.C: New test.
---
 gcc/cp/constexpr.cc                           | 23 +++++++++++++------
 .../g++.dg/cpp0x/constexpr-empty18.C          |  7 ++++++
 .../g++.dg/cpp0x/constexpr-empty19.C          | 12 ++++++++++
 3 files changed, 35 insertions(+), 7 deletions(-)
 create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C
 create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C
  

Comments

Jason Merrill July 26, 2023, 5:14 p.m. UTC | #1
On 7/26/23 12:57, Patrick Palka wrote:
> Bootstrapped and regtested on x86_64-pc-linux-gnu, does this look OK for
> trunk/13 (later)?

OK.

> -- >8 --
> 
> Now that init_subob_ctx no longer sets new_ctx.ctor for a subobject of
> empty type, it seems we need to ensure its callers cxx_eval_bare_aggregate
> and cxx_eval_vec_init_1 consistently omit entries for such subobjects in
> the parent ctx->ctor.  We also need to allow cxx_eval_array_reference
> to synthesize an empty element object even if the array CONSTRUCTOR
> has CONSTRUCTOR_NO_CLEARING set.
> 
> 	PR c++/110197
> 
> gcc/cp/ChangeLog:
> 
> 	* constexpr.cc (cxx_eval_array_reference): Return a synthesized
> 	empty subobject even if CONSTRUCTOR_NO_CLEARING is set.
> 	(cxx_eval_bare_aggregate): Set 'no_slot' to true more generally
> 	whenever new_ctx.ctor is empty, i.e. for any subobject of empty
> 	type.
> 	(cxx_eval_vec_init_1): Define 'no_slot' as above and use it
> 	accordingly.
> 
> gcc/testsuite/ChangeLog:
> 
> 	* g++.dg/cpp0x/constexpr-empty18.C: New test.
> 	* g++.dg/cpp0x/constexpr-empty19.C: New test.
> ---
>   gcc/cp/constexpr.cc                           | 23 +++++++++++++------
>   .../g++.dg/cpp0x/constexpr-empty18.C          |  7 ++++++
>   .../g++.dg/cpp0x/constexpr-empty19.C          | 12 ++++++++++
>   3 files changed, 35 insertions(+), 7 deletions(-)
>   create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C
>   create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C
> 
> diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
> index f2fcb54626d..da2c3116810 100644
> --- a/gcc/cp/constexpr.cc
> +++ b/gcc/cp/constexpr.cc
> @@ -4297,6 +4297,9 @@ cxx_eval_array_reference (const constexpr_ctx *ctx, tree t,
>   
>     /* Not found.  */
>   
> +  if (is_really_empty_class (elem_type, /*ignore_vptr*/false))
> +    return build_constructor (elem_type, NULL);
> +
>     if (TREE_CODE (ary) == CONSTRUCTOR
>         && CONSTRUCTOR_NO_CLEARING (ary))
>       {
> @@ -4314,9 +4317,7 @@ cxx_eval_array_reference (const constexpr_ctx *ctx, tree t,
>        directly for non-aggregates to avoid creating a garbage CONSTRUCTOR.  */
>     tree val;
>     constexpr_ctx new_ctx;
> -  if (is_really_empty_class (elem_type, /*ignore_vptr*/false))
> -    return build_constructor (elem_type, NULL);
> -  else if (CP_AGGREGATE_TYPE_P (elem_type))
> +  if (CP_AGGREGATE_TYPE_P (elem_type))
>       {
>         tree empty_ctor = build_constructor (init_list_type_node, NULL);
>         val = digest_init (elem_type, empty_ctor, tf_warning_or_error);
> @@ -5095,9 +5096,9 @@ cxx_eval_bare_aggregate (const constexpr_ctx *ctx, tree t,
>     FOR_EACH_CONSTRUCTOR_ELT (v, i, index, value)
>       {
>         tree orig_value = value;
> -      /* Like in cxx_eval_store_expression, omit entries for empty fields.  */
> -      bool no_slot = TREE_CODE (type) == RECORD_TYPE && is_empty_field (index);
>         init_subob_ctx (ctx, new_ctx, index, value);
> +      /* Like in cxx_eval_store_expression, omit entries for empty fields.  */
> +      bool no_slot = new_ctx.ctor == NULL_TREE;
>         int pos_hint = -1;
>         if (new_ctx.ctor != ctx->ctor && !no_slot)
>   	{
> @@ -5261,7 +5262,8 @@ cxx_eval_vec_init_1 (const constexpr_ctx *ctx, tree atype, tree init,
>         bool reuse = false;
>         constexpr_ctx new_ctx;
>         init_subob_ctx (ctx, new_ctx, idx, pre_init ? init : elttype);
> -      if (new_ctx.ctor != ctx->ctor)
> +      bool no_slot = new_ctx.ctor == NULL_TREE;
> +      if (new_ctx.ctor != ctx->ctor && !no_slot)
>   	{
>   	  if (zeroed_out)
>   	    CONSTRUCTOR_NO_CLEARING (new_ctx.ctor) = false;
> @@ -5306,7 +5308,14 @@ cxx_eval_vec_init_1 (const constexpr_ctx *ctx, tree atype, tree init,
>   	}
>         if (*non_constant_p)
>   	break;
> -      if (new_ctx.ctor != ctx->ctor)
> +      if (no_slot)
> +	{
> +	  /* This is an initializer for an empty subobject; now that we've
> +	     checked that it's constant, we can ignore it.  */
> +	  gcc_checking_assert (i == 0);
> +	  break;
> +	}
> +      else if (new_ctx.ctor != ctx->ctor)
>   	{
>   	  /* We appended this element above; update the value.  */
>   	  gcc_assert ((*p)->last().index == idx);
> diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C
> new file mode 100644
> index 00000000000..4bb9e3dcb64
> --- /dev/null
> +++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C
> @@ -0,0 +1,7 @@
> +// PR c++/110197
> +// { dg-do compile { target c++11 } }
> +
> +struct A { constexpr A(int) { } };
> +struct B { A a; };
> +constexpr B f(int n) { return B{A{n}}; }
> +constexpr B b = f(1);
> diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C
> new file mode 100644
> index 00000000000..5ad67682c5b
> --- /dev/null
> +++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C
> @@ -0,0 +1,12 @@
> +// PR c++/110197
> +// { dg-do compile { target c++11 } }
> +
> +struct A {
> +  constexpr A() : A(__builtin_is_constant_evaluated()) { }
> +  constexpr A(int) { }
> +};
> +constexpr A a1[1] = {{}};
> +constexpr A a2[2] = {{}, {}};
> +constexpr A a3[3] = {{}, {}, {}};
> +constexpr A a4[4] = {};
> +constexpr A a5[5] = {};
  

Patch

diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
index f2fcb54626d..da2c3116810 100644
--- a/gcc/cp/constexpr.cc
+++ b/gcc/cp/constexpr.cc
@@ -4297,6 +4297,9 @@  cxx_eval_array_reference (const constexpr_ctx *ctx, tree t,
 
   /* Not found.  */
 
+  if (is_really_empty_class (elem_type, /*ignore_vptr*/false))
+    return build_constructor (elem_type, NULL);
+
   if (TREE_CODE (ary) == CONSTRUCTOR
       && CONSTRUCTOR_NO_CLEARING (ary))
     {
@@ -4314,9 +4317,7 @@  cxx_eval_array_reference (const constexpr_ctx *ctx, tree t,
      directly for non-aggregates to avoid creating a garbage CONSTRUCTOR.  */
   tree val;
   constexpr_ctx new_ctx;
-  if (is_really_empty_class (elem_type, /*ignore_vptr*/false))
-    return build_constructor (elem_type, NULL);
-  else if (CP_AGGREGATE_TYPE_P (elem_type))
+  if (CP_AGGREGATE_TYPE_P (elem_type))
     {
       tree empty_ctor = build_constructor (init_list_type_node, NULL);
       val = digest_init (elem_type, empty_ctor, tf_warning_or_error);
@@ -5095,9 +5096,9 @@  cxx_eval_bare_aggregate (const constexpr_ctx *ctx, tree t,
   FOR_EACH_CONSTRUCTOR_ELT (v, i, index, value)
     {
       tree orig_value = value;
-      /* Like in cxx_eval_store_expression, omit entries for empty fields.  */
-      bool no_slot = TREE_CODE (type) == RECORD_TYPE && is_empty_field (index);
       init_subob_ctx (ctx, new_ctx, index, value);
+      /* Like in cxx_eval_store_expression, omit entries for empty fields.  */
+      bool no_slot = new_ctx.ctor == NULL_TREE;
       int pos_hint = -1;
       if (new_ctx.ctor != ctx->ctor && !no_slot)
 	{
@@ -5261,7 +5262,8 @@  cxx_eval_vec_init_1 (const constexpr_ctx *ctx, tree atype, tree init,
       bool reuse = false;
       constexpr_ctx new_ctx;
       init_subob_ctx (ctx, new_ctx, idx, pre_init ? init : elttype);
-      if (new_ctx.ctor != ctx->ctor)
+      bool no_slot = new_ctx.ctor == NULL_TREE;
+      if (new_ctx.ctor != ctx->ctor && !no_slot)
 	{
 	  if (zeroed_out)
 	    CONSTRUCTOR_NO_CLEARING (new_ctx.ctor) = false;
@@ -5306,7 +5308,14 @@  cxx_eval_vec_init_1 (const constexpr_ctx *ctx, tree atype, tree init,
 	}
       if (*non_constant_p)
 	break;
-      if (new_ctx.ctor != ctx->ctor)
+      if (no_slot)
+	{
+	  /* This is an initializer for an empty subobject; now that we've
+	     checked that it's constant, we can ignore it.  */
+	  gcc_checking_assert (i == 0);
+	  break;
+	}
+      else if (new_ctx.ctor != ctx->ctor)
 	{
 	  /* We appended this element above; update the value.  */
 	  gcc_assert ((*p)->last().index == idx);
diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C
new file mode 100644
index 00000000000..4bb9e3dcb64
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty18.C
@@ -0,0 +1,7 @@ 
+// PR c++/110197
+// { dg-do compile { target c++11 } }
+
+struct A { constexpr A(int) { } };
+struct B { A a; };
+constexpr B f(int n) { return B{A{n}}; }
+constexpr B b = f(1);
diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C
new file mode 100644
index 00000000000..5ad67682c5b
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-empty19.C
@@ -0,0 +1,12 @@ 
+// PR c++/110197
+// { dg-do compile { target c++11 } }
+
+struct A {
+  constexpr A() : A(__builtin_is_constant_evaluated()) { }
+  constexpr A(int) { }
+};
+constexpr A a1[1] = {{}};
+constexpr A a2[2] = {{}, {}};
+constexpr A a3[3] = {{}, {}, {}};
+constexpr A a4[4] = {};
+constexpr A a5[5] = {};