forked from OSchip/llvm-project
The injected-class-name of class templates and class template
specializations can be treated as a template. Finally, we can parse and process the first implementation of Fibonacci I wrote! Note that this code does not handle all of the cases where injected-class-names can be treated as templates. In particular, there's an ambiguity case that we should be able to handle (but can't), e.g., template <class T> struct Base { }; template <class T> struct Derived : Base<int>, Base<char> { typename Derived::Base b; // error: ambiguous typename Derived::Base<double> d; // OK }; llvm-svn: 67720
This commit is contained in:
parent
4555618854
commit
97f1f1c46e
|
@ -515,7 +515,8 @@ public:
|
|||
/// the first declaration of that tag.
|
||||
TagDecl *getCanonicalDecl(TagDecl *Tag) {
|
||||
QualType T = getTagDeclType(Tag);
|
||||
return cast<TagDecl>(cast<TagType>(T)->getDecl());
|
||||
return cast<TagDecl>(cast<TagType>(T.getTypePtr()->CanonicalType)
|
||||
->getDecl());
|
||||
}
|
||||
|
||||
/// Type Query functions. If the type is an instance of the specified class,
|
||||
|
|
|
@ -3425,6 +3425,8 @@ void Sema::ActOnTagStartDefinition(Scope *S, DeclTy *TagD) {
|
|||
Record->getIdentifier(), Record);
|
||||
InjectedClassName->setImplicit();
|
||||
InjectedClassName->setAccess(AS_public);
|
||||
if (ClassTemplateDecl *Template = Record->getDescribedClassTemplate())
|
||||
InjectedClassName->setDescribedClassTemplate(Template);
|
||||
PushOnScopeChains(InjectedClassName, S);
|
||||
assert(InjectedClassName->isInjectedClassName() &&
|
||||
"Broken injected-class-name");
|
||||
|
|
|
@ -42,6 +42,28 @@ TemplateNameKind Sema::isTemplateName(IdentifierInfo &II, Scope *S,
|
|||
return TNK_Template_template_parm;
|
||||
else
|
||||
assert(false && "Unknown TemplateDecl");
|
||||
} else if (CXXRecordDecl *Record = dyn_cast<CXXRecordDecl>(IIDecl)) {
|
||||
// C++ [temp.local]p1:
|
||||
// Like normal (non-template) classes, class templates have an
|
||||
// injected-class-name (Clause 9). The injected-class-name
|
||||
// can be used with or without a template-argument-list. When
|
||||
// it is used without a template-argument-list, it is
|
||||
// equivalent to the injected-class-name followed by the
|
||||
// template-parameters of the class template enclosed in
|
||||
// <>. When it is used with a template-argument-list, it
|
||||
// refers to the specified class template specialization,
|
||||
// which could be the current specialization or another
|
||||
// specialization.
|
||||
if (Record->isInjectedClassName()) {
|
||||
Record = cast<CXXRecordDecl>(Context.getCanonicalDecl(Record));
|
||||
if ((Template = Record->getDescribedClassTemplate()))
|
||||
return TNK_Class_template;
|
||||
else if (ClassTemplateSpecializationDecl *Spec
|
||||
= dyn_cast<ClassTemplateSpecializationDecl>(Record)) {
|
||||
Template = Spec->getSpecializedTemplate();
|
||||
return TNK_Class_template;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: What follows is a gross hack.
|
||||
|
@ -469,7 +491,7 @@ Sema::ActOnClassTemplate(Scope *S, unsigned TagSpec, TagKind TK,
|
|||
// If we had a scope specifier, we better have a previous template
|
||||
// declaration!
|
||||
|
||||
TagDecl *NewClass =
|
||||
CXXRecordDecl *NewClass =
|
||||
CXXRecordDecl::Create(Context, Kind, SemanticContext, NameLoc, Name,
|
||||
PrevClassTemplate?
|
||||
PrevClassTemplate->getTemplatedDecl() : 0);
|
||||
|
@ -478,7 +500,8 @@ Sema::ActOnClassTemplate(Scope *S, unsigned TagSpec, TagKind TK,
|
|||
= ClassTemplateDecl::Create(Context, SemanticContext, NameLoc,
|
||||
DeclarationName(Name), TemplateParams,
|
||||
NewClass, PrevClassTemplate);
|
||||
|
||||
NewClass->setDescribedClassTemplate(NewTemplate);
|
||||
|
||||
// Set the lexical context of these templates
|
||||
NewClass->setLexicalDeclContext(CurContext);
|
||||
NewTemplate->setLexicalDeclContext(CurContext);
|
||||
|
|
|
@ -279,6 +279,8 @@ Decl *TemplateDeclInstantiator::VisitCXXRecordDecl(CXXRecordDecl *D) {
|
|||
|
||||
if (!D->isInjectedClassName())
|
||||
Record->setInstantiationOfMemberClass(D);
|
||||
else
|
||||
Record->setDescribedClassTemplate(D->getDescribedClassTemplate());
|
||||
|
||||
Owner->addDecl(Record);
|
||||
return Record;
|
||||
|
|
|
@ -1,7 +1,5 @@
|
|||
// RUN: clang-cc -fsyntax-only %s
|
||||
|
||||
// FIXME: The Fibonacci/FibonacciEval dance is here to work around our
|
||||
// inability to parse injected-class-name<template-argument-list>.
|
||||
template<unsigned I>
|
||||
struct FibonacciEval;
|
||||
|
||||
|
@ -50,3 +48,19 @@ template<> struct Fibonacci2<1> {
|
|||
|
||||
int array5_2[Fibonacci2<5>::value == 5? 1 : -1];
|
||||
int array10_2[Fibonacci2<10>::value == 55? 1 : -1];
|
||||
|
||||
template<unsigned I>
|
||||
struct Fibonacci3 {
|
||||
static const unsigned value = Fibonacci3<I-1>::value + Fibonacci3<I-2>::value;
|
||||
};
|
||||
|
||||
template<> struct Fibonacci3<0> {
|
||||
static const unsigned value = 0;
|
||||
};
|
||||
|
||||
template<> struct Fibonacci3<1> {
|
||||
static const unsigned value = 1;
|
||||
};
|
||||
|
||||
int array5_3[Fibonacci3<5>::value == 5? 1 : -1];
|
||||
int array10_3[Fibonacci3<10>::value == 55? 1 : -1];
|
||||
|
|
|
@ -0,0 +1,17 @@
|
|||
// RUN: clang-cc -fsyntax-only -verify %s
|
||||
|
||||
template<typename T>
|
||||
struct X {
|
||||
X<T*> *ptr;
|
||||
};
|
||||
|
||||
X<int> x;
|
||||
|
||||
template<>
|
||||
struct X<int***> {
|
||||
typedef X<int***> *ptr;
|
||||
};
|
||||
|
||||
// FIXME: EDG rejects this in their strict-conformance mode, but I
|
||||
// don't see any wording making this ill-formed.
|
||||
X<float>::X<int> xi = x;
|
Loading…
Reference in New Issue