On 10/4/20 11:28 PM, Patrick Palka wrote:
cp_tree_equal currently considers alignof the same as __alignof__, but
these operators are semantically different ever since r8-7957.  In the
testcase below, this causes the second static_assert to fail on targets
where alignof(double) != __alignof__(double) because the specialization
cache (which uses cp_tree_equal as the equality predicate) conflates the
two dependent specializations integral_constant<__alignof__(T)> and
integral_constant<alignof(T)>.

This patch makes cp_tree_equal distinguish between these two operators
by inspecting the ALIGNOF_EXPR_STD_P flag.

Bootstrapped and regtested on x86_64-pc-linux-gnu, and also verified
that we now correctly compile the  PR97273 testcase, does this look OK
for trunk and the release branches?

OK.

gcc/cp/ChangeLog:

        PR c++/88115
        PR libstdc++/97273
        * tree.c (cp_tree_equal) <case ALIGNOF_EXPR>: Return false if
        ALIGNOF_EXPR_STD_P differ.

gcc/testsuite/ChangeLog:

        PR c++/88115
        PR libstdc++/97273
        * g++.dg/template/alignof3.C: New test.
---
  gcc/cp/tree.c                            |  2 ++
  gcc/testsuite/g++.dg/template/alignof3.C | 13 +++++++++++++
  2 files changed, 15 insertions(+)
  create mode 100644 gcc/testsuite/g++.dg/template/alignof3.C

diff --git a/gcc/cp/tree.c b/gcc/cp/tree.c
index 8b7c6798ee9..a3fc5372cb7 100644
--- a/gcc/cp/tree.c
+++ b/gcc/cp/tree.c
@@ -3800,6 +3800,8 @@ cp_tree_equal (tree t1, tree t2)
            if (SIZEOF_EXPR_TYPE_P (t2))
              o2 = TREE_TYPE (o2);
          }
+       else if (ALIGNOF_EXPR_STD_P (t1) != ALIGNOF_EXPR_STD_P (t2))
+         return false;
if (TREE_CODE (o1) != TREE_CODE (o2))
          return false;
diff --git a/gcc/testsuite/g++.dg/template/alignof3.C 
b/gcc/testsuite/g++.dg/template/alignof3.C
new file mode 100644
index 00000000000..e573727c5f2
--- /dev/null
+++ b/gcc/testsuite/g++.dg/template/alignof3.C
@@ -0,0 +1,13 @@
+// PR c++/88115
+// { dg-do compile { target c++11 } }
+
+template<int __v>
+struct integral_constant {
+  static constexpr int value = __v;
+};
+
+template <class T> using StdAlignOf = integral_constant<alignof(T)>;
+template <class T> using GCCAlignOf = integral_constant<__alignof__(T)>;
+
+static_assert(StdAlignOf<double>::value == alignof(double), "");
+static_assert(GCCAlignOf<double>::value == __alignof__(double), "");


Reply via email to