On Tue, Sep 17, 2024 at 05:16:47PM -0400, Jason Merrill wrote:
> Tested x86_64-pc-linux-gnu.  Marek, any concerns?

LGTM, thanks.  The ChangeLog may not pass the hook because it's
missing some entries.
 
> -- 8< --
> 
> We can't have a dangling reference to an empty class unless it's
> specifically to that class or one of its bases.  This was giving a
> false positive on the _ExtractKey pattern in libstdc++ hashtable.h.
> 
> This also adjusts the order of arguments to reference_related_p, which
> is relevant for empty classes (unlike scalars).
> 
> Several of the classes in the testsuite needed to gain data members to
> continue to warn.
> 
>       PR c++/115361
> 
> gcc/cp/ChangeLog:
> 
>       * call.cc (do_warn_dangling_reference): Check is_empty_class.
> 
> gcc/testsuite/ChangeLog:
> 
>       * g++.dg/ext/attr-no-dangling6.C:
>       * g++.dg/ext/attr-no-dangling7.C:
>       * g++.dg/ext/attr-no-dangling8.C:
>       * g++.dg/ext/attr-no-dangling9.C:
>       * g++.dg/warn/Wdangling-reference1.C:
>       * g++.dg/warn/Wdangling-reference2.C:
>       * g++.dg/warn/Wdangling-reference3.C: Make classes non-empty.
>       * g++.dg/warn/Wdangling-reference23.C: New test.
> ---
>  gcc/cp/call.cc                                    | 12 +++++++-----
>  gcc/testsuite/g++.dg/ext/attr-no-dangling6.C      |  6 +++---
>  gcc/testsuite/g++.dg/ext/attr-no-dangling7.C      |  6 +++---
>  gcc/testsuite/g++.dg/ext/attr-no-dangling8.C      |  2 ++
>  gcc/testsuite/g++.dg/ext/attr-no-dangling9.C      |  1 +
>  gcc/testsuite/g++.dg/warn/Wdangling-reference1.C  |  1 +
>  gcc/testsuite/g++.dg/warn/Wdangling-reference2.C  |  2 +-
>  gcc/testsuite/g++.dg/warn/Wdangling-reference23.C | 14 ++++++++++++++
>  gcc/testsuite/g++.dg/warn/Wdangling-reference3.C  |  1 +
>  9 files changed, 33 insertions(+), 12 deletions(-)
>  create mode 100644 gcc/testsuite/g++.dg/warn/Wdangling-reference23.C
> 
> diff --git a/gcc/cp/call.cc b/gcc/cp/call.cc
> index 664088eed9c..1ecf3aac705 100644
> --- a/gcc/cp/call.cc
> +++ b/gcc/cp/call.cc
> @@ -14356,12 +14356,14 @@ do_warn_dangling_reference (tree expr, bool arg_p)
>           if ((arg = do_warn_dangling_reference (arg, /*arg_p=*/true)))
>             {
>               /* If we know the temporary could not bind to the return type,
> -                don't warn.  This is for scalars only because for classes
> -                we can't be sure we are not returning its sub-object.  */
> -             if (SCALAR_TYPE_P (TREE_TYPE (arg))
> +                don't warn.  This is for scalars and empty classes only
> +                because for other classes we can't be sure we are not
> +                returning its sub-object.  */
> +             if ((SCALAR_TYPE_P (TREE_TYPE (arg))
> +                  || is_empty_class (TREE_TYPE (arg)))
>                   && TYPE_REF_P (rettype)
> -                 && !reference_related_p (TREE_TYPE (arg),
> -                                          TREE_TYPE (rettype)))
> +                 && !reference_related_p (TREE_TYPE (rettype),
> +                                          TREE_TYPE (arg)))
>                 continue;
>               return expr;
>             }
> diff --git a/gcc/testsuite/g++.dg/ext/attr-no-dangling6.C 
> b/gcc/testsuite/g++.dg/ext/attr-no-dangling6.C
> index 5b349e8e682..1fc426d20d3 100644
> --- a/gcc/testsuite/g++.dg/ext/attr-no-dangling6.C
> +++ b/gcc/testsuite/g++.dg/ext/attr-no-dangling6.C
> @@ -2,9 +2,9 @@
>  // { dg-do compile { target c++20 } }
>  // { dg-options "-Wdangling-reference" }
>  
> -class X { };
> -const X x1;
> -const X x2;
> +class X { int i; };
> +const X x1 {};
> +const X x2 {};
>  
>  constexpr bool val () { return true; }
>  struct ST { static constexpr bool value = true; };
> diff --git a/gcc/testsuite/g++.dg/ext/attr-no-dangling7.C 
> b/gcc/testsuite/g++.dg/ext/attr-no-dangling7.C
> index a5fb809e6bd..04c6badf0b6 100644
> --- a/gcc/testsuite/g++.dg/ext/attr-no-dangling7.C
> +++ b/gcc/testsuite/g++.dg/ext/attr-no-dangling7.C
> @@ -2,9 +2,9 @@
>  // { dg-do compile { target c++20 } }
>  // { dg-options "-Wdangling-reference" }
>  
> -class X { };
> -const X x1;
> -const X x2;
> +class X { int i; };
> +const X x1 {};
> +const X x2 {};
>  
>  template<bool... N>
>  [[gnu::no_dangling(N)]] const X& get(const int& i); // { dg-error "parameter 
> packs not expanded" }
> diff --git a/gcc/testsuite/g++.dg/ext/attr-no-dangling8.C 
> b/gcc/testsuite/g++.dg/ext/attr-no-dangling8.C
> index 8208d751a4b..aa196315a38 100644
> --- a/gcc/testsuite/g++.dg/ext/attr-no-dangling8.C
> +++ b/gcc/testsuite/g++.dg/ext/attr-no-dangling8.C
> @@ -8,6 +8,7 @@ template<class T> constexpr bool is_reference_v<T&&> = true;
>  
>  template <typename T>
>  struct [[gnu::no_dangling(is_reference_v<T>)]] S {
> +  int i;
>    int &foo (const int &);
>  };
>  
> @@ -15,6 +16,7 @@ template <typename T1, typename T2>
>  struct X {
>    template <typename U1 = T1, typename U2 = T2>
>    struct [[gnu::no_dangling(is_reference_v<U1> && is_reference_v<U2>)]] Y {
> +    int i;
>      int &foo (const int &);
>    };
>  };
> diff --git a/gcc/testsuite/g++.dg/ext/attr-no-dangling9.C 
> b/gcc/testsuite/g++.dg/ext/attr-no-dangling9.C
> index 65b4f7145a9..d7fd897de53 100644
> --- a/gcc/testsuite/g++.dg/ext/attr-no-dangling9.C
> +++ b/gcc/testsuite/g++.dg/ext/attr-no-dangling9.C
> @@ -12,6 +12,7 @@ using true_type = bool_constant<true>;
>  using false_type = bool_constant<false>;
>  
>  struct S {
> +  int i;
>    template<bool B>
>    [[gnu::no_dangling(B)]] int &foo (const int &);
>  };
> diff --git a/gcc/testsuite/g++.dg/warn/Wdangling-reference1.C 
> b/gcc/testsuite/g++.dg/warn/Wdangling-reference1.C
> index 1718c28165e..a184317dd5c 100644
> --- a/gcc/testsuite/g++.dg/warn/Wdangling-reference1.C
> +++ b/gcc/testsuite/g++.dg/warn/Wdangling-reference1.C
> @@ -131,6 +131,7 @@ int n = 1;
>  const int& refmax = max(n - 1, n + 1); // { dg-warning "dangling reference" }
>  
>  struct Y {
> +  int i;
>    operator int&();
>    operator int&&();
>    const int& foo(const int&);
> diff --git a/gcc/testsuite/g++.dg/warn/Wdangling-reference2.C 
> b/gcc/testsuite/g++.dg/warn/Wdangling-reference2.C
> index dafdb43f1b9..a3d5ad6d867 100644
> --- a/gcc/testsuite/g++.dg/warn/Wdangling-reference2.C
> +++ b/gcc/testsuite/g++.dg/warn/Wdangling-reference2.C
> @@ -3,7 +3,7 @@
>  // { dg-options "-Wdangling-reference" }
>  
>  namespace std {
> -struct any {};
> +struct any { void *p; ~any(); };
>  template <typename _ValueType> _ValueType any_cast(any &&);
>  template <typename _Tp> struct remove_reference { using type = _Tp; };
>  template <typename _Tp> _Tp forward(typename remove_reference<_Tp>::type);
> diff --git a/gcc/testsuite/g++.dg/warn/Wdangling-reference23.C 
> b/gcc/testsuite/g++.dg/warn/Wdangling-reference23.C
> new file mode 100644
> index 00000000000..e59ccc5057b
> --- /dev/null
> +++ b/gcc/testsuite/g++.dg/warn/Wdangling-reference23.C
> @@ -0,0 +1,14 @@
> +// PR c++/115361
> +// { dg-additional-options -Wdangling-reference }
> +
> +struct B { int i; };
> +
> +struct A {
> +  const int & operator()(const B& b) { return b.i; }
> +};
> +
> +int main()
> +{
> +  B b = {};
> +  const int &r = A()(b);
> +}
> diff --git a/gcc/testsuite/g++.dg/warn/Wdangling-reference3.C 
> b/gcc/testsuite/g++.dg/warn/Wdangling-reference3.C
> index 4bc20c13b3f..7db1dc86855 100644
> --- a/gcc/testsuite/g++.dg/warn/Wdangling-reference3.C
> +++ b/gcc/testsuite/g++.dg/warn/Wdangling-reference3.C
> @@ -18,6 +18,7 @@ struct G {
>  };
>  
>  struct F {
> +  int i;
>    G& f();
>  };
>  
> 
> base-commit: a9f9391e1eeccb9d062b9e73ef27ac577b23ed64
> -- 
> 2.46.0
> 

Marek

Reply via email to