зеркало из https://github.com/microsoft/clang-1.git
C++98/03 [temp.friend]p4 requires that inline function definitions
within class templates be instantiated along with each class template specialization, even if the functions are not used. Do so, as a baby step toward PR6952. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@103943 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Родитель
3d91bbcdab
Коммит
ac7c2c8a9d
|
@ -1009,6 +1009,7 @@ Decl *TemplateDeclInstantiator::VisitFunctionDecl(FunctionDecl *D,
|
|||
Params[P]->setOwningFunction(Function);
|
||||
Function->setParams(Params.data(), Params.size());
|
||||
|
||||
SourceLocation InstantiateAtPOI;
|
||||
if (TemplateParams) {
|
||||
// Our resulting instantiation is actually a function template, since we
|
||||
// are substituting only the outer template parameters. For example, given
|
||||
|
@ -1047,6 +1048,23 @@ Decl *TemplateDeclInstantiator::VisitFunctionDecl(FunctionDecl *D,
|
|||
// TODO: should we remember this connection regardless of whether
|
||||
// the friend declaration provided a body?
|
||||
Function->setInstantiationOfMemberFunction(D, TSK_ImplicitInstantiation);
|
||||
if (!SemaRef.getLangOptions().CPlusPlus0x) {
|
||||
// C++03 [temp.friend]p4:
|
||||
// When a function is defined in a friend function declaration in a
|
||||
// class template, the function is defined at each instantiation of the
|
||||
// class template. The function is defined even if it is never used.
|
||||
if (CXXRecordDecl *Record = dyn_cast<CXXRecordDecl>(Owner)) {
|
||||
if (ClassTemplateSpecializationDecl *Spec
|
||||
= dyn_cast<ClassTemplateSpecializationDecl>(Record))
|
||||
InstantiateAtPOI = Spec->getPointOfInstantiation();
|
||||
else if (MemberSpecializationInfo *MSInfo
|
||||
= Record->getMemberSpecializationInfo())
|
||||
InstantiateAtPOI = MSInfo->getPointOfInstantiation();
|
||||
}
|
||||
|
||||
if (InstantiateAtPOI.isInvalid())
|
||||
InstantiateAtPOI = Function->getLocation();
|
||||
}
|
||||
}
|
||||
|
||||
if (InitFunctionInstantiation(Function, D))
|
||||
|
@ -1133,6 +1151,11 @@ Decl *TemplateDeclInstantiator::VisitFunctionDecl(FunctionDecl *D,
|
|||
PrincipalDecl->isInIdentifierNamespace(Decl::IDNS_Ordinary))
|
||||
PrincipalDecl->setNonMemberOperator();
|
||||
|
||||
// If we need to instantiate this function now (because it is a C++98/03
|
||||
// friend function defined inside a class template), do so.
|
||||
if (InstantiateAtPOI.isValid())
|
||||
SemaRef.MarkDeclarationReferenced(InstantiateAtPOI, Function);
|
||||
|
||||
return Function;
|
||||
}
|
||||
|
||||
|
|
|
@ -188,8 +188,8 @@ namespace test4 {
|
|||
|
||||
struct Inequal {};
|
||||
bool test() {
|
||||
Holder<Inequal> a, b;
|
||||
return a == b; // expected-note {{requested here}}
|
||||
Holder<Inequal> a, b; // expected-note {{requested here}}
|
||||
return a == b;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -0,0 +1,9 @@
|
|||
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
||||
|
||||
template<typename T>
|
||||
struct X {
|
||||
friend void f(int x) { T* y = x; } // expected-error{{cannot initialize a variable of type 'int *' with an lvalue of type 'int'}}
|
||||
};
|
||||
|
||||
X<int> xi; // expected-note{{in instantiation of member function 'f' requested here}}
|
||||
|
Загрузка…
Ссылка в новой задаче