Template instantiation for destructors. This is somewhat repetitive;

eliminating the duplication is next on the list.

llvm-svn: 67579
This commit is contained in:
Douglas Gregor 2009-03-24 00:15:49 +00:00
parent 32dfb35281
commit 654b07e029
2 changed files with 52 additions and 2 deletions

View File

@ -44,9 +44,9 @@ namespace {
Decl *VisitStaticAssertDecl(StaticAssertDecl *D);
Decl *VisitEnumDecl(EnumDecl *D);
Decl *VisitCXXMethodDecl(CXXMethodDecl *D);
Decl *VisitCXXDestructorDecl(CXXDestructorDecl *D);
Decl *VisitParmVarDecl(ParmVarDecl *D);
Decl *VisitOriginalParmVarDecl(OriginalParmVarDecl *D);
// Base case. FIXME: Remove once we can instantiate everything.
Decl *VisitDecl(Decl *) {
return 0;
@ -265,6 +265,48 @@ Decl *TemplateDeclInstantiator::VisitCXXMethodDecl(CXXMethodDecl *D) {
return Method;
}
Decl *TemplateDeclInstantiator::VisitCXXDestructorDecl(CXXDestructorDecl *D) {
QualType T = SemaRef.InstantiateType(D->getType(), TemplateArgs,
NumTemplateArgs, D->getLocation(),
D->getDeclName());
if (T.isNull())
return 0;
// Build the instantiated destructor declaration.
CXXRecordDecl *Record = cast<CXXRecordDecl>(Owner);
QualType ClassTy = SemaRef.Context.getTypeDeclType(Record);
CXXDestructorDecl *Destructor
= CXXDestructorDecl::Create(SemaRef.Context, Record,
D->getLocation(),
SemaRef.Context.DeclarationNames.getCXXDestructorName(ClassTy),
T, D->isInline(), false);
Destructor->setAccess(D->getAccess());
// FIXME: Duplicates some logic in ActOnFunctionDeclarator,
// VisitCXXDestructorDecl.
if (D->isVirtual()) {
Destructor->setVirtual();
Record->setAggregate(false);
Record->setPOD(false);
Record->setPolymorphic(true);
}
if (D->isDeleted())
Destructor->setDeleted();
if (D->isPure()) {
Destructor->setPure();
Record->setAbstract(true);
}
bool Redeclaration = false;
bool OverloadableAttrRequired = false;
NamedDecl *PrevDecl = 0;
if (SemaRef.CheckFunctionDeclaration(Destructor, PrevDecl, Redeclaration,
/*FIXME:*/OverloadableAttrRequired))
Destructor->setInvalidDecl();
Owner->addDecl(Destructor);
return Destructor;
}
Decl *TemplateDeclInstantiator::VisitParmVarDecl(ParmVarDecl *D) {
QualType OrigT = SemaRef.InstantiateType(D->getOriginalType(), TemplateArgs,
NumTemplateArgs, D->getLocation(),

View File

@ -1,5 +1,4 @@
// RUN: clang -fsyntax-only -verify %s
template<typename T>
class X {
public:
@ -41,3 +40,12 @@ void test_ovl(Overloading<int, long> *oil, int i, long l) {
void test_ovl_bad() {
Overloading<float, float> off; // expected-note{{in instantiation of template class 'class Overloading<float, float>' requested here}}
}
template<typename T>
class HasDestructor {
virtual ~HasDestructor() = 0;
};
int i = sizeof(HasDestructor<int>); // FIXME: forces instantiation, but
// the code below should probably instantiate by itself.
int abstract_destructor[__is_abstract(HasDestructor<int>)? 1 : -1];