diff options
author | Patrick Palka <ppalka@redhat.com> | 2021-01-20 09:44:33 -0500 |
---|---|---|
committer | Patrick Palka <ppalka@redhat.com> | 2021-01-20 09:44:33 -0500 |
commit | cafcfcb5840b62d9fc80c12192189351e995a4f2 (patch) | |
tree | 74a8e556110a58df2190f0b14ae701e68cfc3fde | |
parent | 79e1251b642db038df276153c9f2ec6b82e56162 (diff) | |
download | gcc-cafcfcb5840b62d9fc80c12192189351e995a4f2.zip gcc-cafcfcb5840b62d9fc80c12192189351e995a4f2.tar.gz gcc-cafcfcb5840b62d9fc80c12192189351e995a4f2.tar.bz2 |
c++: Fix tsubsting CLASS_PLACEHOLDER_TEMPLATE [PR95434]
Here, during partial instantiation of the generic lambda, we do
tsubst_copy on the CLASS_PLACEHOLDER_TEMPLATE for U{0} which yields a
(level-lowered) TEMPLATE_TEMPLATE_PARM rather than the corresponding
TEMPLATE_DECL. This later confuses do_class_deduction which expects
that a CLASS_PLACEHOLDER_TEMPLATE is always a TEMPLATE_DECL.
gcc/cp/ChangeLog:
PR c++/95434
* pt.c (tsubst) <case TEMPLATE_TYPE_PARM>: If tsubsting
CLASS_PLACEHOLDER_TEMPLATE yields a TEMPLATE_TEMPLATE_PARM,
adjust to its TEMPLATE_TEMPLATE_PARM_TEMPLATE_DECL.
gcc/testsuite/ChangeLog:
PR c++/95434
* g++.dg/cpp2a/lambda-generic9.C: New test.
-rw-r--r-- | gcc/cp/pt.c | 2 | ||||
-rw-r--r-- | gcc/testsuite/g++.dg/cpp2a/lambda-generic9.C | 9 |
2 files changed, 11 insertions, 0 deletions
diff --git a/gcc/cp/pt.c b/gcc/cp/pt.c index bcf4c6e..373f827 100644 --- a/gcc/cp/pt.c +++ b/gcc/cp/pt.c @@ -15688,6 +15688,8 @@ tsubst (tree t, tree args, tsubst_flags_t complain, tree in_decl) else if (tree pl = CLASS_PLACEHOLDER_TEMPLATE (t)) { pl = tsubst_copy (pl, args, complain, in_decl); + if (TREE_CODE (pl) == TEMPLATE_TEMPLATE_PARM) + pl = TEMPLATE_TEMPLATE_PARM_TEMPLATE_DECL (pl); CLASS_PLACEHOLDER_TEMPLATE (r) = pl; } } diff --git a/gcc/testsuite/g++.dg/cpp2a/lambda-generic9.C b/gcc/testsuite/g++.dg/cpp2a/lambda-generic9.C new file mode 100644 index 0000000..20ceb37 --- /dev/null +++ b/gcc/testsuite/g++.dg/cpp2a/lambda-generic9.C @@ -0,0 +1,9 @@ +// PR c++/95434 +// { dg-do compile { target c++20 } } + +template <class> +void f() { + [] <template <class> class U> { U{0}; }; +} + +template void f<int>(); |