mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-25 10:48:14 +00:00
RecursiveASTVisitor: When in 'shouldVisitTemplateInstantiations' mode, visit
all instantiations of a template when we visit the canonical declaration of the primary template, rather than trying to match them up to the partial specialization from which they are instantiated. This fixes a bug where we failed to visit instantiations of partial specializations of member templates of class templates, and naturally extends to allow us to visit instantiations where we have instantiated only a declaration. llvm-svn: 155597
This commit is contained in:
parent
e38c006049
commit
f333acd686
@ -392,8 +392,8 @@ public:
|
||||
private:
|
||||
// These are helper methods used by more than one Traverse* method.
|
||||
bool TraverseTemplateParameterListHelper(TemplateParameterList *TPL);
|
||||
bool TraverseClassInstantiations(ClassTemplateDecl* D, Decl *Pattern);
|
||||
bool TraverseFunctionInstantiations(FunctionTemplateDecl* D) ;
|
||||
bool TraverseClassInstantiations(ClassTemplateDecl *D);
|
||||
bool TraverseFunctionInstantiations(FunctionTemplateDecl *D) ;
|
||||
bool TraverseTemplateArgumentLocsHelper(const TemplateArgumentLoc *TAL,
|
||||
unsigned Count);
|
||||
bool TraverseArrayTypeLocHelper(ArrayTypeLoc TL);
|
||||
@ -1377,35 +1377,19 @@ bool RecursiveASTVisitor<Derived>::TraverseTemplateParameterListHelper(
|
||||
}
|
||||
|
||||
// A helper method for traversing the implicit instantiations of a
|
||||
// class.
|
||||
// class template.
|
||||
template<typename Derived>
|
||||
bool RecursiveASTVisitor<Derived>::TraverseClassInstantiations(
|
||||
ClassTemplateDecl* D, Decl *Pattern) {
|
||||
assert(isa<ClassTemplateDecl>(Pattern) ||
|
||||
isa<ClassTemplatePartialSpecializationDecl>(Pattern));
|
||||
|
||||
ClassTemplateDecl *D) {
|
||||
ClassTemplateDecl::spec_iterator end = D->spec_end();
|
||||
for (ClassTemplateDecl::spec_iterator it = D->spec_begin(); it != end; ++it) {
|
||||
ClassTemplateSpecializationDecl* SD = *it;
|
||||
|
||||
switch (SD->getSpecializationKind()) {
|
||||
// Visit the implicit instantiations with the requested pattern.
|
||||
case TSK_ImplicitInstantiation: {
|
||||
llvm::PointerUnion<ClassTemplateDecl *,
|
||||
ClassTemplatePartialSpecializationDecl *> U
|
||||
= SD->getInstantiatedFrom();
|
||||
|
||||
bool ShouldVisit;
|
||||
if (U.is<ClassTemplateDecl*>())
|
||||
ShouldVisit = (U.get<ClassTemplateDecl*>() == Pattern);
|
||||
else
|
||||
ShouldVisit
|
||||
= (U.get<ClassTemplatePartialSpecializationDecl*>() == Pattern);
|
||||
|
||||
if (ShouldVisit)
|
||||
TRY_TO(TraverseDecl(SD));
|
||||
break;
|
||||
}
|
||||
case TSK_Undeclared:
|
||||
case TSK_ImplicitInstantiation:
|
||||
TRY_TO(TraverseDecl(SD));
|
||||
|
||||
// We don't need to do anything on an explicit instantiation
|
||||
// or explicit specialization because there will be an explicit
|
||||
@ -1414,11 +1398,6 @@ bool RecursiveASTVisitor<Derived>::TraverseClassInstantiations(
|
||||
case TSK_ExplicitInstantiationDefinition:
|
||||
case TSK_ExplicitSpecialization:
|
||||
break;
|
||||
|
||||
// We don't need to do anything for an uninstantiated
|
||||
// specialization.
|
||||
case TSK_Undeclared:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
@ -1433,13 +1412,12 @@ DEF_TRAVERSE_DECL(ClassTemplateDecl, {
|
||||
// By default, we do not traverse the instantiations of
|
||||
// class templates since they do not appear in the user code. The
|
||||
// following code optionally traverses them.
|
||||
if (getDerived().shouldVisitTemplateInstantiations()) {
|
||||
// If this is the definition of the primary template, visit
|
||||
// instantiations which were formed from this pattern.
|
||||
if (D->isThisDeclarationADefinition() ||
|
||||
D->getInstantiatedFromMemberTemplate())
|
||||
TRY_TO(TraverseClassInstantiations(D, D));
|
||||
}
|
||||
//
|
||||
// We only traverse the class instantiations when we see the canonical
|
||||
// declaration of the template, to ensure we only visit them once.
|
||||
if (getDerived().shouldVisitTemplateInstantiations() &&
|
||||
D == D->getCanonicalDecl())
|
||||
TRY_TO(TraverseClassInstantiations(D));
|
||||
|
||||
// Note that getInstantiatedFromMemberTemplate() is just a link
|
||||
// from a template instantiation back to the template from which
|
||||
@ -1450,12 +1428,13 @@ DEF_TRAVERSE_DECL(ClassTemplateDecl, {
|
||||
// function while skipping its specializations.
|
||||
template<typename Derived>
|
||||
bool RecursiveASTVisitor<Derived>::TraverseFunctionInstantiations(
|
||||
FunctionTemplateDecl* D) {
|
||||
FunctionTemplateDecl *D) {
|
||||
FunctionTemplateDecl::spec_iterator end = D->spec_end();
|
||||
for (FunctionTemplateDecl::spec_iterator it = D->spec_begin(); it != end;
|
||||
++it) {
|
||||
FunctionDecl* FD = *it;
|
||||
switch (FD->getTemplateSpecializationKind()) {
|
||||
case TSK_Undeclared:
|
||||
case TSK_ImplicitInstantiation:
|
||||
// We don't know what kind of FunctionDecl this is.
|
||||
TRY_TO(TraverseDecl(FD));
|
||||
@ -1467,7 +1446,6 @@ bool RecursiveASTVisitor<Derived>::TraverseFunctionInstantiations(
|
||||
case TSK_ExplicitInstantiationDefinition:
|
||||
break;
|
||||
|
||||
case TSK_Undeclared: // Declaration of the template definition.
|
||||
case TSK_ExplicitSpecialization:
|
||||
break;
|
||||
}
|
||||
@ -1481,20 +1459,14 @@ DEF_TRAVERSE_DECL(FunctionTemplateDecl, {
|
||||
TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters()));
|
||||
|
||||
// By default, we do not traverse the instantiations of
|
||||
// function templates since they do not apprear in the user code. The
|
||||
// function templates since they do not appear in the user code. The
|
||||
// following code optionally traverses them.
|
||||
if (getDerived().shouldVisitTemplateInstantiations()) {
|
||||
// Explicit function specializations will be traversed from the
|
||||
// context of their declaration. There is therefore no need to
|
||||
// traverse them for here.
|
||||
//
|
||||
// In addition, we only traverse the function instantiations when
|
||||
// the function template is a function template definition.
|
||||
if (D->isThisDeclarationADefinition() ||
|
||||
D->getInstantiatedFromMemberTemplate()) {
|
||||
TRY_TO(TraverseFunctionInstantiations(D));
|
||||
}
|
||||
}
|
||||
//
|
||||
// We only traverse the function instantiations when we see the canonical
|
||||
// declaration of the template, to ensure we only visit them once.
|
||||
if (getDerived().shouldVisitTemplateInstantiations() &&
|
||||
D == D->getCanonicalDecl())
|
||||
TRY_TO(TraverseFunctionInstantiations(D));
|
||||
})
|
||||
|
||||
DEF_TRAVERSE_DECL(TemplateTemplateParmDecl, {
|
||||
@ -1636,11 +1608,7 @@ DEF_TRAVERSE_DECL(ClassTemplatePartialSpecializationDecl, {
|
||||
// template args here.
|
||||
TRY_TO(TraverseCXXRecordHelper(D));
|
||||
|
||||
// If we're visiting instantiations, visit the instantiations of
|
||||
// this template now.
|
||||
if (getDerived().shouldVisitTemplateInstantiations() &&
|
||||
D->isThisDeclarationADefinition())
|
||||
TRY_TO(TraverseClassInstantiations(D->getSpecializedTemplate(), D));
|
||||
// Instantiations will have been visited with the primary template.
|
||||
})
|
||||
|
||||
DEF_TRAVERSE_DECL(EnumConstantDecl, {
|
||||
|
@ -152,6 +152,19 @@ public:
|
||||
}
|
||||
};
|
||||
|
||||
class NamedDeclVisitor
|
||||
: public ExpectedLocationVisitor<NamedDeclVisitor> {
|
||||
public:
|
||||
bool VisitNamedDecl(NamedDecl *Decl) {
|
||||
std::string NameWithTemplateArgs;
|
||||
Decl->getNameForDiagnostic(NameWithTemplateArgs,
|
||||
Decl->getASTContext().getPrintingPolicy(),
|
||||
true);
|
||||
Match(NameWithTemplateArgs, Decl->getLocation());
|
||||
return true;
|
||||
}
|
||||
};
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsBaseClassDeclarations) {
|
||||
TypeLocVisitor Visitor;
|
||||
Visitor.ExpectMatch("class X", 1, 30);
|
||||
@ -251,4 +264,85 @@ TEST(RecursiveASTVisitor, VisitsBaseClassTemplateArgumentsInInstantiation) {
|
||||
}
|
||||
*/
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsCallInPartialTemplateSpecialization) {
|
||||
CXXMemberCallVisitor Visitor;
|
||||
Visitor.ExpectMatch("A::x", 6, 20);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template <typename T1> struct X {\n"
|
||||
" template <typename T2, bool B> struct Y { void g(); };\n"
|
||||
"};\n"
|
||||
"template <typename T1> template <typename T2>\n"
|
||||
"struct X<T1>::Y<T2, true> {\n"
|
||||
" void f() { T2 y; y.x(); }\n"
|
||||
"};\n"
|
||||
"struct A { void x(); };\n"
|
||||
"int main() {\n"
|
||||
" (new X<A>::Y<A, true>())->f();\n"
|
||||
"}\n"));
|
||||
}
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsPartialTemplateSpecialization) {
|
||||
// From cfe-commits/Week-of-Mon-20100830/033998.html
|
||||
// Contrary to the approach sugggested in that email, we visit all
|
||||
// specializations when we visit the primary template. Visiting them when we
|
||||
// visit the associated specialization is problematic for specializations of
|
||||
// template members of class templates.
|
||||
NamedDeclVisitor Visitor;
|
||||
Visitor.ExpectMatch("A<bool>", 1, 26);
|
||||
Visitor.ExpectMatch("A<char *>", 2, 26);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template <class T> class A {};\n"
|
||||
"template <class T> class A<T*> {};\n"
|
||||
"A<bool> ab;\n"
|
||||
"A<char*> acp;\n"));
|
||||
}
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsUndefinedClassTemplateSpecialization) {
|
||||
NamedDeclVisitor Visitor;
|
||||
Visitor.ExpectMatch("A<int>", 1, 29);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template<typename T> struct A;\n"
|
||||
"A<int> *p;\n"));
|
||||
}
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsNestedUndefinedClassTemplateSpecialization) {
|
||||
NamedDeclVisitor Visitor;
|
||||
Visitor.ExpectMatch("A<int>::B<char>", 2, 31);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template<typename T> struct A {\n"
|
||||
" template<typename U> struct B;\n"
|
||||
"};\n"
|
||||
"A<int>::B<char> *p;\n"));
|
||||
}
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsUndefinedFunctionTemplateSpecialization) {
|
||||
NamedDeclVisitor Visitor;
|
||||
Visitor.ExpectMatch("A<int>", 1, 26);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template<typename T> int A();\n"
|
||||
"int k = A<int>();\n"));
|
||||
}
|
||||
|
||||
TEST(RecursiveASTVisitor, VisitsNestedUndefinedFunctionTemplateSpecialization) {
|
||||
NamedDeclVisitor Visitor;
|
||||
Visitor.ExpectMatch("A<int>::B<char>", 2, 35);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template<typename T> struct A {\n"
|
||||
" template<typename U> static int B();\n"
|
||||
"};\n"
|
||||
"int k = A<int>::B<char>();\n"));
|
||||
}
|
||||
|
||||
TEST(RecursiveASTVisitor, NoRecursionInSelfFriend) {
|
||||
// From cfe-commits/Week-of-Mon-20100830/033977.html
|
||||
NamedDeclVisitor Visitor;
|
||||
Visitor.ExpectMatch("vector_iterator<int>", 2, 7);
|
||||
EXPECT_TRUE(Visitor.runOver(
|
||||
"template<typename Container>\n"
|
||||
"class vector_iterator {\n"
|
||||
" template <typename C> friend class vector_iterator;\n"
|
||||
"};\n"
|
||||
"vector_iterator<int> it_int;\n"));
|
||||
}
|
||||
|
||||
} // end namespace clang
|
||||
|
Loading…
x
Reference in New Issue
Block a user