forked from OSchip/llvm-project
When recursively instantiating function templates, keep track of the
instantiation stack so that we provide a full instantiation backtrace. Previously, we performed all of the instantiations implied by the recursion, but each looked like a "top-level" instantiation. The included test case tests the previous fix for the instantiation of DeclRefExprs. Note that the "instantiated from" diagnostics still don't tell us which template arguments we're instantiating with. llvm-svn: 74540
This commit is contained in:
parent
b2b37c62d0
commit
dda7ced32e
|
@ -29,9 +29,9 @@
|
|||
#include "llvm/ADT/DenseSet.h"
|
||||
#include "llvm/ADT/SmallPtrSet.h"
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include <deque>
|
||||
#include <list>
|
||||
#include <string>
|
||||
#include <queue>
|
||||
#include <vector>
|
||||
|
||||
namespace llvm {
|
||||
|
@ -2581,7 +2581,7 @@ public:
|
|||
|
||||
/// \brief The queue of implicit template instantiations that are required
|
||||
/// but have not yet been performed.
|
||||
std::queue<PendingImplicitInstantiation> PendingImplicitInstantiations;
|
||||
std::deque<PendingImplicitInstantiation> PendingImplicitInstantiations;
|
||||
|
||||
void PerformPendingImplicitInstantiations();
|
||||
|
||||
|
@ -2636,7 +2636,8 @@ public:
|
|||
const TemplateArgumentList &TemplateArgs);
|
||||
|
||||
void InstantiateFunctionDefinition(SourceLocation PointOfInstantiation,
|
||||
FunctionDecl *Function);
|
||||
FunctionDecl *Function,
|
||||
bool Recursive = false);
|
||||
void InstantiateVariableDefinition(VarDecl *Var);
|
||||
|
||||
NamedDecl *InstantiateCurrentDeclRef(NamedDecl *D);
|
||||
|
|
|
@ -5641,7 +5641,7 @@ void Sema::MarkDeclarationReferenced(SourceLocation Loc, Decl *D) {
|
|||
// instantiated, naturally).
|
||||
if (Function->getInstantiatedFromMemberFunction() ||
|
||||
Function->getPrimaryTemplate())
|
||||
PendingImplicitInstantiations.push(std::make_pair(Function, Loc));
|
||||
PendingImplicitInstantiations.push_back(std::make_pair(Function, Loc));
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -663,10 +663,19 @@ TemplateDeclInstantiator::InitMethodInstantiation(CXXMethodDecl *New,
|
|||
/// \brief Instantiate the definition of the given function from its
|
||||
/// template.
|
||||
///
|
||||
/// \param PointOfInstantiation the point at which the instantiation was
|
||||
/// required. Note that this is not precisely a "point of instantiation"
|
||||
/// for the function, but it's close.
|
||||
///
|
||||
/// \param Function the already-instantiated declaration of a
|
||||
/// function.
|
||||
/// function template specialization or member function of a class template
|
||||
/// specialization.
|
||||
///
|
||||
/// \param Recursive if true, recursively instantiates any functions that
|
||||
/// are required by this instantiation.
|
||||
void Sema::InstantiateFunctionDefinition(SourceLocation PointOfInstantiation,
|
||||
FunctionDecl *Function) {
|
||||
FunctionDecl *Function,
|
||||
bool Recursive) {
|
||||
if (Function->isInvalidDecl())
|
||||
return;
|
||||
|
||||
|
@ -689,6 +698,13 @@ void Sema::InstantiateFunctionDefinition(SourceLocation PointOfInstantiation,
|
|||
if (Inst)
|
||||
return;
|
||||
|
||||
// If we're performing recursive template instantiation, create our own
|
||||
// queue of pending implicit instantiations that we will instantiate later,
|
||||
// while we're still within our own instantiation context.
|
||||
std::deque<PendingImplicitInstantiation> SavedPendingImplicitInstantiations;
|
||||
if (Recursive)
|
||||
PendingImplicitInstantiations.swap(SavedPendingImplicitInstantiations);
|
||||
|
||||
ActOnStartOfFunctionDef(0, DeclPtrTy::make(Function));
|
||||
|
||||
// Introduce a new scope where local variable instantiations will be
|
||||
|
@ -717,6 +733,15 @@ void Sema::InstantiateFunctionDefinition(SourceLocation PointOfInstantiation,
|
|||
|
||||
DeclGroupRef DG(Function);
|
||||
Consumer.HandleTopLevelDecl(DG);
|
||||
|
||||
if (Recursive) {
|
||||
// Instantiate any pending implicit instantiations found during the
|
||||
// instantiation of this template.
|
||||
PerformPendingImplicitInstantiations();
|
||||
|
||||
// Restore the set of pending implicit instantiations.
|
||||
PendingImplicitInstantiations.swap(SavedPendingImplicitInstantiations);
|
||||
}
|
||||
}
|
||||
|
||||
/// \brief Instantiate the definition of the given variable from its
|
||||
|
@ -859,11 +884,11 @@ NamedDecl * Sema::InstantiateCurrentDeclRef(NamedDecl *D) {
|
|||
void Sema::PerformPendingImplicitInstantiations() {
|
||||
while (!PendingImplicitInstantiations.empty()) {
|
||||
PendingImplicitInstantiation Inst = PendingImplicitInstantiations.front();
|
||||
PendingImplicitInstantiations.pop();
|
||||
PendingImplicitInstantiations.pop_front();
|
||||
|
||||
if (FunctionDecl *Function = dyn_cast<FunctionDecl>(Inst.first))
|
||||
if (!Function->getBody())
|
||||
InstantiateFunctionDefinition(/*FIXME:*/Inst.second, Function);
|
||||
InstantiateFunctionDefinition(/*FIXME:*/Inst.second, Function, true);
|
||||
|
||||
// FIXME: instantiate static member variables
|
||||
}
|
||||
|
|
|
@ -32,5 +32,5 @@ void test(X<Poly> xp, X<Poly, Poly&> xpr, X<NonPoly> xnp, X<NonPoly, NonPoly&> x
|
|||
xnpr.g(NonPoly());
|
||||
|
||||
// Triggers an error (as it should);
|
||||
xpr.g(Poly());
|
||||
xpr.g(Poly()); // expected-note{{instantiation of member function}}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,10 @@
|
|||
// RUN: clang-cc -fsyntax-only -verify %s
|
||||
|
||||
template<typename T> void f(T* t) {
|
||||
f(*t); // expected-error{{no matching function}}\
|
||||
// expected-note 3{{requested here}}
|
||||
}
|
||||
|
||||
void test_f(int ****p) {
|
||||
f(p); // expected-note{{requested here}}
|
||||
}
|
Loading…
Reference in New Issue