aboutsummaryrefslogtreecommitdiff
path: root/gcc
diff options
context:
space:
mode:
authorJakub Jelinek <jakub@redhat.com>2020-03-26 09:31:15 +0100
committerJakub Jelinek <jakub@redhat.com>2020-03-26 09:31:15 +0100
commitdab932d1519ba07fb4c49e6849ee7ceb02c0d603 (patch)
treea74636b0d3ef492d71bcfc0cd9dd9f65c3a554ed /gcc
parent5a1706f63a2024a5c2d878f2efeb8d198214542f (diff)
downloadgcc-dab932d1519ba07fb4c49e6849ee7ceb02c0d603.zip
gcc-dab932d1519ba07fb4c49e6849ee7ceb02c0d603.tar.gz
gcc-dab932d1519ba07fb4c49e6849ee7ceb02c0d603.tar.bz2
c++: Fix up user_provided_p [PR81349]
The standard says: "A function is user-provided if it is user-declared and not explicitly defaulted or deleted on its first declaration." I don't see anything about function templates having different rules here, but user_provided_p does return true for all TEMPLATE_DECLs. The following patch fixes it by treating as user-provided only templates that aren't deleted. 2020-03-26 Jakub Jelinek <jakub@redhat.com> PR c++/81349 * class.c (user_provided_p): Use STRIP_TEMPLATE instead of returning true for all TEMPLATE_DECLs. * g++.dg/cpp1z/pr81349.C: New test.
Diffstat (limited to 'gcc')
-rw-r--r--gcc/cp/ChangeLog4
-rw-r--r--gcc/cp/class.c10
-rw-r--r--gcc/testsuite/ChangeLog3
-rw-r--r--gcc/testsuite/g++.dg/cpp1z/pr81349.C29
4 files changed, 40 insertions, 6 deletions
diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog
index 9ff2525..af8f49b 100644
--- a/gcc/cp/ChangeLog
+++ b/gcc/cp/ChangeLog
@@ -1,5 +1,9 @@
2020-03-26 Jakub Jelinek <jakub@redhat.com>
+ PR c++/81349
+ * class.c (user_provided_p): Use STRIP_TEMPLATE instead of returning
+ true for all TEMPLATE_DECLs.
+
PR c++/94272
* cp-gimplify.c (cp_genericize_r): Handle STATEMENT_LIST.
diff --git a/gcc/cp/class.c b/gcc/cp/class.c
index 5340799..41f52e5 100644
--- a/gcc/cp/class.c
+++ b/gcc/cp/class.c
@@ -5159,12 +5159,10 @@ in_class_defaulted_default_constructor (tree t)
bool
user_provided_p (tree fn)
{
- if (TREE_CODE (fn) == TEMPLATE_DECL)
- return true;
- else
- return (!DECL_ARTIFICIAL (fn)
- && !(DECL_INITIALIZED_IN_CLASS_P (fn)
- && (DECL_DEFAULTED_FN (fn) || DECL_DELETED_FN (fn))));
+ fn = STRIP_TEMPLATE (fn);
+ return (!DECL_ARTIFICIAL (fn)
+ && !(DECL_INITIALIZED_IN_CLASS_P (fn)
+ && (DECL_DEFAULTED_FN (fn) || DECL_DELETED_FN (fn))));
}
/* Returns true iff class T has a user-provided constructor. */
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index 6d5c12f..6091bcc 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,5 +1,8 @@
2020-03-26 Jakub Jelinek <jakub@redhat.com>
+ PR c++/81349
+ * g++.dg/cpp1z/pr81349.C: New test.
+
PR c++/94272
* g++.dg/debug/pr94272.C: New test.
diff --git a/gcc/testsuite/g++.dg/cpp1z/pr81349.C b/gcc/testsuite/g++.dg/cpp1z/pr81349.C
new file mode 100644
index 0000000..5195d19
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp1z/pr81349.C
@@ -0,0 +1,29 @@
+// PR c++/81349
+// { dg-do compile { target c++17_only } }
+
+#include <type_traits>
+
+struct A {
+ A (int) = delete;
+};
+
+struct B {
+ template <typename T>
+ B (T) = delete;
+};
+
+template <typename U>
+struct C {
+ C (U) = delete;
+};
+
+template <typename U>
+struct D {
+ template <typename T>
+ D (T, U) = delete;
+};
+
+static_assert (std::is_aggregate_v<A>);
+static_assert (std::is_aggregate_v<B>);
+static_assert (std::is_aggregate_v<C<int>>);
+static_assert (std::is_aggregate_v<D<int>>);