Erich Keane 06bd74ba4a
Fix implementation of [temp.param]p14's first sentence. (#83487)
The first sentence says:

If a template-parameter of a class template, variable template, or alias
template has a default template-argument, each subsequent
template-parameter shall either have a default template-argument
supplied or be a template parameter pack.

However, we were only testing for "not a function function template",
and referring to an older version of the standard. As far as I can tell,
CWG2032 added the variable-template, and the alias-template pre-dates
the standard on github.

This patch started as a bug fix for #83461 , but ended up fixing a
number of similar cases, so those are validated as well.
2024-03-01 07:27:06 -08:00

61 lines
1.8 KiB
C++

// RUN: %clang_cc1 -std=c++20 -fsyntax-only -verify %s
struct S {
template<typename Ty = int>
friend void foo(auto){}
template<typename Ty = int, typename Tz>
friend void foo2(){}
};
template<typename T>
struct TemplS {
template<typename Ty = int>
friend void foo3(auto){}
template<typename Ty = int, typename Tz>
friend void foo4(){}
};
void Inst() {
TemplS<int>();
}
// expected-error@+2{{template parameter missing a default argument}}
// expected-note@+1{{previous default template argument defined here}}
template<typename T = int, typename U>
struct ClassTempl{};
struct HasFriendClassTempl {
// expected-error@+1{{default template argument not permitted on a friend template}}
template<typename T = int, typename U>
friend struct Friend;
// expected-error@+3{{cannot define a type in a friend declaration}}
// expected-error@+1{{default template argument not permitted on a friend template}}
template<typename T = int, typename U>
friend struct Friend2{};
};
template<typename Ty>
struct HasFriendClassTempl2 {
// expected-error@+3{{template parameter missing a default argument}}
// expected-note@+2{{previous default template argument defined here}}
// expected-note@#INST2{{in instantiation of template class}}
template<typename T = int, typename U>
friend struct Friend;
};
void Inst2() {
HasFriendClassTempl2<int>(); // #INST2
}
// expected-error@+2{{template parameter missing a default argument}}
// expected-note@+1{{previous default template argument defined here}}
template<typename T = int, typename U>
static constexpr U VarTempl;
// expected-error@+2{{template parameter missing a default argument}}
// expected-note@+1{{previous default template argument defined here}}
template<typename T = int, typename U>
using TypeAlias = U;