Diag(Function->getLocation(), diag::err_redefinition)
<< Function->getDeclName();
Diag(Definition->getLocation(), diag::note_previous_definition);
- }
+ Function->setInvalidDecl();
+ }
// We have an explicit instantiation (which already occurred) and an
// implicit instantiation. Return without complaint.
if (PatternDecl)
Diag(PatternDecl->getLocation(),
diag::note_explicit_instantiation_here);
+ Function->setInvalidDecl();
}
return;
}
+ // If this is an instantiation of friend function defined within a class
+ // template or class template specialization or member class thereof,
+ // determine whether there were multiple instantiations of its lexical class.
+ if (PatternDecl->getFriendObjectKind() != Decl::FOK_None) {
+ for (FunctionDecl::redecl_iterator R = Function->redecls_begin(),
+ REnd = Function->redecls_end();
+ R != REnd; ++R) {
+ if (*R != Function &&
+ ((*R)->getFriendObjectKind() != Decl::FOK_None)) {
+ if (const FunctionDecl *RPattern
+ = (*R)->getTemplateInstantiationPattern())
+ if (RPattern->getBody(RPattern)) {
+ InstantiatingTemplate Inst(*this, PointOfInstantiation, Function);
+ if (Inst)
+ return;
+
+ Diag(Function->getLocation(), diag::err_redefinition)
+ << Function->getDeclName();
+ Diag((*R)->getLocation(), diag::note_previous_definition);
+ Function->setInvalidDecl();
+ return;
+ }
+ }
+ }
+ }
+
// C++0x [temp.explicit]p9:
// Except for inline functions, other explicit instantiation declarations
// have the effect of suppressing the implicit instantiation of the entity
friend void f3(T) { } // expected-error{{redefinition of}}
friend void f4(T) { } // expected-error{{redefinition of}}
friend void f5(T) { } // expected-error{{redefinition of}}
-
- // FIXME: should have a redefinition error for f6(int)
- friend void f6(int) { }
+ friend void f6(int) { } // expected-error{{redefinition of}} \
+ // expected-note{{previous definition}}
};
void f2(int) { } // expected-note{{previous definition}}
void f4(int) { } // expected-note{{previous definition}}
-X1<int> x1a; // expected-note 6{{in instantiation of}}
+X1<int> x1a; // expected-note 7{{in instantiation of}}
void f3(int) { } // expected-note{{previous definition}}
void f5(int) { } // expected-note{{previous definition}}
-X1<float> x1b;
+X1<float> x1b;
X1<double> *X0d() { return 0;}
+
+template<typename T>
+struct X2 {
+ friend void g0(T) { } // expected-error{{redefinition of 'g0'}}
+};
+
+template<typename T>
+struct X3 {
+ friend void g0(T) { } // expected-note{{previous definition is here}}
+};
+
+X2<float> x2; // expected-note{{in instantiation of}}
+X3<float> x3;