diff mbox series

[v2] c++: Improve diagnostics for constexpr cast from void*

Message ID 6525e4d7.170a0220.2ad8c.28d8@mx.google.com
State New
Headers show
Series [v2] c++: Improve diagnostics for constexpr cast from void* | expand

Commit Message

Nathaniel Shead Oct. 10, 2023, 11:57 p.m. UTC
On Mon, Oct 09, 2023 at 04:10:20PM -0400, Jason Merrill wrote:
> On 10/9/23 06:03, Nathaniel Shead wrote:
> > Bootstrapped and regtested on x86_64-pc-linux-gnu with
> > GXX_TESTSUITE_STDS=98,11,14,17,20,23,26,impcx.
> > 
> > -- >8 --
> > 
> > This patch improves the errors given when casting from void* in C++26 to
> > include the expected type if the type of the pointed-to object was
> > not similar to the casted-to type.
> > 
> > It also ensures (for all standard modes) that void* casts are checked
> > even for DECL_ARTIFICIAL declarations, such as lifetime-extended
> > temporaries, and is only ignored for cases where we know it's OK (heap
> > identifiers and source_location::current). This provides more accurate
> > diagnostics when using the pointer and ensures that some other casts
> > from void* are now correctly rejected.
> > 
> > gcc/cp/ChangeLog:
> > 
> > 	* constexpr.cc (is_std_source_location_current): New.
> > 	(cxx_eval_constant_expression): Only ignore cast from void* for
> > 	specific cases and improve other diagnostics.
> > 
> > gcc/testsuite/ChangeLog:
> > 
> > 	* g++.dg/cpp0x/constexpr-cast4.C: New test.
> > 
> > Signed-off-by: Nathaniel Shead <nathanieloshead@gmail.com>
> > ---
> >   gcc/cp/constexpr.cc                          | 83 +++++++++++++++++---
> >   gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C |  7 ++
> >   2 files changed, 78 insertions(+), 12 deletions(-)
> >   create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
> > 
> > diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
> > index 0f948db7c2d..f38d541a662 100644
> > --- a/gcc/cp/constexpr.cc
> > +++ b/gcc/cp/constexpr.cc
> > @@ -2301,6 +2301,36 @@ is_std_allocator_allocate (const constexpr_call *call)
> >   	  && is_std_allocator_allocate (call->fundef->decl));
> >   }
> > +/* Return true if FNDECL is std::source_location::current.  */
> > +
> > +static inline bool
> > +is_std_source_location_current (tree fndecl)
> > +{
> > +  if (!decl_in_std_namespace_p (fndecl))
> > +    return false;
> > +
> > +  tree name = DECL_NAME (fndecl);
> > +  if (name == NULL_TREE || !id_equal (name, "current"))
> > +    return false;
> > +
> > +  tree ctx = DECL_CONTEXT (fndecl);
> > +  if (ctx == NULL_TREE || !CLASS_TYPE_P (ctx) || !TYPE_MAIN_DECL (ctx))
> > +    return false;
> > +
> > +  name = DECL_NAME (TYPE_MAIN_DECL (ctx));
> > +  return name && id_equal (name, "source_location");
> > +}
> > +
> > +/* Overload for the above taking constexpr_call*.  */
> > +
> > +static inline bool
> > +is_std_source_location_current (const constexpr_call *call)
> > +{
> > +  return (call
> > +	  && call->fundef
> > +	  && is_std_source_location_current (call->fundef->decl));
> > +}
> > +
> >   /* Return true if FNDECL is __dynamic_cast.  */
> >   static inline bool
> > @@ -7850,33 +7880,62 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
> >   	if (TYPE_PTROB_P (type)
> >   	    && TYPE_PTR_P (TREE_TYPE (op))
> >   	    && VOID_TYPE_P (TREE_TYPE (TREE_TYPE (op)))
> > -	    /* Inside a call to std::construct_at or to
> > -	       std::allocator<T>::{,de}allocate, we permit casting from void*
> > +	    /* Inside a call to std::construct_at,
> > +	       std::allocator<T>::{,de}allocate, or
> > +	       std::source_location::current, we permit casting from void*
> >   	       because that is compiler-generated code.  */
> >   	    && !is_std_construct_at (ctx->call)
> > -	    && !is_std_allocator_allocate (ctx->call))
> > +	    && !is_std_allocator_allocate (ctx->call)
> > +	    && !is_std_source_location_current (ctx->call))
> >   	  {
> >   	    /* Likewise, don't error when casting from void* when OP is
> >   	       &heap uninit and similar.  */
> >   	    tree sop = tree_strip_nop_conversions (op);
> > -	    if (TREE_CODE (sop) == ADDR_EXPR
> > -		&& VAR_P (TREE_OPERAND (sop, 0))
> > -		&& DECL_ARTIFICIAL (TREE_OPERAND (sop, 0)))
> > +	    tree decl = NULL_TREE;
> > +	    if (TREE_CODE (sop) == ADDR_EXPR)
> > +	      decl = TREE_OPERAND (sop, 0);
> > +	    if (decl
> > +		&& VAR_P (decl)
> > +		&& DECL_ARTIFICIAL (decl)
> > +		&& (DECL_NAME (decl) == heap_identifier
> > +		    || DECL_NAME (decl) == heap_uninit_identifier
> > +		    || DECL_NAME (decl) == heap_vec_identifier
> > +		    || DECL_NAME (decl) == heap_vec_uninit_identifier))
> >   	      /* OK */;
> >   	    /* P2738 (C++26): a conversion from a prvalue P of type "pointer to
> >   	       cv void" to a pointer-to-object type T unless P points to an
> >   	       object whose type is similar to T.  */
> > -	    else if (cxx_dialect > cxx23
> > -		     && (sop = cxx_fold_indirect_ref (ctx, loc,
> > -						      TREE_TYPE (type), sop)))
> > +	    else if (cxx_dialect > cxx23)
> >   	      {
> > -		r = build1 (ADDR_EXPR, type, sop);
> > -		break;
> > +		r = cxx_fold_indirect_ref (ctx, loc, TREE_TYPE (type), sop);
> > +		if (r)
> > +		  {
> > +		    r = build1 (ADDR_EXPR, type, r);
> > +		    break;
> > +		  }
> > +		if (!ctx->quiet)
> > +		  {
> > +		    if (TREE_CODE (sop) == ADDR_EXPR)
> > +		      {
> > +			error_at (loc, "cast from %qT is not allowed because "
> > +				  "pointed-to type %qT is not similar to %qT",
> > +				  TREE_TYPE (op), TREE_TYPE (TREE_TYPE (sop)),
> > +				  TREE_TYPE (type));
> > +			tree obj = build_fold_indirect_ref (sop);
> > +			inform (DECL_SOURCE_LOCATION (obj),
> > +				"pointed-to object declared here");
> > +		      }
> > +		    else
> > +		      error_at (loc, "cast from %qT is not allowed",
> 
> Can this be more helpful as well, i.e. say because op is not the address of
> an object of similar type?
> 
> Can we only get here if op is null, since we would have returned already for
> non-constant op?
> 
> Jason
> 

Hm, yes; I'd kept the error as it was because I wasn't sure what other
kinds of trees might end up here, but I've done a fair amount of testing
and I've only been able to reach here with null pointers: everything
else gets caught earlier. I've updated the error message appropriately
and documented this assumption with an assertion.

Bootstrapped + regtested on x86_64-pc-linux-gnu as above.

-- >8 --

This patch improves the errors given when casting from void* in C++26 to
include the expected type if the types of the pointed-to objects were
not similar. It also ensures (for all standard modes) that void* casts
are checked even for DECL_ARTIFICIAL declarations, such as
lifetime-extended temporaries, and is only ignored for cases where we
know it's OK (e.g. source_location::current) or have no other choice
(heap-allocated data).

gcc/cp/ChangeLog:

	* constexpr.cc (is_std_source_location_current): New.
	(cxx_eval_constant_expression): Only ignore cast from void* for
	specific cases and improve other diagnostics.

gcc/testsuite/ChangeLog:

	* g++.dg/cpp0x/constexpr-cast4.C: New test.

Signed-off-by: Nathaniel Shead <nathanieloshead@gmail.com>
---
 gcc/cp/constexpr.cc                          | 87 +++++++++++++++++---
 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C | 11 +++
 2 files changed, 86 insertions(+), 12 deletions(-)
 create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C

Comments

Marek Polacek Oct. 11, 2023, 3:41 p.m. UTC | #1
On Wed, Oct 11, 2023 at 10:57:06AM +1100, Nathaniel Shead wrote:
> On Mon, Oct 09, 2023 at 04:10:20PM -0400, Jason Merrill wrote:
> > On 10/9/23 06:03, Nathaniel Shead wrote:
> > > Bootstrapped and regtested on x86_64-pc-linux-gnu with
> > > GXX_TESTSUITE_STDS=98,11,14,17,20,23,26,impcx.
> > > 
> > > -- >8 --
> > > 
> > > This patch improves the errors given when casting from void* in C++26 to
> > > include the expected type if the type of the pointed-to object was
> > > not similar to the casted-to type.
> > > 
> > > It also ensures (for all standard modes) that void* casts are checked
> > > even for DECL_ARTIFICIAL declarations, such as lifetime-extended
> > > temporaries, and is only ignored for cases where we know it's OK (heap
> > > identifiers and source_location::current). This provides more accurate
> > > diagnostics when using the pointer and ensures that some other casts
> > > from void* are now correctly rejected.
> > > 
> > > gcc/cp/ChangeLog:
> > > 
> > > 	* constexpr.cc (is_std_source_location_current): New.
> > > 	(cxx_eval_constant_expression): Only ignore cast from void* for
> > > 	specific cases and improve other diagnostics.
> > > 
> > > gcc/testsuite/ChangeLog:
> > > 
> > > 	* g++.dg/cpp0x/constexpr-cast4.C: New test.
> > > 
> > > Signed-off-by: Nathaniel Shead <nathanieloshead@gmail.com>
> > > ---
> > >   gcc/cp/constexpr.cc                          | 83 +++++++++++++++++---
> > >   gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C |  7 ++
> > >   2 files changed, 78 insertions(+), 12 deletions(-)
> > >   create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
> > > 
> > > diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
> > > index 0f948db7c2d..f38d541a662 100644
> > > --- a/gcc/cp/constexpr.cc
> > > +++ b/gcc/cp/constexpr.cc
> > > @@ -2301,6 +2301,36 @@ is_std_allocator_allocate (const constexpr_call *call)
> > >   	  && is_std_allocator_allocate (call->fundef->decl));
> > >   }
> > > +/* Return true if FNDECL is std::source_location::current.  */
> > > +
> > > +static inline bool
> > > +is_std_source_location_current (tree fndecl)
> > > +{
> > > +  if (!decl_in_std_namespace_p (fndecl))
> > > +    return false;
> > > +
> > > +  tree name = DECL_NAME (fndecl);
> > > +  if (name == NULL_TREE || !id_equal (name, "current"))
> > > +    return false;
> > > +
> > > +  tree ctx = DECL_CONTEXT (fndecl);
> > > +  if (ctx == NULL_TREE || !CLASS_TYPE_P (ctx) || !TYPE_MAIN_DECL (ctx))
> > > +    return false;
> > > +
> > > +  name = DECL_NAME (TYPE_MAIN_DECL (ctx));
> > > +  return name && id_equal (name, "source_location");
> > > +}
> > > +
> > > +/* Overload for the above taking constexpr_call*.  */
> > > +
> > > +static inline bool
> > > +is_std_source_location_current (const constexpr_call *call)
> > > +{
> > > +  return (call
> > > +	  && call->fundef
> > > +	  && is_std_source_location_current (call->fundef->decl));
> > > +}
> > > +
> > >   /* Return true if FNDECL is __dynamic_cast.  */
> > >   static inline bool
> > > @@ -7850,33 +7880,62 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
> > >   	if (TYPE_PTROB_P (type)
> > >   	    && TYPE_PTR_P (TREE_TYPE (op))
> > >   	    && VOID_TYPE_P (TREE_TYPE (TREE_TYPE (op)))
> > > -	    /* Inside a call to std::construct_at or to
> > > -	       std::allocator<T>::{,de}allocate, we permit casting from void*
> > > +	    /* Inside a call to std::construct_at,
> > > +	       std::allocator<T>::{,de}allocate, or
> > > +	       std::source_location::current, we permit casting from void*
> > >   	       because that is compiler-generated code.  */
> > >   	    && !is_std_construct_at (ctx->call)
> > > -	    && !is_std_allocator_allocate (ctx->call))
> > > +	    && !is_std_allocator_allocate (ctx->call)
> > > +	    && !is_std_source_location_current (ctx->call))
> > >   	  {
> > >   	    /* Likewise, don't error when casting from void* when OP is
> > >   	       &heap uninit and similar.  */
> > >   	    tree sop = tree_strip_nop_conversions (op);
> > > -	    if (TREE_CODE (sop) == ADDR_EXPR
> > > -		&& VAR_P (TREE_OPERAND (sop, 0))
> > > -		&& DECL_ARTIFICIAL (TREE_OPERAND (sop, 0)))
> > > +	    tree decl = NULL_TREE;
> > > +	    if (TREE_CODE (sop) == ADDR_EXPR)
> > > +	      decl = TREE_OPERAND (sop, 0);
> > > +	    if (decl
> > > +		&& VAR_P (decl)
> > > +		&& DECL_ARTIFICIAL (decl)
> > > +		&& (DECL_NAME (decl) == heap_identifier
> > > +		    || DECL_NAME (decl) == heap_uninit_identifier
> > > +		    || DECL_NAME (decl) == heap_vec_identifier
> > > +		    || DECL_NAME (decl) == heap_vec_uninit_identifier))
> > >   	      /* OK */;
> > >   	    /* P2738 (C++26): a conversion from a prvalue P of type "pointer to
> > >   	       cv void" to a pointer-to-object type T unless P points to an
> > >   	       object whose type is similar to T.  */
> > > -	    else if (cxx_dialect > cxx23
> > > -		     && (sop = cxx_fold_indirect_ref (ctx, loc,
> > > -						      TREE_TYPE (type), sop)))
> > > +	    else if (cxx_dialect > cxx23)
> > >   	      {
> > > -		r = build1 (ADDR_EXPR, type, sop);
> > > -		break;
> > > +		r = cxx_fold_indirect_ref (ctx, loc, TREE_TYPE (type), sop);
> > > +		if (r)
> > > +		  {
> > > +		    r = build1 (ADDR_EXPR, type, r);
> > > +		    break;
> > > +		  }
> > > +		if (!ctx->quiet)
> > > +		  {
> > > +		    if (TREE_CODE (sop) == ADDR_EXPR)
> > > +		      {
> > > +			error_at (loc, "cast from %qT is not allowed because "
> > > +				  "pointed-to type %qT is not similar to %qT",
> > > +				  TREE_TYPE (op), TREE_TYPE (TREE_TYPE (sop)),
> > > +				  TREE_TYPE (type));
> > > +			tree obj = build_fold_indirect_ref (sop);
> > > +			inform (DECL_SOURCE_LOCATION (obj),
> > > +				"pointed-to object declared here");
> > > +		      }
> > > +		    else
> > > +		      error_at (loc, "cast from %qT is not allowed",
> > 
> > Can this be more helpful as well, i.e. say because op is not the address of
> > an object of similar type?
> > 
> > Can we only get here if op is null, since we would have returned already for
> > non-constant op?
> > 
> > Jason
> > 
> 
> Hm, yes; I'd kept the error as it was because I wasn't sure what other
> kinds of trees might end up here, but I've done a fair amount of testing
> and I've only been able to reach here with null pointers: everything
> else gets caught earlier. I've updated the error message appropriately
> and documented this assumption with an assertion.
> 
> Bootstrapped + regtested on x86_64-pc-linux-gnu as above.
> 
> -- >8 --
> 
> This patch improves the errors given when casting from void* in C++26 to
> include the expected type if the types of the pointed-to objects were
> not similar. It also ensures (for all standard modes) that void* casts
> are checked even for DECL_ARTIFICIAL declarations, such as
> lifetime-extended temporaries, and is only ignored for cases where we
> know it's OK (e.g. source_location::current) or have no other choice
> (heap-allocated data).
> 
> gcc/cp/ChangeLog:
> 
> 	* constexpr.cc (is_std_source_location_current): New.
> 	(cxx_eval_constant_expression): Only ignore cast from void* for
> 	specific cases and improve other diagnostics.
> 
> gcc/testsuite/ChangeLog:
> 
> 	* g++.dg/cpp0x/constexpr-cast4.C: New test.
> 
> Signed-off-by: Nathaniel Shead <nathanieloshead@gmail.com>
> ---
>  gcc/cp/constexpr.cc                          | 87 +++++++++++++++++---
>  gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C | 11 +++
>  2 files changed, 86 insertions(+), 12 deletions(-)
>  create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
> 
> diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
> index 0f948db7c2d..d060e374cb3 100644
> --- a/gcc/cp/constexpr.cc
> +++ b/gcc/cp/constexpr.cc
> @@ -2301,6 +2301,36 @@ is_std_allocator_allocate (const constexpr_call *call)
>  	  && is_std_allocator_allocate (call->fundef->decl));
>  }
>  
> +/* Return true if FNDECL is std::source_location::current.  */
> +
> +static inline bool
> +is_std_source_location_current (tree fndecl)
> +{
> +  if (!decl_in_std_namespace_p (fndecl))
> +    return false;
> +
> +  tree name = DECL_NAME (fndecl);
> +  if (name == NULL_TREE || !id_equal (name, "current"))
> +    return false;
> +
> +  tree ctx = DECL_CONTEXT (fndecl);
> +  if (ctx == NULL_TREE || !CLASS_TYPE_P (ctx) || !TYPE_MAIN_DECL (ctx))
> +    return false;
> +
> +  name = DECL_NAME (TYPE_MAIN_DECL (ctx));
> +  return name && id_equal (name, "source_location");
> +}
> +
> +/* Overload for the above taking constexpr_call*.  */
> +
> +static inline bool
> +is_std_source_location_current (const constexpr_call *call)
> +{
> +  return (call
> +	  && call->fundef
> +	  && is_std_source_location_current (call->fundef->decl));
> +}
> +
>  /* Return true if FNDECL is __dynamic_cast.  */
>  
>  static inline bool
> @@ -7850,33 +7880,66 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
>  	if (TYPE_PTROB_P (type)
>  	    && TYPE_PTR_P (TREE_TYPE (op))
>  	    && VOID_TYPE_P (TREE_TYPE (TREE_TYPE (op)))
> -	    /* Inside a call to std::construct_at or to
> -	       std::allocator<T>::{,de}allocate, we permit casting from void*
> +	    /* Inside a call to std::construct_at,
> +	       std::allocator<T>::{,de}allocate, or
> +	       std::source_location::current, we permit casting from void*
>  	       because that is compiler-generated code.  */
>  	    && !is_std_construct_at (ctx->call)
> -	    && !is_std_allocator_allocate (ctx->call))
> +	    && !is_std_allocator_allocate (ctx->call)
> +	    && !is_std_source_location_current (ctx->call))
>  	  {
>  	    /* Likewise, don't error when casting from void* when OP is
>  	       &heap uninit and similar.  */
>  	    tree sop = tree_strip_nop_conversions (op);
> -	    if (TREE_CODE (sop) == ADDR_EXPR
> -		&& VAR_P (TREE_OPERAND (sop, 0))
> -		&& DECL_ARTIFICIAL (TREE_OPERAND (sop, 0)))
> +	    tree decl = NULL_TREE;
> +	    if (TREE_CODE (sop) == ADDR_EXPR)
> +	      decl = TREE_OPERAND (sop, 0);
> +	    if (decl
> +		&& VAR_P (decl)
> +		&& DECL_ARTIFICIAL (decl)
> +		&& (DECL_NAME (decl) == heap_identifier
> +		    || DECL_NAME (decl) == heap_uninit_identifier
> +		    || DECL_NAME (decl) == heap_vec_identifier
> +		    || DECL_NAME (decl) == heap_vec_uninit_identifier))
>  	      /* OK */;
>  	    /* P2738 (C++26): a conversion from a prvalue P of type "pointer to
>  	       cv void" to a pointer-to-object type T unless P points to an
>  	       object whose type is similar to T.  */
> -	    else if (cxx_dialect > cxx23
> -		     && (sop = cxx_fold_indirect_ref (ctx, loc,
> -						      TREE_TYPE (type), sop)))
> +	    else if (cxx_dialect > cxx23)
>  	      {
> -		r = build1 (ADDR_EXPR, type, sop);
> -		break;
> +		r = cxx_fold_indirect_ref (ctx, loc, TREE_TYPE (type), sop);
> +		if (r)
> +		  {
> +		    r = build1 (ADDR_EXPR, type, r);
> +		    break;
> +		  }
> +		if (!ctx->quiet)
> +		  {
> +		    if (TREE_CODE (sop) == ADDR_EXPR)
> +		      {

A very minor point (sorry), but I think you want

  auto_diagnostic_group d;

here.  Not need to repost the patch only because of this.

> +			error_at (loc, "cast from %qT is not allowed because "
> +				  "pointed-to type %qT is not similar to %qT",
> +				  TREE_TYPE (op), TREE_TYPE (TREE_TYPE (sop)),
> +				  TREE_TYPE (type));
> +			tree obj = build_fold_indirect_ref (sop);
> +			inform (DECL_SOURCE_LOCATION (obj),
> +				"pointed-to object declared here");
> +		      }

Marek
Jason Merrill Oct. 20, 2023, 3:25 a.m. UTC | #2
On 10/11/23 11:41, Marek Polacek wrote:
> On Wed, Oct 11, 2023 at 10:57:06AM +1100, Nathaniel Shead wrote:
>> On Mon, Oct 09, 2023 at 04:10:20PM -0400, Jason Merrill wrote:
>>> On 10/9/23 06:03, Nathaniel Shead wrote:
>>>> Bootstrapped and regtested on x86_64-pc-linux-gnu with
>>>> GXX_TESTSUITE_STDS=98,11,14,17,20,23,26,impcx.
>>>>
>>>> -- >8 --
>>>>
>>>> This patch improves the errors given when casting from void* in C++26 to
>>>> include the expected type if the type of the pointed-to object was
>>>> not similar to the casted-to type.
>>>>
>>>> It also ensures (for all standard modes) that void* casts are checked
>>>> even for DECL_ARTIFICIAL declarations, such as lifetime-extended
>>>> temporaries, and is only ignored for cases where we know it's OK (heap
>>>> identifiers and source_location::current). This provides more accurate
>>>> diagnostics when using the pointer and ensures that some other casts
>>>> from void* are now correctly rejected.
>>>>
>>>> gcc/cp/ChangeLog:
>>>>
>>>> 	* constexpr.cc (is_std_source_location_current): New.
>>>> 	(cxx_eval_constant_expression): Only ignore cast from void* for
>>>> 	specific cases and improve other diagnostics.
>>>>
>>>> gcc/testsuite/ChangeLog:
>>>>
>>>> 	* g++.dg/cpp0x/constexpr-cast4.C: New test.
>>>>
>>>> Signed-off-by: Nathaniel Shead <nathanieloshead@gmail.com>
>>>> ---
>>>>    gcc/cp/constexpr.cc                          | 83 +++++++++++++++++---
>>>>    gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C |  7 ++
>>>>    2 files changed, 78 insertions(+), 12 deletions(-)
>>>>    create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
>>>>
>>>> diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
>>>> index 0f948db7c2d..f38d541a662 100644
>>>> --- a/gcc/cp/constexpr.cc
>>>> +++ b/gcc/cp/constexpr.cc
>>>> @@ -2301,6 +2301,36 @@ is_std_allocator_allocate (const constexpr_call *call)
>>>>    	  && is_std_allocator_allocate (call->fundef->decl));
>>>>    }
>>>> +/* Return true if FNDECL is std::source_location::current.  */
>>>> +
>>>> +static inline bool
>>>> +is_std_source_location_current (tree fndecl)
>>>> +{
>>>> +  if (!decl_in_std_namespace_p (fndecl))
>>>> +    return false;
>>>> +
>>>> +  tree name = DECL_NAME (fndecl);
>>>> +  if (name == NULL_TREE || !id_equal (name, "current"))
>>>> +    return false;
>>>> +
>>>> +  tree ctx = DECL_CONTEXT (fndecl);
>>>> +  if (ctx == NULL_TREE || !CLASS_TYPE_P (ctx) || !TYPE_MAIN_DECL (ctx))
>>>> +    return false;
>>>> +
>>>> +  name = DECL_NAME (TYPE_MAIN_DECL (ctx));
>>>> +  return name && id_equal (name, "source_location");
>>>> +}
>>>> +
>>>> +/* Overload for the above taking constexpr_call*.  */
>>>> +
>>>> +static inline bool
>>>> +is_std_source_location_current (const constexpr_call *call)
>>>> +{
>>>> +  return (call
>>>> +	  && call->fundef
>>>> +	  && is_std_source_location_current (call->fundef->decl));
>>>> +}
>>>> +
>>>>    /* Return true if FNDECL is __dynamic_cast.  */
>>>>    static inline bool
>>>> @@ -7850,33 +7880,62 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
>>>>    	if (TYPE_PTROB_P (type)
>>>>    	    && TYPE_PTR_P (TREE_TYPE (op))
>>>>    	    && VOID_TYPE_P (TREE_TYPE (TREE_TYPE (op)))
>>>> -	    /* Inside a call to std::construct_at or to
>>>> -	       std::allocator<T>::{,de}allocate, we permit casting from void*
>>>> +	    /* Inside a call to std::construct_at,
>>>> +	       std::allocator<T>::{,de}allocate, or
>>>> +	       std::source_location::current, we permit casting from void*
>>>>    	       because that is compiler-generated code.  */
>>>>    	    && !is_std_construct_at (ctx->call)
>>>> -	    && !is_std_allocator_allocate (ctx->call))
>>>> +	    && !is_std_allocator_allocate (ctx->call)
>>>> +	    && !is_std_source_location_current (ctx->call))
>>>>    	  {
>>>>    	    /* Likewise, don't error when casting from void* when OP is
>>>>    	       &heap uninit and similar.  */
>>>>    	    tree sop = tree_strip_nop_conversions (op);
>>>> -	    if (TREE_CODE (sop) == ADDR_EXPR
>>>> -		&& VAR_P (TREE_OPERAND (sop, 0))
>>>> -		&& DECL_ARTIFICIAL (TREE_OPERAND (sop, 0)))
>>>> +	    tree decl = NULL_TREE;
>>>> +	    if (TREE_CODE (sop) == ADDR_EXPR)
>>>> +	      decl = TREE_OPERAND (sop, 0);
>>>> +	    if (decl
>>>> +		&& VAR_P (decl)
>>>> +		&& DECL_ARTIFICIAL (decl)
>>>> +		&& (DECL_NAME (decl) == heap_identifier
>>>> +		    || DECL_NAME (decl) == heap_uninit_identifier
>>>> +		    || DECL_NAME (decl) == heap_vec_identifier
>>>> +		    || DECL_NAME (decl) == heap_vec_uninit_identifier))
>>>>    	      /* OK */;
>>>>    	    /* P2738 (C++26): a conversion from a prvalue P of type "pointer to
>>>>    	       cv void" to a pointer-to-object type T unless P points to an
>>>>    	       object whose type is similar to T.  */
>>>> -	    else if (cxx_dialect > cxx23
>>>> -		     && (sop = cxx_fold_indirect_ref (ctx, loc,
>>>> -						      TREE_TYPE (type), sop)))
>>>> +	    else if (cxx_dialect > cxx23)
>>>>    	      {
>>>> -		r = build1 (ADDR_EXPR, type, sop);
>>>> -		break;
>>>> +		r = cxx_fold_indirect_ref (ctx, loc, TREE_TYPE (type), sop);
>>>> +		if (r)
>>>> +		  {
>>>> +		    r = build1 (ADDR_EXPR, type, r);
>>>> +		    break;
>>>> +		  }
>>>> +		if (!ctx->quiet)
>>>> +		  {
>>>> +		    if (TREE_CODE (sop) == ADDR_EXPR)
>>>> +		      {
>>>> +			error_at (loc, "cast from %qT is not allowed because "
>>>> +				  "pointed-to type %qT is not similar to %qT",
>>>> +				  TREE_TYPE (op), TREE_TYPE (TREE_TYPE (sop)),
>>>> +				  TREE_TYPE (type));
>>>> +			tree obj = build_fold_indirect_ref (sop);
>>>> +			inform (DECL_SOURCE_LOCATION (obj),
>>>> +				"pointed-to object declared here");
>>>> +		      }
>>>> +		    else
>>>> +		      error_at (loc, "cast from %qT is not allowed",
>>>
>>> Can this be more helpful as well, i.e. say because op is not the address of
>>> an object of similar type?
>>>
>>> Can we only get here if op is null, since we would have returned already for
>>> non-constant op?
>>>
>>> Jason
>>>
>>
>> Hm, yes; I'd kept the error as it was because I wasn't sure what other
>> kinds of trees might end up here, but I've done a fair amount of testing
>> and I've only been able to reach here with null pointers: everything
>> else gets caught earlier. I've updated the error message appropriately
>> and documented this assumption with an assertion.
>>
>> Bootstrapped + regtested on x86_64-pc-linux-gnu as above.
>>
>> -- >8 --
>>
>> This patch improves the errors given when casting from void* in C++26 to
>> include the expected type if the types of the pointed-to objects were
>> not similar. It also ensures (for all standard modes) that void* casts
>> are checked even for DECL_ARTIFICIAL declarations, such as
>> lifetime-extended temporaries, and is only ignored for cases where we
>> know it's OK (e.g. source_location::current) or have no other choice
>> (heap-allocated data).
>>
>> gcc/cp/ChangeLog:
>>
>> 	* constexpr.cc (is_std_source_location_current): New.
>> 	(cxx_eval_constant_expression): Only ignore cast from void* for
>> 	specific cases and improve other diagnostics.
>>
>> gcc/testsuite/ChangeLog:
>>
>> 	* g++.dg/cpp0x/constexpr-cast4.C: New test.
>>
>> Signed-off-by: Nathaniel Shead <nathanieloshead@gmail.com>
>> ---
>>   gcc/cp/constexpr.cc                          | 87 +++++++++++++++++---
>>   gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C | 11 +++
>>   2 files changed, 86 insertions(+), 12 deletions(-)
>>   create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
>>
>> diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
>> index 0f948db7c2d..d060e374cb3 100644
>> --- a/gcc/cp/constexpr.cc
>> +++ b/gcc/cp/constexpr.cc
>> @@ -2301,6 +2301,36 @@ is_std_allocator_allocate (const constexpr_call *call)
>>   	  && is_std_allocator_allocate (call->fundef->decl));
>>   }
>>   
>> +/* Return true if FNDECL is std::source_location::current.  */
>> +
>> +static inline bool
>> +is_std_source_location_current (tree fndecl)
>> +{
>> +  if (!decl_in_std_namespace_p (fndecl))
>> +    return false;
>> +
>> +  tree name = DECL_NAME (fndecl);
>> +  if (name == NULL_TREE || !id_equal (name, "current"))
>> +    return false;
>> +
>> +  tree ctx = DECL_CONTEXT (fndecl);
>> +  if (ctx == NULL_TREE || !CLASS_TYPE_P (ctx) || !TYPE_MAIN_DECL (ctx))
>> +    return false;
>> +
>> +  name = DECL_NAME (TYPE_MAIN_DECL (ctx));
>> +  return name && id_equal (name, "source_location");
>> +}
>> +
>> +/* Overload for the above taking constexpr_call*.  */
>> +
>> +static inline bool
>> +is_std_source_location_current (const constexpr_call *call)
>> +{
>> +  return (call
>> +	  && call->fundef
>> +	  && is_std_source_location_current (call->fundef->decl));
>> +}
>> +
>>   /* Return true if FNDECL is __dynamic_cast.  */
>>   
>>   static inline bool
>> @@ -7850,33 +7880,66 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
>>   	if (TYPE_PTROB_P (type)
>>   	    && TYPE_PTR_P (TREE_TYPE (op))
>>   	    && VOID_TYPE_P (TREE_TYPE (TREE_TYPE (op)))
>> -	    /* Inside a call to std::construct_at or to
>> -	       std::allocator<T>::{,de}allocate, we permit casting from void*
>> +	    /* Inside a call to std::construct_at,
>> +	       std::allocator<T>::{,de}allocate, or
>> +	       std::source_location::current, we permit casting from void*
>>   	       because that is compiler-generated code.  */
>>   	    && !is_std_construct_at (ctx->call)
>> -	    && !is_std_allocator_allocate (ctx->call))
>> +	    && !is_std_allocator_allocate (ctx->call)
>> +	    && !is_std_source_location_current (ctx->call))
>>   	  {
>>   	    /* Likewise, don't error when casting from void* when OP is
>>   	       &heap uninit and similar.  */
>>   	    tree sop = tree_strip_nop_conversions (op);
>> -	    if (TREE_CODE (sop) == ADDR_EXPR
>> -		&& VAR_P (TREE_OPERAND (sop, 0))
>> -		&& DECL_ARTIFICIAL (TREE_OPERAND (sop, 0)))
>> +	    tree decl = NULL_TREE;
>> +	    if (TREE_CODE (sop) == ADDR_EXPR)
>> +	      decl = TREE_OPERAND (sop, 0);
>> +	    if (decl
>> +		&& VAR_P (decl)
>> +		&& DECL_ARTIFICIAL (decl)
>> +		&& (DECL_NAME (decl) == heap_identifier
>> +		    || DECL_NAME (decl) == heap_uninit_identifier
>> +		    || DECL_NAME (decl) == heap_vec_identifier
>> +		    || DECL_NAME (decl) == heap_vec_uninit_identifier))
>>   	      /* OK */;
>>   	    /* P2738 (C++26): a conversion from a prvalue P of type "pointer to
>>   	       cv void" to a pointer-to-object type T unless P points to an
>>   	       object whose type is similar to T.  */
>> -	    else if (cxx_dialect > cxx23
>> -		     && (sop = cxx_fold_indirect_ref (ctx, loc,
>> -						      TREE_TYPE (type), sop)))
>> +	    else if (cxx_dialect > cxx23)
>>   	      {
>> -		r = build1 (ADDR_EXPR, type, sop);
>> -		break;
>> +		r = cxx_fold_indirect_ref (ctx, loc, TREE_TYPE (type), sop);
>> +		if (r)
>> +		  {
>> +		    r = build1 (ADDR_EXPR, type, r);
>> +		    break;
>> +		  }
>> +		if (!ctx->quiet)
>> +		  {
>> +		    if (TREE_CODE (sop) == ADDR_EXPR)
>> +		      {
> 
> A very minor point (sorry), but I think you want
> 
>    auto_diagnostic_group d;
> 
> here.  Not need to repost the patch only because of this.

I'm applying this patch with that tweak and also adjusting "not allowed" 
to "not allowed in a constant expression".  Thanks!
diff mbox series

Patch

diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
index 0f948db7c2d..d060e374cb3 100644
--- a/gcc/cp/constexpr.cc
+++ b/gcc/cp/constexpr.cc
@@ -2301,6 +2301,36 @@  is_std_allocator_allocate (const constexpr_call *call)
 	  && is_std_allocator_allocate (call->fundef->decl));
 }
 
+/* Return true if FNDECL is std::source_location::current.  */
+
+static inline bool
+is_std_source_location_current (tree fndecl)
+{
+  if (!decl_in_std_namespace_p (fndecl))
+    return false;
+
+  tree name = DECL_NAME (fndecl);
+  if (name == NULL_TREE || !id_equal (name, "current"))
+    return false;
+
+  tree ctx = DECL_CONTEXT (fndecl);
+  if (ctx == NULL_TREE || !CLASS_TYPE_P (ctx) || !TYPE_MAIN_DECL (ctx))
+    return false;
+
+  name = DECL_NAME (TYPE_MAIN_DECL (ctx));
+  return name && id_equal (name, "source_location");
+}
+
+/* Overload for the above taking constexpr_call*.  */
+
+static inline bool
+is_std_source_location_current (const constexpr_call *call)
+{
+  return (call
+	  && call->fundef
+	  && is_std_source_location_current (call->fundef->decl));
+}
+
 /* Return true if FNDECL is __dynamic_cast.  */
 
 static inline bool
@@ -7850,33 +7880,66 @@  cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
 	if (TYPE_PTROB_P (type)
 	    && TYPE_PTR_P (TREE_TYPE (op))
 	    && VOID_TYPE_P (TREE_TYPE (TREE_TYPE (op)))
-	    /* Inside a call to std::construct_at or to
-	       std::allocator<T>::{,de}allocate, we permit casting from void*
+	    /* Inside a call to std::construct_at,
+	       std::allocator<T>::{,de}allocate, or
+	       std::source_location::current, we permit casting from void*
 	       because that is compiler-generated code.  */
 	    && !is_std_construct_at (ctx->call)
-	    && !is_std_allocator_allocate (ctx->call))
+	    && !is_std_allocator_allocate (ctx->call)
+	    && !is_std_source_location_current (ctx->call))
 	  {
 	    /* Likewise, don't error when casting from void* when OP is
 	       &heap uninit and similar.  */
 	    tree sop = tree_strip_nop_conversions (op);
-	    if (TREE_CODE (sop) == ADDR_EXPR
-		&& VAR_P (TREE_OPERAND (sop, 0))
-		&& DECL_ARTIFICIAL (TREE_OPERAND (sop, 0)))
+	    tree decl = NULL_TREE;
+	    if (TREE_CODE (sop) == ADDR_EXPR)
+	      decl = TREE_OPERAND (sop, 0);
+	    if (decl
+		&& VAR_P (decl)
+		&& DECL_ARTIFICIAL (decl)
+		&& (DECL_NAME (decl) == heap_identifier
+		    || DECL_NAME (decl) == heap_uninit_identifier
+		    || DECL_NAME (decl) == heap_vec_identifier
+		    || DECL_NAME (decl) == heap_vec_uninit_identifier))
 	      /* OK */;
 	    /* P2738 (C++26): a conversion from a prvalue P of type "pointer to
 	       cv void" to a pointer-to-object type T unless P points to an
 	       object whose type is similar to T.  */
-	    else if (cxx_dialect > cxx23
-		     && (sop = cxx_fold_indirect_ref (ctx, loc,
-						      TREE_TYPE (type), sop)))
+	    else if (cxx_dialect > cxx23)
 	      {
-		r = build1 (ADDR_EXPR, type, sop);
-		break;
+		r = cxx_fold_indirect_ref (ctx, loc, TREE_TYPE (type), sop);
+		if (r)
+		  {
+		    r = build1 (ADDR_EXPR, type, r);
+		    break;
+		  }
+		if (!ctx->quiet)
+		  {
+		    if (TREE_CODE (sop) == ADDR_EXPR)
+		      {
+			error_at (loc, "cast from %qT is not allowed because "
+				  "pointed-to type %qT is not similar to %qT",
+				  TREE_TYPE (op), TREE_TYPE (TREE_TYPE (sop)),
+				  TREE_TYPE (type));
+			tree obj = build_fold_indirect_ref (sop);
+			inform (DECL_SOURCE_LOCATION (obj),
+				"pointed-to object declared here");
+		      }
+		    else
+		      {
+			gcc_assert (integer_zerop (sop));
+			error_at (loc, "cast from %qT is not allowed because "
+				  "%qE does not point to an object",
+				  TREE_TYPE (op), oldop);
+		      }
+		  }
+		*non_constant_p = true;
+		return t;
 	      }
 	    else
 	      {
 		if (!ctx->quiet)
-		  error_at (loc, "cast from %qT is not allowed",
+		  error_at (loc, "cast from %qT is not allowed before C++26",
 			    TREE_TYPE (op));
 		*non_constant_p = true;
 		return t;
diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C b/gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
new file mode 100644
index 00000000000..cfeaa1ac3bf
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-cast4.C
@@ -0,0 +1,11 @@ 
+// { dg-do compile { target c++11 } }
+
+constexpr int&& r = 1 + 2;  // { dg-message "pointed-to object declared here" "" { target c++26 } }
+constexpr void* vpr = &r;
+constexpr int* pi = static_cast<int*>(vpr);  // { dg-error "cast from .void\\*. is not allowed" "" { target c++23_down } }
+constexpr float* pf = static_cast<float*>(vpr);  // { dg-error "cast from .void\\*. is not allowed" "" { target c++23_down } }
+// { dg-error "cast from .void\\*. is not allowed because pointed-to type .int. is not similar to .float." "" { target c++26 } .-1 }
+
+constexpr void* vnp = nullptr;
+constexpr int* pi2 = static_cast<int*>(vnp);  // { dg-error "cast from .void\\*. is not allowed" "" { target c++23_down } }
+// { dg-error "cast from .void\\*. is not allowed because .vnp. does not point to an object" "" { target c++26 } .-1 }