зеркало из https://github.com/microsoft/clang.git
Fix PR10053: Improve diagnostics and error recovery for code which some compilers incorrectly accept due to a lack of proper support for two-phase name lookup.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@132672 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Родитель
25a857b803
Коммит
f50e88a793
|
@ -2173,6 +2173,10 @@ def err_unexpected_namespace : Error<
|
||||||
def err_undeclared_var_use : Error<"use of undeclared identifier %0">;
|
def err_undeclared_var_use : Error<"use of undeclared identifier %0">;
|
||||||
def note_dependent_var_use : Note<"must qualify identifier to find this "
|
def note_dependent_var_use : Note<"must qualify identifier to find this "
|
||||||
"declaration in dependent base class">;
|
"declaration in dependent base class">;
|
||||||
|
def err_not_found_by_two_phase_lookup : Error<"call to function %0 that is neither "
|
||||||
|
"visible in the template definition nor found by argument dependent lookup">;
|
||||||
|
def note_not_found_by_two_phase_lookup : Note<"%0 should be declared prior to the "
|
||||||
|
"call site%select{| or in %2| or in an associated namespace of one of its arguments}1">;
|
||||||
def err_undeclared_use : Error<"use of undeclared %0">;
|
def err_undeclared_use : Error<"use of undeclared %0">;
|
||||||
def warn_deprecated : Warning<"%0 is deprecated">,
|
def warn_deprecated : Warning<"%0 is deprecated">,
|
||||||
InGroup<DeprecatedDeclarations>;
|
InGroup<DeprecatedDeclarations>;
|
||||||
|
|
|
@ -7826,6 +7826,111 @@ void Sema::AddOverloadedCallCandidates(UnresolvedLookupExpr *ULE,
|
||||||
ULE->isStdAssociatedNamespace());
|
ULE->isStdAssociatedNamespace());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Attempt to recover from an ill-formed use of a non-dependent name in a
|
||||||
|
/// template, where the non-dependent name was declared after the template
|
||||||
|
/// was defined. This is common in code written for a compilers which do not
|
||||||
|
/// correctly implement two-stage name lookup.
|
||||||
|
///
|
||||||
|
/// Returns true if a viable candidate was found and a diagnostic was issued.
|
||||||
|
static bool
|
||||||
|
DiagnoseTwoPhaseLookup(Sema &SemaRef, SourceLocation FnLoc,
|
||||||
|
const CXXScopeSpec &SS, LookupResult &R,
|
||||||
|
TemplateArgumentListInfo *ExplicitTemplateArgs,
|
||||||
|
Expr **Args, unsigned NumArgs) {
|
||||||
|
if (SemaRef.ActiveTemplateInstantiations.empty() || !SS.isEmpty())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
for (DeclContext *DC = SemaRef.CurContext; DC; DC = DC->getParent()) {
|
||||||
|
SemaRef.LookupQualifiedName(R, DC);
|
||||||
|
|
||||||
|
if (!R.empty()) {
|
||||||
|
R.suppressDiagnostics();
|
||||||
|
|
||||||
|
if (isa<CXXRecordDecl>(DC)) {
|
||||||
|
// Don't diagnose names we find in classes; we get much better
|
||||||
|
// diagnostics for these from DiagnoseEmptyLookup.
|
||||||
|
R.clear();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
OverloadCandidateSet Candidates(FnLoc);
|
||||||
|
for (LookupResult::iterator I = R.begin(), E = R.end(); I != E; ++I)
|
||||||
|
AddOverloadedCallCandidate(SemaRef, I.getPair(),
|
||||||
|
ExplicitTemplateArgs, Args, NumArgs,
|
||||||
|
Candidates, false);
|
||||||
|
|
||||||
|
OverloadCandidateSet::iterator Best;
|
||||||
|
if (Candidates.BestViableFunction(SemaRef, FnLoc, Best) != OR_Success)
|
||||||
|
// No viable functions. Don't bother the user with notes for functions
|
||||||
|
// which don't work and shouldn't be found anyway.
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Find the namespaces where ADL would have looked, and suggest
|
||||||
|
// declaring the function there instead.
|
||||||
|
Sema::AssociatedNamespaceSet AssociatedNamespaces;
|
||||||
|
Sema::AssociatedClassSet AssociatedClasses;
|
||||||
|
SemaRef.FindAssociatedClassesAndNamespaces(Args, NumArgs,
|
||||||
|
AssociatedNamespaces,
|
||||||
|
AssociatedClasses);
|
||||||
|
// Never suggest declaring a function within namespace 'std'.
|
||||||
|
if (DeclContext *Std = SemaRef.getStdNamespace()) {
|
||||||
|
// Use two passes: SmallPtrSet::erase invalidates too many iterators
|
||||||
|
// to be used in the loop.
|
||||||
|
llvm::SmallVector<DeclContext*, 4> StdNamespaces;
|
||||||
|
for (Sema::AssociatedNamespaceSet::iterator
|
||||||
|
it = AssociatedNamespaces.begin(),
|
||||||
|
end = AssociatedNamespaces.end(); it != end; ++it)
|
||||||
|
if (Std->Encloses(*it))
|
||||||
|
StdNamespaces.push_back(*it);
|
||||||
|
for (unsigned I = 0; I != StdNamespaces.size(); ++I)
|
||||||
|
AssociatedNamespaces.erase(StdNamespaces[I]);
|
||||||
|
}
|
||||||
|
|
||||||
|
SemaRef.Diag(R.getNameLoc(), diag::err_not_found_by_two_phase_lookup)
|
||||||
|
<< R.getLookupName();
|
||||||
|
if (AssociatedNamespaces.empty()) {
|
||||||
|
SemaRef.Diag(Best->Function->getLocation(),
|
||||||
|
diag::note_not_found_by_two_phase_lookup)
|
||||||
|
<< R.getLookupName() << 0;
|
||||||
|
} else if (AssociatedNamespaces.size() == 1) {
|
||||||
|
SemaRef.Diag(Best->Function->getLocation(),
|
||||||
|
diag::note_not_found_by_two_phase_lookup)
|
||||||
|
<< R.getLookupName() << 1 << *AssociatedNamespaces.begin();
|
||||||
|
} else {
|
||||||
|
// FIXME: It would be useful to list the associated namespaces here,
|
||||||
|
// but the diagnostics infrastructure doesn't provide a way to produce
|
||||||
|
// a localized representation of a list of items.
|
||||||
|
SemaRef.Diag(Best->Function->getLocation(),
|
||||||
|
diag::note_not_found_by_two_phase_lookup)
|
||||||
|
<< R.getLookupName() << 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to recover by calling this function.
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
R.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempt to recover from ill-formed use of a non-dependent operator in a
|
||||||
|
/// template, where the non-dependent operator was declared after the template
|
||||||
|
/// was defined.
|
||||||
|
///
|
||||||
|
/// Returns true if a viable candidate was found and a diagnostic was issued.
|
||||||
|
static bool
|
||||||
|
DiagnoseTwoPhaseOperatorLookup(Sema &SemaRef, OverloadedOperatorKind Op,
|
||||||
|
SourceLocation OpLoc,
|
||||||
|
Expr **Args, unsigned NumArgs) {
|
||||||
|
DeclarationName OpName =
|
||||||
|
SemaRef.Context.DeclarationNames.getCXXOperatorName(Op);
|
||||||
|
LookupResult R(SemaRef, OpName, OpLoc, Sema::LookupOperatorName);
|
||||||
|
return DiagnoseTwoPhaseLookup(SemaRef, OpLoc, CXXScopeSpec(), R,
|
||||||
|
/*ExplicitTemplateArgs=*/0, Args, NumArgs);
|
||||||
|
}
|
||||||
|
|
||||||
/// Attempts to recover from a call where no functions were found.
|
/// Attempts to recover from a call where no functions were found.
|
||||||
///
|
///
|
||||||
/// Returns true if new candidates were found.
|
/// Returns true if new candidates were found.
|
||||||
|
@ -7834,13 +7939,14 @@ BuildRecoveryCallExpr(Sema &SemaRef, Scope *S, Expr *Fn,
|
||||||
UnresolvedLookupExpr *ULE,
|
UnresolvedLookupExpr *ULE,
|
||||||
SourceLocation LParenLoc,
|
SourceLocation LParenLoc,
|
||||||
Expr **Args, unsigned NumArgs,
|
Expr **Args, unsigned NumArgs,
|
||||||
SourceLocation RParenLoc) {
|
SourceLocation RParenLoc,
|
||||||
|
bool EmptyLookup) {
|
||||||
|
|
||||||
CXXScopeSpec SS;
|
CXXScopeSpec SS;
|
||||||
SS.Adopt(ULE->getQualifierLoc());
|
SS.Adopt(ULE->getQualifierLoc());
|
||||||
|
|
||||||
TemplateArgumentListInfo TABuffer;
|
TemplateArgumentListInfo TABuffer;
|
||||||
const TemplateArgumentListInfo *ExplicitTemplateArgs = 0;
|
TemplateArgumentListInfo *ExplicitTemplateArgs = 0;
|
||||||
if (ULE->hasExplicitTemplateArgs()) {
|
if (ULE->hasExplicitTemplateArgs()) {
|
||||||
ULE->copyTemplateArgumentsInto(TABuffer);
|
ULE->copyTemplateArgumentsInto(TABuffer);
|
||||||
ExplicitTemplateArgs = &TABuffer;
|
ExplicitTemplateArgs = &TABuffer;
|
||||||
|
@ -7848,7 +7954,10 @@ BuildRecoveryCallExpr(Sema &SemaRef, Scope *S, Expr *Fn,
|
||||||
|
|
||||||
LookupResult R(SemaRef, ULE->getName(), ULE->getNameLoc(),
|
LookupResult R(SemaRef, ULE->getName(), ULE->getNameLoc(),
|
||||||
Sema::LookupOrdinaryName);
|
Sema::LookupOrdinaryName);
|
||||||
if (SemaRef.DiagnoseEmptyLookup(S, SS, R, Sema::CTC_Expression))
|
if (!DiagnoseTwoPhaseLookup(SemaRef, Fn->getExprLoc(), SS, R,
|
||||||
|
ExplicitTemplateArgs, Args, NumArgs) &&
|
||||||
|
(!EmptyLookup ||
|
||||||
|
SemaRef.DiagnoseEmptyLookup(S, SS, R, Sema::CTC_Expression)))
|
||||||
return ExprError();
|
return ExprError();
|
||||||
|
|
||||||
assert(!R.empty() && "lookup results empty despite recovery");
|
assert(!R.empty() && "lookup results empty despite recovery");
|
||||||
|
@ -7868,7 +7977,7 @@ BuildRecoveryCallExpr(Sema &SemaRef, Scope *S, Expr *Fn,
|
||||||
return ExprError();
|
return ExprError();
|
||||||
|
|
||||||
// This shouldn't cause an infinite loop because we're giving it
|
// This shouldn't cause an infinite loop because we're giving it
|
||||||
// an expression with non-empty lookup results, which should never
|
// an expression with viable lookup results, which should never
|
||||||
// end up here.
|
// end up here.
|
||||||
return SemaRef.ActOnCallExpr(/*Scope*/ 0, NewFn.take(), LParenLoc,
|
return SemaRef.ActOnCallExpr(/*Scope*/ 0, NewFn.take(), LParenLoc,
|
||||||
MultiExprArg(Args, NumArgs), RParenLoc);
|
MultiExprArg(Args, NumArgs), RParenLoc);
|
||||||
|
@ -7914,11 +8023,11 @@ Sema::BuildOverloadedCallExpr(Scope *S, Expr *Fn, UnresolvedLookupExpr *ULE,
|
||||||
AddOverloadedCallCandidates(ULE, Args, NumArgs, CandidateSet);
|
AddOverloadedCallCandidates(ULE, Args, NumArgs, CandidateSet);
|
||||||
|
|
||||||
// If we found nothing, try to recover.
|
// If we found nothing, try to recover.
|
||||||
// AddRecoveryCallCandidates diagnoses the error itself, so we just
|
// BuildRecoveryCallExpr diagnoses the error itself, so we just bail
|
||||||
// bailout out if it fails.
|
// out if it fails.
|
||||||
if (CandidateSet.empty())
|
if (CandidateSet.empty())
|
||||||
return BuildRecoveryCallExpr(*this, S, Fn, ULE, LParenLoc, Args, NumArgs,
|
return BuildRecoveryCallExpr(*this, S, Fn, ULE, LParenLoc, Args, NumArgs,
|
||||||
RParenLoc);
|
RParenLoc, /*EmptyLookup=*/true);
|
||||||
|
|
||||||
OverloadCandidateSet::iterator Best;
|
OverloadCandidateSet::iterator Best;
|
||||||
switch (CandidateSet.BestViableFunction(*this, Fn->getLocStart(), Best)) {
|
switch (CandidateSet.BestViableFunction(*this, Fn->getLocStart(), Best)) {
|
||||||
|
@ -7933,12 +8042,21 @@ Sema::BuildOverloadedCallExpr(Scope *S, Expr *Fn, UnresolvedLookupExpr *ULE,
|
||||||
ExecConfig);
|
ExecConfig);
|
||||||
}
|
}
|
||||||
|
|
||||||
case OR_No_Viable_Function:
|
case OR_No_Viable_Function: {
|
||||||
|
// Try to recover by looking for viable functions which the user might
|
||||||
|
// have meant to call.
|
||||||
|
ExprResult Recovery = BuildRecoveryCallExpr(*this, S, Fn, ULE, LParenLoc,
|
||||||
|
Args, NumArgs, RParenLoc,
|
||||||
|
/*EmptyLookup=*/false);
|
||||||
|
if (!Recovery.isInvalid())
|
||||||
|
return Recovery;
|
||||||
|
|
||||||
Diag(Fn->getSourceRange().getBegin(),
|
Diag(Fn->getSourceRange().getBegin(),
|
||||||
diag::err_ovl_no_viable_function_in_call)
|
diag::err_ovl_no_viable_function_in_call)
|
||||||
<< ULE->getName() << Fn->getSourceRange();
|
<< ULE->getName() << Fn->getSourceRange();
|
||||||
CandidateSet.NoteCandidates(*this, OCD_AllCandidates, Args, NumArgs);
|
CandidateSet.NoteCandidates(*this, OCD_AllCandidates, Args, NumArgs);
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case OR_Ambiguous:
|
case OR_Ambiguous:
|
||||||
Diag(Fn->getSourceRange().getBegin(), diag::err_ovl_ambiguous_call)
|
Diag(Fn->getSourceRange().getBegin(), diag::err_ovl_ambiguous_call)
|
||||||
|
@ -8127,6 +8245,13 @@ Sema::CreateOverloadedUnaryOp(SourceLocation OpLoc, unsigned OpcIn,
|
||||||
}
|
}
|
||||||
|
|
||||||
case OR_No_Viable_Function:
|
case OR_No_Viable_Function:
|
||||||
|
// This is an erroneous use of an operator which can be overloaded by
|
||||||
|
// a non-member function. Check for non-member operators which were
|
||||||
|
// defined too late to be candidates.
|
||||||
|
if (DiagnoseTwoPhaseOperatorLookup(*this, Op, OpLoc, Args, NumArgs))
|
||||||
|
// FIXME: Recover by calling the found function.
|
||||||
|
return ExprError();
|
||||||
|
|
||||||
// No viable function; fall through to handling this as a
|
// No viable function; fall through to handling this as a
|
||||||
// built-in operator, which will produce an error message for us.
|
// built-in operator, which will produce an error message for us.
|
||||||
break;
|
break;
|
||||||
|
@ -8408,6 +8533,13 @@ Sema::CreateOverloadedBinOp(SourceLocation OpLoc,
|
||||||
<< BinaryOperator::getOpcodeStr(Opc)
|
<< BinaryOperator::getOpcodeStr(Opc)
|
||||||
<< Args[0]->getSourceRange() << Args[1]->getSourceRange();
|
<< Args[0]->getSourceRange() << Args[1]->getSourceRange();
|
||||||
} else {
|
} else {
|
||||||
|
// This is an erroneous use of an operator which can be overloaded by
|
||||||
|
// a non-member function. Check for non-member operators which were
|
||||||
|
// defined too late to be candidates.
|
||||||
|
if (DiagnoseTwoPhaseOperatorLookup(*this, Op, OpLoc, Args, 2))
|
||||||
|
// FIXME: Recover by calling the found function.
|
||||||
|
return ExprError();
|
||||||
|
|
||||||
// No viable function; try to create a built-in operation, which will
|
// No viable function; try to create a built-in operation, which will
|
||||||
// produce an error. Then, show the non-viable candidates.
|
// produce an error. Then, show the non-viable candidates.
|
||||||
Result = CreateBuiltinBinOp(OpLoc, Opc, Args[0], Args[1]);
|
Result = CreateBuiltinBinOp(OpLoc, Opc, Args[0], Args[1]);
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
// RUN: %clang_cc1 -fsyntax-only -verify -std=c++0x %s
|
||||||
|
|
||||||
typedef double A;
|
typedef double A;
|
||||||
template<typename T> class B {
|
template<typename T> class B {
|
||||||
|
@ -129,3 +129,136 @@ namespace PR8966 {
|
||||||
template <class T>
|
template <class T>
|
||||||
const char* MyClass<T>::array [MyClass<T>::N] = { "A", "B", "C" };
|
const char* MyClass<T>::array [MyClass<T>::N] = { "A", "B", "C" };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
inline namespace v1 {
|
||||||
|
template<typename T> struct basic_ostream;
|
||||||
|
}
|
||||||
|
namespace inner {
|
||||||
|
template<typename T> struct vector {};
|
||||||
|
}
|
||||||
|
using inner::vector;
|
||||||
|
template<typename T, typename U> struct pair {};
|
||||||
|
typedef basic_ostream<char> ostream;
|
||||||
|
extern ostream cout;
|
||||||
|
std::ostream &operator<<(std::ostream &out, const char *);
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace PR10053 {
|
||||||
|
template<typename T> struct A {
|
||||||
|
T t;
|
||||||
|
A() {
|
||||||
|
f(t); // expected-error {{call to function 'f' that is neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void f(int&); // expected-note {{'f' should be declared prior to the call site}}
|
||||||
|
|
||||||
|
A<int> a; // expected-note {{in instantiation of member function}}
|
||||||
|
|
||||||
|
|
||||||
|
namespace N {
|
||||||
|
namespace M {
|
||||||
|
template<typename T> int g(T t) {
|
||||||
|
f(t); // expected-error {{call to function 'f' that is neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void f(char&); // expected-note {{'f' should be declared prior to the call site}}
|
||||||
|
}
|
||||||
|
|
||||||
|
void f(char&);
|
||||||
|
|
||||||
|
int k = N::M::g<char>(0);; // expected-note {{in instantiation of function}}
|
||||||
|
|
||||||
|
|
||||||
|
namespace O {
|
||||||
|
void f(char&); // expected-note {{candidate function not viable}}
|
||||||
|
|
||||||
|
template<typename T> struct C {
|
||||||
|
static const int n = f(T()); // expected-error {{no matching function}}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
int f(double); // no note, shadowed by O::f
|
||||||
|
O::C<double> c; // expected-note {{requested here}}
|
||||||
|
|
||||||
|
|
||||||
|
// Example from www/compatibility.html
|
||||||
|
namespace my_file {
|
||||||
|
template <typename T> T Squared(T x) {
|
||||||
|
return Multiply(x, x); // expected-error {{neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
|
}
|
||||||
|
|
||||||
|
int Multiply(int x, int y) { // expected-note {{should be declared prior to the call site}}
|
||||||
|
return x * y;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
Squared(5); // expected-note {{here}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Example from www/compatibility.html
|
||||||
|
namespace my_file2 {
|
||||||
|
template<typename T>
|
||||||
|
void Dump(const T& value) {
|
||||||
|
std::cout << value << "\n"; // expected-error {{neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace ns {
|
||||||
|
struct Data {};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostream& operator<<(std::ostream& out, ns::Data data) { // expected-note {{should be declared prior to the call site or in namespace 'PR10053::my_file2::ns'}}
|
||||||
|
return out << "Some data";
|
||||||
|
}
|
||||||
|
|
||||||
|
void Use() {
|
||||||
|
Dump(ns::Data()); // expected-note {{here}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace my_file2_a {
|
||||||
|
template<typename T>
|
||||||
|
void Dump(const T &value) {
|
||||||
|
print(std::cout, value); // expected-error 4{{neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace ns {
|
||||||
|
struct Data {};
|
||||||
|
}
|
||||||
|
namespace ns2 {
|
||||||
|
struct Data {};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostream &print(std::ostream &out, int); // expected-note-re {{should be declared prior to the call site$}}
|
||||||
|
std::ostream &print(std::ostream &out, ns::Data); // expected-note {{should be declared prior to the call site or in namespace 'PR10053::my_file2_a::ns'}}
|
||||||
|
std::ostream &print(std::ostream &out, std::vector<ns2::Data>); // expected-note {{should be declared prior to the call site or in namespace 'PR10053::my_file2_a::ns2'}}
|
||||||
|
std::ostream &print(std::ostream &out, std::pair<ns::Data, ns2::Data>); // expected-note {{should be declared prior to the call site or in an associated namespace of one of its arguments}}
|
||||||
|
|
||||||
|
void Use() {
|
||||||
|
Dump(0); // expected-note {{requested here}}
|
||||||
|
Dump(ns::Data()); // expected-note {{requested here}}
|
||||||
|
Dump(std::vector<ns2::Data>()); // expected-note {{requested here}}
|
||||||
|
Dump(std::pair<ns::Data, ns2::Data>()); // expected-note {{requested here}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace unary {
|
||||||
|
template<typename T>
|
||||||
|
T Negate(const T& value) {
|
||||||
|
return !value; // expected-error {{call to function 'operator!' that is neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace ns {
|
||||||
|
struct Data {};
|
||||||
|
}
|
||||||
|
|
||||||
|
ns::Data operator!(ns::Data); // expected-note {{should be declared prior to the call site or in namespace 'PR10053::unary::ns'}}
|
||||||
|
|
||||||
|
void Use() {
|
||||||
|
Negate(ns::Data()); // expected-note {{requested here}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -24,7 +24,8 @@ namespace N3 {
|
||||||
template<typename T, typename Result>
|
template<typename T, typename Result>
|
||||||
struct call_f0 {
|
struct call_f0 {
|
||||||
void test_f0(T t) {
|
void test_f0(T t) {
|
||||||
Result &result = f0(t); // expected-error 2{{undeclared identifier}}
|
Result &result = f0(t); // expected-error {{undeclared identifier}} \
|
||||||
|
expected-error {{neither visible in the template definition nor found by argument dependent lookup}}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -32,7 +33,7 @@ namespace N3 {
|
||||||
template struct N3::call_f0<int, char&>; // expected-note{{instantiation}}
|
template struct N3::call_f0<int, char&>; // expected-note{{instantiation}}
|
||||||
template struct N3::call_f0<N1::X0, int&>;
|
template struct N3::call_f0<N1::X0, int&>;
|
||||||
|
|
||||||
short& f0(char);
|
short& f0(char); // expected-note {{should be declared prior to the call site}}
|
||||||
namespace N4 {
|
namespace N4 {
|
||||||
template<typename T, typename Result>
|
template<typename T, typename Result>
|
||||||
struct call_f0 {
|
struct call_f0 {
|
||||||
|
|
|
@ -459,13 +459,15 @@ int main() {
|
||||||
|
|
||||||
<p>Clang complains:
|
<p>Clang complains:
|
||||||
|
|
||||||
<pre> <b>my_file.cpp:2:10: <span class="error">error:</span> use of undeclared identifier 'Multiply'</b>
|
<pre> <b>my_file.cpp:2:10: <span class="error">error:</span> call to function 'Multiply' that is neither visible in the template definition nor found by argument dependent lookup</b>
|
||||||
return Multiply(x, x);
|
return Multiply(x, x);
|
||||||
<span class="caret"> ^</span>
|
<span class="caret"> ^</span>
|
||||||
|
|
||||||
<b>my_file.cpp:10:3: <span class="note">note:</span> in instantiation of function template specialization 'Squared<int>' requested here</b>
|
<b>my_file.cpp:10:3: <span class="note">note:</span> in instantiation of function template specialization 'Squared<int>' requested here</b>
|
||||||
Squared(5);
|
Squared(5);
|
||||||
<span class="caret"> ^</span>
|
<span class="caret"> ^</span>
|
||||||
|
<b>my_file.cpp:5:5: <span class="note">note:</span> 'Multiply' should be declared prior to the call site</b>
|
||||||
|
int Multiply(int x, int y) {
|
||||||
|
<span class="caret"> ^</span>
|
||||||
</pre>
|
</pre>
|
||||||
|
|
||||||
<p>The C++ standard says that unqualified names like <q>Multiply</q>
|
<p>The C++ standard says that unqualified names like <q>Multiply</q>
|
||||||
|
@ -516,15 +518,17 @@ void Use() {
|
||||||
Dump(ns::Data());
|
Dump(ns::Data());
|
||||||
}</pre>
|
}</pre>
|
||||||
|
|
||||||
<p>Again, Clang complains about not finding a matching function:</p>
|
<p>Again, Clang complains:</p>
|
||||||
|
|
||||||
<pre>
|
<pre> <b>my_file2.cpp:5:13: <span class="error">error:</span> call to function 'operator<<' that is neither visible in the template definition nor found by argument dependent lookup</b>
|
||||||
<b>my_file.cpp:5:13: <span class="error">error:</span> invalid operands to binary expression ('ostream' (aka 'basic_ostream<char>') and 'ns::Data const')</b>
|
|
||||||
std::cout << value << "\n";
|
std::cout << value << "\n";
|
||||||
<span class="caret">~~~~~~~~~ ^ ~~~~~</span>
|
<span class="caret"> ^</span>
|
||||||
<b>my_file.cpp:17:3: <span class="note">note:</span> in instantiation of function template specialization 'Dump<ns::Data>' requested here</b>
|
<b>my_file2.cpp:17:3: <span class="error">note:</span> in instantiation of function template specialization 'Dump<ns::Data>' requested here</b>
|
||||||
Dump(ns::Data());
|
Dump(ns::Data());
|
||||||
<span class="caret">^</span>
|
<span class="caret"> ^</span>
|
||||||
|
<b>my_file2.cpp:12:15: <span class="error">note:</span> 'operator<<' should be declared prior to the call site or in namespace 'ns'</b>
|
||||||
|
std::ostream& operator<<(std::ostream& out, ns::Data data) {
|
||||||
|
<span class="caret"> ^</span>
|
||||||
</pre>
|
</pre>
|
||||||
|
|
||||||
<p>Just like before, unqualified lookup didn't find any declarations
|
<p>Just like before, unqualified lookup didn't find any declarations
|
||||||
|
|
Загрузка…
Ссылка в новой задаче