forked from OSchip/llvm-project
Tie DefineVTablesUsed() in with recursive function instantiation so that we emit
a useful template instantiation stack. Fixes PR8640. This also causes a slight change to where the "instantianted from" note shows up in truly esoteric cases (see the change to test/SemaCXX/destructor.cpp), but that isn't directly the fault of this patch. llvm-svn: 120135
This commit is contained in:
parent
19bda8af16
commit
ef4f456866
|
@ -2557,8 +2557,11 @@ public:
|
||||||
/// \brief The list of classes whose vtables have been used within
|
/// \brief The list of classes whose vtables have been used within
|
||||||
/// this translation unit, and the source locations at which the
|
/// this translation unit, and the source locations at which the
|
||||||
/// first use occurred.
|
/// first use occurred.
|
||||||
llvm::SmallVector<std::pair<CXXRecordDecl *, SourceLocation>, 16>
|
typedef std::pair<CXXRecordDecl*, SourceLocation> VTableUse;
|
||||||
VTableUses;
|
|
||||||
|
/// \brief The list of vtables that are required but have not yet been
|
||||||
|
/// materialized.
|
||||||
|
llvm::SmallVector<VTableUse, 16> VTableUses;
|
||||||
|
|
||||||
/// \brief The set of classes whose vtables have been used within
|
/// \brief The set of classes whose vtables have been used within
|
||||||
/// this translation unit, and a bit that will be true if the vtable is
|
/// this translation unit, and a bit that will be true if the vtable is
|
||||||
|
|
|
@ -284,26 +284,24 @@ static bool ShouldRemoveFromUnused(Sema *SemaRef, const DeclaratorDecl *D) {
|
||||||
void Sema::ActOnEndOfTranslationUnit() {
|
void Sema::ActOnEndOfTranslationUnit() {
|
||||||
// At PCH writing, implicit instantiations and VTable handling info are
|
// At PCH writing, implicit instantiations and VTable handling info are
|
||||||
// stored and performed when the PCH is included.
|
// stored and performed when the PCH is included.
|
||||||
if (CompleteTranslationUnit)
|
if (CompleteTranslationUnit) {
|
||||||
while (1) {
|
// If DefinedUsedVTables ends up marking any virtual member functions it
|
||||||
// C++: Perform implicit template instantiations.
|
// might lead to more pending template instantiations, which we then need
|
||||||
//
|
// to instantiate.
|
||||||
// FIXME: When we perform these implicit instantiations, we do not
|
DefineUsedVTables();
|
||||||
// carefully keep track of the point of instantiation (C++ [temp.point]).
|
|
||||||
// This means that name lookup that occurs within the template
|
|
||||||
// instantiation will always happen at the end of the translation unit,
|
|
||||||
// so it will find some names that should not be found. Although this is
|
|
||||||
// common behavior for C++ compilers, it is technically wrong. In the
|
|
||||||
// future, we either need to be able to filter the results of name lookup
|
|
||||||
// or we need to perform template instantiations earlier.
|
|
||||||
PerformPendingInstantiations();
|
|
||||||
|
|
||||||
/// If DefinedUsedVTables ends up marking any virtual member
|
// C++: Perform implicit template instantiations.
|
||||||
/// functions it might lead to more pending template
|
//
|
||||||
/// instantiations, which is why we need to loop here.
|
// FIXME: When we perform these implicit instantiations, we do not
|
||||||
if (!DefineUsedVTables())
|
// carefully keep track of the point of instantiation (C++ [temp.point]).
|
||||||
break;
|
// This means that name lookup that occurs within the template
|
||||||
}
|
// instantiation will always happen at the end of the translation unit,
|
||||||
|
// so it will find some names that should not be found. Although this is
|
||||||
|
// common behavior for C++ compilers, it is technically wrong. In the
|
||||||
|
// future, we either need to be able to filter the results of name lookup
|
||||||
|
// or we need to perform template instantiations earlier.
|
||||||
|
PerformPendingInstantiations();
|
||||||
|
}
|
||||||
|
|
||||||
// Remove file scoped decls that turned out to be used.
|
// Remove file scoped decls that turned out to be used.
|
||||||
UnusedFileScopedDecls.erase(std::remove_if(UnusedFileScopedDecls.begin(),
|
UnusedFileScopedDecls.erase(std::remove_if(UnusedFileScopedDecls.begin(),
|
||||||
|
|
|
@ -2108,9 +2108,12 @@ void Sema::InstantiateFunctionDefinition(SourceLocation PointOfInstantiation,
|
||||||
// If we're performing recursive template instantiation, create our own
|
// If we're performing recursive template instantiation, create our own
|
||||||
// queue of pending implicit instantiations that we will instantiate later,
|
// queue of pending implicit instantiations that we will instantiate later,
|
||||||
// while we're still within our own instantiation context.
|
// while we're still within our own instantiation context.
|
||||||
|
llvm::SmallVector<VTableUse, 16> SavedVTableUses;
|
||||||
std::deque<PendingImplicitInstantiation> SavedPendingInstantiations;
|
std::deque<PendingImplicitInstantiation> SavedPendingInstantiations;
|
||||||
if (Recursive)
|
if (Recursive) {
|
||||||
|
VTableUses.swap(SavedVTableUses);
|
||||||
PendingInstantiations.swap(SavedPendingInstantiations);
|
PendingInstantiations.swap(SavedPendingInstantiations);
|
||||||
|
}
|
||||||
|
|
||||||
EnterExpressionEvaluationContext EvalContext(*this,
|
EnterExpressionEvaluationContext EvalContext(*this,
|
||||||
Sema::PotentiallyEvaluated);
|
Sema::PotentiallyEvaluated);
|
||||||
|
@ -2173,10 +2176,16 @@ void Sema::InstantiateFunctionDefinition(SourceLocation PointOfInstantiation,
|
||||||
Scope.Exit();
|
Scope.Exit();
|
||||||
|
|
||||||
if (Recursive) {
|
if (Recursive) {
|
||||||
|
// Define any pending vtables.
|
||||||
|
DefineUsedVTables();
|
||||||
|
|
||||||
// Instantiate any pending implicit instantiations found during the
|
// Instantiate any pending implicit instantiations found during the
|
||||||
// instantiation of this template.
|
// instantiation of this template.
|
||||||
PerformPendingInstantiations();
|
PerformPendingInstantiations();
|
||||||
|
|
||||||
|
// Restore the set of pending vtables.
|
||||||
|
VTableUses.swap(SavedVTableUses);
|
||||||
|
|
||||||
// Restore the set of pending implicit instantiations.
|
// Restore the set of pending implicit instantiations.
|
||||||
PendingInstantiations.swap(SavedPendingInstantiations);
|
PendingInstantiations.swap(SavedPendingInstantiations);
|
||||||
}
|
}
|
||||||
|
|
|
@ -100,11 +100,11 @@ namespace test6 {
|
||||||
T::deleteIt(p); // expected-error {{type 'int' cannot be used prior to '::'}}
|
T::deleteIt(p); // expected-error {{type 'int' cannot be used prior to '::'}}
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual ~A() {} // expected-note {{in instantiation of member function 'test6::A<int>::operator delete' requested here}}
|
virtual ~A() {}
|
||||||
};
|
};
|
||||||
|
|
||||||
class B : A<int> { B(); };
|
class B : A<int> { B(); }; // expected-note {{in instantiation of member function 'test6::A<int>::operator delete' requested here}}
|
||||||
B::B() {} // expected-note {{in instantiation of member function 'test6::A<int>::~A' requested here}}
|
B::B() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure classes are marked invalid when they have invalid
|
// Make sure classes are marked invalid when they have invalid
|
||||||
|
|
|
@ -0,0 +1,20 @@
|
||||||
|
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
||||||
|
// PR8640
|
||||||
|
|
||||||
|
template<class T1> struct C1 {
|
||||||
|
virtual void c1() {
|
||||||
|
T1 t1 = 3; // expected-error {{cannot initialize a variable}}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<class T2> struct C2 {
|
||||||
|
void c2() {
|
||||||
|
new C1<T2>(); // expected-note {{in instantiation of member function}}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void f() {
|
||||||
|
C2<int*> c2;
|
||||||
|
c2.c2(); // expected-note {{in instantiation of member function}}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue