Do not implicitly instantiate the definition of a class template specialization

that has been explicitly specialized!

We assume in various places that we can tell the template specialization kind
of a class type by looking at the declaration produced by TagType::getDecl.
That was previously not quite true: for an explicit specialization, we could
have first seen a template-id denoting the specialization (with a use that does
not trigger an implicit instantiation of the defintiion) and then seen the
first explicit specialization declaration. TagType::getDecl would previously
return an arbitrary declaration when called on a not-yet-defined class; it
now consistently returns the most recent declaration in that case.

llvm-svn: 295118
This commit is contained in:
Richard Smith 2017-02-14 23:27:44 +00:00
parent 32c5004cf5
commit 1d5f95f52f
2 changed files with 15 additions and 2 deletions

View File

@ -3023,8 +3023,10 @@ static TagDecl *getInterestingTagDecl(TagDecl *decl) {
if (I->isCompleteDefinition() || I->isBeingDefined()) if (I->isCompleteDefinition() || I->isBeingDefined())
return I; return I;
} }
// If there's no definition (not even in progress), return what we have. // If there's no definition (not even in progress), return the most recent
return decl; // declaration. This is important for template specializations, in order to
// pick the declaration with the most complete TemplateSpecializationKind.
return decl->getMostRecentDecl();
} }
TagDecl *TagType::getDecl() const { TagDecl *TagType::getDecl() const {

View File

@ -57,3 +57,14 @@ template<typename T> struct Helper {
template<typename T> void Helper<T>::func<2>() {} // expected-error {{cannot specialize a member}} \ template<typename T> void Helper<T>::func<2>() {} // expected-error {{cannot specialize a member}} \
// expected-error {{no function template matches}} // expected-error {{no function template matches}}
} }
namespace b35070233 {
template <typename T> struct Cls {
static void f() {}
};
void g(Cls<int>);
template<> struct Cls<int>; // expected-note {{forward declaration}}
template<> void Cls<int>::f(); // expected-error {{incomplete type}}
}