| //===--- DeclTemplate.cpp - Template Declaration AST Node Implementation --===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file implements the C++ related Decl classes for templates. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "clang/AST/DeclTemplate.h" |
| #include "clang/AST/ASTContext.h" |
| #include "clang/AST/ASTMutationListener.h" |
| #include "clang/AST/DeclCXX.h" |
| #include "clang/AST/Expr.h" |
| #include "clang/AST/ExprCXX.h" |
| #include "clang/AST/TypeLoc.h" |
| #include "clang/Basic/IdentifierTable.h" |
| #include "llvm/ADT/STLExtras.h" |
| #include <memory> |
| using namespace clang; |
| |
| //===----------------------------------------------------------------------===// |
| // TemplateParameterList Implementation |
| //===----------------------------------------------------------------------===// |
| |
| TemplateParameterList::TemplateParameterList(SourceLocation TemplateLoc, |
| SourceLocation LAngleLoc, |
| NamedDecl **Params, unsigned NumParams, |
| SourceLocation RAngleLoc) |
| : TemplateLoc(TemplateLoc), LAngleLoc(LAngleLoc), RAngleLoc(RAngleLoc), |
| NumParams(NumParams), ContainsUnexpandedParameterPack(false) { |
| assert(this->NumParams == NumParams && "Too many template parameters"); |
| for (unsigned Idx = 0; Idx < NumParams; ++Idx) { |
| NamedDecl *P = Params[Idx]; |
| begin()[Idx] = P; |
| |
| if (!P->isTemplateParameterPack()) { |
| if (NonTypeTemplateParmDecl *NTTP = dyn_cast<NonTypeTemplateParmDecl>(P)) |
| if (NTTP->getType()->containsUnexpandedParameterPack()) |
| ContainsUnexpandedParameterPack = true; |
| |
| if (TemplateTemplateParmDecl *TTP = dyn_cast<TemplateTemplateParmDecl>(P)) |
| if (TTP->getTemplateParameters()->containsUnexpandedParameterPack()) |
| ContainsUnexpandedParameterPack = true; |
| |
| // FIXME: If a default argument contains an unexpanded parameter pack, the |
| // template parameter list does too. |
| } |
| } |
| } |
| |
| TemplateParameterList * |
| TemplateParameterList::Create(const ASTContext &C, SourceLocation TemplateLoc, |
| SourceLocation LAngleLoc, NamedDecl **Params, |
| unsigned NumParams, SourceLocation RAngleLoc) { |
| unsigned Size = sizeof(TemplateParameterList) |
| + sizeof(NamedDecl *) * NumParams; |
| unsigned Align = std::max(llvm::alignOf<TemplateParameterList>(), |
| llvm::alignOf<NamedDecl*>()); |
| void *Mem = C.Allocate(Size, Align); |
| return new (Mem) TemplateParameterList(TemplateLoc, LAngleLoc, Params, |
| NumParams, RAngleLoc); |
| } |
| |
| unsigned TemplateParameterList::getMinRequiredArguments() const { |
| unsigned NumRequiredArgs = 0; |
| for (iterator P = const_cast<TemplateParameterList *>(this)->begin(), |
| PEnd = const_cast<TemplateParameterList *>(this)->end(); |
| P != PEnd; ++P) { |
| if ((*P)->isTemplateParameterPack()) { |
| if (NonTypeTemplateParmDecl *NTTP = dyn_cast<NonTypeTemplateParmDecl>(*P)) |
| if (NTTP->isExpandedParameterPack()) { |
| NumRequiredArgs += NTTP->getNumExpansionTypes(); |
| continue; |
| } |
| |
| break; |
| } |
| |
| if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(*P)) { |
| if (TTP->hasDefaultArgument()) |
| break; |
| } else if (NonTypeTemplateParmDecl *NTTP |
| = dyn_cast<NonTypeTemplateParmDecl>(*P)) { |
| if (NTTP->hasDefaultArgument()) |
| break; |
| } else if (cast<TemplateTemplateParmDecl>(*P)->hasDefaultArgument()) |
| break; |
| |
| ++NumRequiredArgs; |
| } |
| |
| return NumRequiredArgs; |
| } |
| |
| unsigned TemplateParameterList::getDepth() const { |
| if (size() == 0) |
| return 0; |
| |
| const NamedDecl *FirstParm = getParam(0); |
| if (const TemplateTypeParmDecl *TTP |
| = dyn_cast<TemplateTypeParmDecl>(FirstParm)) |
| return TTP->getDepth(); |
| else if (const NonTypeTemplateParmDecl *NTTP |
| = dyn_cast<NonTypeTemplateParmDecl>(FirstParm)) |
| return NTTP->getDepth(); |
| else |
| return cast<TemplateTemplateParmDecl>(FirstParm)->getDepth(); |
| } |
| |
| static void AdoptTemplateParameterList(TemplateParameterList *Params, |
| DeclContext *Owner) { |
| for (TemplateParameterList::iterator P = Params->begin(), |
| PEnd = Params->end(); |
| P != PEnd; ++P) { |
| (*P)->setDeclContext(Owner); |
| |
| if (TemplateTemplateParmDecl *TTP = dyn_cast<TemplateTemplateParmDecl>(*P)) |
| AdoptTemplateParameterList(TTP->getTemplateParameters(), Owner); |
| } |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // RedeclarableTemplateDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| RedeclarableTemplateDecl::CommonBase *RedeclarableTemplateDecl::getCommonPtr() const { |
| if (!Common) { |
| // Walk the previous-declaration chain until we either find a declaration |
| // with a common pointer or we run out of previous declarations. |
| SmallVector<const RedeclarableTemplateDecl *, 2> PrevDecls; |
| for (const RedeclarableTemplateDecl *Prev = getPreviousDecl(); Prev; |
| Prev = Prev->getPreviousDecl()) { |
| if (Prev->Common) { |
| Common = Prev->Common; |
| break; |
| } |
| |
| PrevDecls.push_back(Prev); |
| } |
| |
| // If we never found a common pointer, allocate one now. |
| if (!Common) { |
| // FIXME: If any of the declarations is from an AST file, we probably |
| // need an update record to add the common data. |
| |
| Common = newCommon(getASTContext()); |
| } |
| |
| // Update any previous declarations we saw with the common pointer. |
| for (unsigned I = 0, N = PrevDecls.size(); I != N; ++I) |
| PrevDecls[I]->Common = Common; |
| } |
| |
| return Common; |
| } |
| |
| template <class EntryType> |
| typename RedeclarableTemplateDecl::SpecEntryTraits<EntryType>::DeclType* |
| RedeclarableTemplateDecl::findSpecializationImpl( |
| llvm::FoldingSetVector<EntryType> &Specs, |
| const TemplateArgument *Args, unsigned NumArgs, |
| void *&InsertPos) { |
| typedef SpecEntryTraits<EntryType> SETraits; |
| llvm::FoldingSetNodeID ID; |
| EntryType::Profile(ID,Args,NumArgs, getASTContext()); |
| EntryType *Entry = Specs.FindNodeOrInsertPos(ID, InsertPos); |
| return Entry ? SETraits::getMostRecentDecl(Entry) : 0; |
| } |
| |
| /// \brief Generate the injected template arguments for the given template |
| /// parameter list, e.g., for the injected-class-name of a class template. |
| static void GenerateInjectedTemplateArgs(ASTContext &Context, |
| TemplateParameterList *Params, |
| TemplateArgument *Args) { |
| for (TemplateParameterList::iterator Param = Params->begin(), |
| ParamEnd = Params->end(); |
| Param != ParamEnd; ++Param) { |
| TemplateArgument Arg; |
| if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(*Param)) { |
| QualType ArgType = Context.getTypeDeclType(TTP); |
| if (TTP->isParameterPack()) |
| ArgType = Context.getPackExpansionType(ArgType, None); |
| |
| Arg = TemplateArgument(ArgType); |
| } else if (NonTypeTemplateParmDecl *NTTP = |
| dyn_cast<NonTypeTemplateParmDecl>(*Param)) { |
| Expr *E = new (Context) DeclRefExpr(NTTP, /*enclosing*/ false, |
| NTTP->getType().getNonLValueExprType(Context), |
| Expr::getValueKindForType(NTTP->getType()), |
| NTTP->getLocation()); |
| |
| if (NTTP->isParameterPack()) |
| E = new (Context) PackExpansionExpr(Context.DependentTy, E, |
| NTTP->getLocation(), None); |
| Arg = TemplateArgument(E); |
| } else { |
| TemplateTemplateParmDecl *TTP = cast<TemplateTemplateParmDecl>(*Param); |
| if (TTP->isParameterPack()) |
| Arg = TemplateArgument(TemplateName(TTP), Optional<unsigned>()); |
| else |
| Arg = TemplateArgument(TemplateName(TTP)); |
| } |
| |
| if ((*Param)->isTemplateParameterPack()) |
| Arg = TemplateArgument::CreatePackCopy(Context, &Arg, 1); |
| |
| *Args++ = Arg; |
| } |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // FunctionTemplateDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| void FunctionTemplateDecl::DeallocateCommon(void *Ptr) { |
| static_cast<Common *>(Ptr)->~Common(); |
| } |
| |
| FunctionTemplateDecl *FunctionTemplateDecl::Create(ASTContext &C, |
| DeclContext *DC, |
| SourceLocation L, |
| DeclarationName Name, |
| TemplateParameterList *Params, |
| NamedDecl *Decl) { |
| AdoptTemplateParameterList(Params, cast<DeclContext>(Decl)); |
| return new (C) FunctionTemplateDecl(DC, L, Name, Params, Decl); |
| } |
| |
| FunctionTemplateDecl *FunctionTemplateDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(FunctionTemplateDecl)); |
| return new (Mem) FunctionTemplateDecl(0, SourceLocation(), DeclarationName(), |
| 0, 0); |
| } |
| |
| RedeclarableTemplateDecl::CommonBase * |
| FunctionTemplateDecl::newCommon(ASTContext &C) const { |
| Common *CommonPtr = new (C) Common; |
| C.AddDeallocation(DeallocateCommon, CommonPtr); |
| return CommonPtr; |
| } |
| |
| FunctionDecl * |
| FunctionTemplateDecl::findSpecialization(const TemplateArgument *Args, |
| unsigned NumArgs, void *&InsertPos) { |
| return findSpecializationImpl(getSpecializations(), Args, NumArgs, InsertPos); |
| } |
| |
| void FunctionTemplateDecl::addSpecialization( |
| FunctionTemplateSpecializationInfo *Info, void *InsertPos) { |
| if (InsertPos) |
| getSpecializations().InsertNode(Info, InsertPos); |
| else |
| getSpecializations().GetOrInsertNode(Info); |
| if (ASTMutationListener *L = getASTMutationListener()) |
| L->AddedCXXTemplateSpecialization(this, Info->Function); |
| } |
| |
| ArrayRef<TemplateArgument> FunctionTemplateDecl::getInjectedTemplateArgs() { |
| TemplateParameterList *Params = getTemplateParameters(); |
| Common *CommonPtr = getCommonPtr(); |
| if (!CommonPtr->InjectedArgs) { |
| CommonPtr->InjectedArgs |
| = new (getASTContext()) TemplateArgument[Params->size()]; |
| GenerateInjectedTemplateArgs(getASTContext(), Params, |
| CommonPtr->InjectedArgs); |
| } |
| |
| return llvm::makeArrayRef(CommonPtr->InjectedArgs, Params->size()); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // ClassTemplateDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| void ClassTemplateDecl::DeallocateCommon(void *Ptr) { |
| static_cast<Common *>(Ptr)->~Common(); |
| } |
| |
| ClassTemplateDecl *ClassTemplateDecl::Create(ASTContext &C, |
| DeclContext *DC, |
| SourceLocation L, |
| DeclarationName Name, |
| TemplateParameterList *Params, |
| NamedDecl *Decl, |
| ClassTemplateDecl *PrevDecl) { |
| AdoptTemplateParameterList(Params, cast<DeclContext>(Decl)); |
| ClassTemplateDecl *New = new (C) ClassTemplateDecl(DC, L, Name, Params, Decl); |
| New->setPreviousDeclaration(PrevDecl); |
| return New; |
| } |
| |
| ClassTemplateDecl *ClassTemplateDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(ClassTemplateDecl)); |
| return new (Mem) ClassTemplateDecl(EmptyShell()); |
| } |
| |
| void ClassTemplateDecl::LoadLazySpecializations() const { |
| Common *CommonPtr = getCommonPtr(); |
| if (CommonPtr->LazySpecializations) { |
| ASTContext &Context = getASTContext(); |
| uint32_t *Specs = CommonPtr->LazySpecializations; |
| CommonPtr->LazySpecializations = 0; |
| for (uint32_t I = 0, N = *Specs++; I != N; ++I) |
| (void)Context.getExternalSource()->GetExternalDecl(Specs[I]); |
| } |
| } |
| |
| llvm::FoldingSetVector<ClassTemplateSpecializationDecl> & |
| ClassTemplateDecl::getSpecializations() const { |
| LoadLazySpecializations(); |
| return getCommonPtr()->Specializations; |
| } |
| |
| llvm::FoldingSetVector<ClassTemplatePartialSpecializationDecl> & |
| ClassTemplateDecl::getPartialSpecializations() { |
| LoadLazySpecializations(); |
| return getCommonPtr()->PartialSpecializations; |
| } |
| |
| RedeclarableTemplateDecl::CommonBase * |
| ClassTemplateDecl::newCommon(ASTContext &C) const { |
| Common *CommonPtr = new (C) Common; |
| C.AddDeallocation(DeallocateCommon, CommonPtr); |
| return CommonPtr; |
| } |
| |
| ClassTemplateSpecializationDecl * |
| ClassTemplateDecl::findSpecialization(const TemplateArgument *Args, |
| unsigned NumArgs, void *&InsertPos) { |
| return findSpecializationImpl(getSpecializations(), Args, NumArgs, InsertPos); |
| } |
| |
| void ClassTemplateDecl::AddSpecialization(ClassTemplateSpecializationDecl *D, |
| void *InsertPos) { |
| if (InsertPos) |
| getSpecializations().InsertNode(D, InsertPos); |
| else { |
| ClassTemplateSpecializationDecl *Existing |
| = getSpecializations().GetOrInsertNode(D); |
| (void)Existing; |
| assert(Existing->isCanonicalDecl() && "Non-canonical specialization?"); |
| } |
| if (ASTMutationListener *L = getASTMutationListener()) |
| L->AddedCXXTemplateSpecialization(this, D); |
| } |
| |
| ClassTemplatePartialSpecializationDecl * |
| ClassTemplateDecl::findPartialSpecialization(const TemplateArgument *Args, |
| unsigned NumArgs, |
| void *&InsertPos) { |
| return findSpecializationImpl(getPartialSpecializations(), Args, NumArgs, |
| InsertPos); |
| } |
| |
| void ClassTemplateDecl::AddPartialSpecialization( |
| ClassTemplatePartialSpecializationDecl *D, |
| void *InsertPos) { |
| if (InsertPos) |
| getPartialSpecializations().InsertNode(D, InsertPos); |
| else { |
| ClassTemplatePartialSpecializationDecl *Existing |
| = getPartialSpecializations().GetOrInsertNode(D); |
| (void)Existing; |
| assert(Existing->isCanonicalDecl() && "Non-canonical specialization?"); |
| } |
| |
| if (ASTMutationListener *L = getASTMutationListener()) |
| L->AddedCXXTemplateSpecialization(this, D); |
| } |
| |
| void ClassTemplateDecl::getPartialSpecializations( |
| SmallVectorImpl<ClassTemplatePartialSpecializationDecl *> &PS) { |
| llvm::FoldingSetVector<ClassTemplatePartialSpecializationDecl> &PartialSpecs |
| = getPartialSpecializations(); |
| PS.clear(); |
| PS.resize(PartialSpecs.size()); |
| for (llvm::FoldingSetVector<ClassTemplatePartialSpecializationDecl>::iterator |
| P = PartialSpecs.begin(), PEnd = PartialSpecs.end(); |
| P != PEnd; ++P) { |
| assert(!PS[P->getSequenceNumber()]); |
| PS[P->getSequenceNumber()] = P->getMostRecentDecl(); |
| } |
| } |
| |
| ClassTemplatePartialSpecializationDecl * |
| ClassTemplateDecl::findPartialSpecialization(QualType T) { |
| ASTContext &Context = getASTContext(); |
| using llvm::FoldingSetVector; |
| typedef FoldingSetVector<ClassTemplatePartialSpecializationDecl>::iterator |
| partial_spec_iterator; |
| for (partial_spec_iterator P = getPartialSpecializations().begin(), |
| PEnd = getPartialSpecializations().end(); |
| P != PEnd; ++P) { |
| if (Context.hasSameType(P->getInjectedSpecializationType(), T)) |
| return P->getMostRecentDecl(); |
| } |
| |
| return 0; |
| } |
| |
| ClassTemplatePartialSpecializationDecl * |
| ClassTemplateDecl::findPartialSpecInstantiatedFromMember( |
| ClassTemplatePartialSpecializationDecl *D) { |
| Decl *DCanon = D->getCanonicalDecl(); |
| for (llvm::FoldingSetVector<ClassTemplatePartialSpecializationDecl>::iterator |
| P = getPartialSpecializations().begin(), |
| PEnd = getPartialSpecializations().end(); |
| P != PEnd; ++P) { |
| if (P->getInstantiatedFromMember()->getCanonicalDecl() == DCanon) |
| return P->getMostRecentDecl(); |
| } |
| |
| return 0; |
| } |
| |
| QualType |
| ClassTemplateDecl::getInjectedClassNameSpecialization() { |
| Common *CommonPtr = getCommonPtr(); |
| if (!CommonPtr->InjectedClassNameType.isNull()) |
| return CommonPtr->InjectedClassNameType; |
| |
| // C++0x [temp.dep.type]p2: |
| // The template argument list of a primary template is a template argument |
| // list in which the nth template argument has the value of the nth template |
| // parameter of the class template. If the nth template parameter is a |
| // template parameter pack (14.5.3), the nth template argument is a pack |
| // expansion (14.5.3) whose pattern is the name of the template parameter |
| // pack. |
| ASTContext &Context = getASTContext(); |
| TemplateParameterList *Params = getTemplateParameters(); |
| SmallVector<TemplateArgument, 16> TemplateArgs; |
| TemplateArgs.resize(Params->size()); |
| GenerateInjectedTemplateArgs(getASTContext(), Params, TemplateArgs.data()); |
| CommonPtr->InjectedClassNameType |
| = Context.getTemplateSpecializationType(TemplateName(this), |
| &TemplateArgs[0], |
| TemplateArgs.size()); |
| return CommonPtr->InjectedClassNameType; |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // TemplateTypeParm Allocation/Deallocation Method Implementations |
| //===----------------------------------------------------------------------===// |
| |
| TemplateTypeParmDecl * |
| TemplateTypeParmDecl::Create(const ASTContext &C, DeclContext *DC, |
| SourceLocation KeyLoc, SourceLocation NameLoc, |
| unsigned D, unsigned P, IdentifierInfo *Id, |
| bool Typename, bool ParameterPack) { |
| TemplateTypeParmDecl *TTPDecl = |
| new (C) TemplateTypeParmDecl(DC, KeyLoc, NameLoc, Id, Typename); |
| QualType TTPType = C.getTemplateTypeParmType(D, P, ParameterPack, TTPDecl); |
| TTPDecl->TypeForDecl = TTPType.getTypePtr(); |
| return TTPDecl; |
| } |
| |
| TemplateTypeParmDecl * |
| TemplateTypeParmDecl::CreateDeserialized(const ASTContext &C, unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(TemplateTypeParmDecl)); |
| return new (Mem) TemplateTypeParmDecl(0, SourceLocation(), SourceLocation(), |
| 0, false); |
| } |
| |
| SourceLocation TemplateTypeParmDecl::getDefaultArgumentLoc() const { |
| return hasDefaultArgument() |
| ? DefaultArgument->getTypeLoc().getBeginLoc() |
| : SourceLocation(); |
| } |
| |
| SourceRange TemplateTypeParmDecl::getSourceRange() const { |
| if (hasDefaultArgument() && !defaultArgumentWasInherited()) |
| return SourceRange(getLocStart(), |
| DefaultArgument->getTypeLoc().getEndLoc()); |
| else |
| return TypeDecl::getSourceRange(); |
| } |
| |
| unsigned TemplateTypeParmDecl::getDepth() const { |
| return TypeForDecl->getAs<TemplateTypeParmType>()->getDepth(); |
| } |
| |
| unsigned TemplateTypeParmDecl::getIndex() const { |
| return TypeForDecl->getAs<TemplateTypeParmType>()->getIndex(); |
| } |
| |
| bool TemplateTypeParmDecl::isParameterPack() const { |
| return TypeForDecl->getAs<TemplateTypeParmType>()->isParameterPack(); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // NonTypeTemplateParmDecl Method Implementations |
| //===----------------------------------------------------------------------===// |
| |
| NonTypeTemplateParmDecl::NonTypeTemplateParmDecl(DeclContext *DC, |
| SourceLocation StartLoc, |
| SourceLocation IdLoc, |
| unsigned D, unsigned P, |
| IdentifierInfo *Id, |
| QualType T, |
| TypeSourceInfo *TInfo, |
| const QualType *ExpandedTypes, |
| unsigned NumExpandedTypes, |
| TypeSourceInfo **ExpandedTInfos) |
| : DeclaratorDecl(NonTypeTemplateParm, DC, IdLoc, Id, T, TInfo, StartLoc), |
| TemplateParmPosition(D, P), DefaultArgumentAndInherited(0, false), |
| ParameterPack(true), ExpandedParameterPack(true), |
| NumExpandedTypes(NumExpandedTypes) |
| { |
| if (ExpandedTypes && ExpandedTInfos) { |
| void **TypesAndInfos = reinterpret_cast<void **>(this + 1); |
| for (unsigned I = 0; I != NumExpandedTypes; ++I) { |
| TypesAndInfos[2*I] = ExpandedTypes[I].getAsOpaquePtr(); |
| TypesAndInfos[2*I + 1] = ExpandedTInfos[I]; |
| } |
| } |
| } |
| |
| NonTypeTemplateParmDecl * |
| NonTypeTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, |
| SourceLocation StartLoc, SourceLocation IdLoc, |
| unsigned D, unsigned P, IdentifierInfo *Id, |
| QualType T, bool ParameterPack, |
| TypeSourceInfo *TInfo) { |
| return new (C) NonTypeTemplateParmDecl(DC, StartLoc, IdLoc, D, P, Id, |
| T, ParameterPack, TInfo); |
| } |
| |
| NonTypeTemplateParmDecl * |
| NonTypeTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, |
| SourceLocation StartLoc, SourceLocation IdLoc, |
| unsigned D, unsigned P, |
| IdentifierInfo *Id, QualType T, |
| TypeSourceInfo *TInfo, |
| const QualType *ExpandedTypes, |
| unsigned NumExpandedTypes, |
| TypeSourceInfo **ExpandedTInfos) { |
| unsigned Size = sizeof(NonTypeTemplateParmDecl) |
| + NumExpandedTypes * 2 * sizeof(void*); |
| void *Mem = C.Allocate(Size); |
| return new (Mem) NonTypeTemplateParmDecl(DC, StartLoc, IdLoc, |
| D, P, Id, T, TInfo, |
| ExpandedTypes, NumExpandedTypes, |
| ExpandedTInfos); |
| } |
| |
| NonTypeTemplateParmDecl * |
| NonTypeTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(NonTypeTemplateParmDecl)); |
| return new (Mem) NonTypeTemplateParmDecl(0, SourceLocation(), |
| SourceLocation(), 0, 0, 0, |
| QualType(), false, 0); |
| } |
| |
| NonTypeTemplateParmDecl * |
| NonTypeTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID, |
| unsigned NumExpandedTypes) { |
| unsigned Size = sizeof(NonTypeTemplateParmDecl) |
| + NumExpandedTypes * 2 * sizeof(void*); |
| |
| void *Mem = AllocateDeserializedDecl(C, ID, Size); |
| return new (Mem) NonTypeTemplateParmDecl(0, SourceLocation(), |
| SourceLocation(), 0, 0, 0, |
| QualType(), 0, 0, NumExpandedTypes, |
| 0); |
| } |
| |
| SourceRange NonTypeTemplateParmDecl::getSourceRange() const { |
| if (hasDefaultArgument() && !defaultArgumentWasInherited()) |
| return SourceRange(getOuterLocStart(), |
| getDefaultArgument()->getSourceRange().getEnd()); |
| return DeclaratorDecl::getSourceRange(); |
| } |
| |
| SourceLocation NonTypeTemplateParmDecl::getDefaultArgumentLoc() const { |
| return hasDefaultArgument() |
| ? getDefaultArgument()->getSourceRange().getBegin() |
| : SourceLocation(); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // TemplateTemplateParmDecl Method Implementations |
| //===----------------------------------------------------------------------===// |
| |
| void TemplateTemplateParmDecl::anchor() { } |
| |
| TemplateTemplateParmDecl::TemplateTemplateParmDecl( |
| DeclContext *DC, SourceLocation L, unsigned D, unsigned P, |
| IdentifierInfo *Id, TemplateParameterList *Params, |
| unsigned NumExpansions, TemplateParameterList * const *Expansions) |
| : TemplateDecl(TemplateTemplateParm, DC, L, Id, Params), |
| TemplateParmPosition(D, P), DefaultArgument(), |
| DefaultArgumentWasInherited(false), ParameterPack(true), |
| ExpandedParameterPack(true), NumExpandedParams(NumExpansions) { |
| if (Expansions) |
| std::memcpy(reinterpret_cast<void*>(this + 1), Expansions, |
| sizeof(TemplateParameterList*) * NumExpandedParams); |
| } |
| |
| TemplateTemplateParmDecl * |
| TemplateTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, |
| SourceLocation L, unsigned D, unsigned P, |
| bool ParameterPack, IdentifierInfo *Id, |
| TemplateParameterList *Params) { |
| return new (C) TemplateTemplateParmDecl(DC, L, D, P, ParameterPack, Id, |
| Params); |
| } |
| |
| TemplateTemplateParmDecl * |
| TemplateTemplateParmDecl::Create(const ASTContext &C, DeclContext *DC, |
| SourceLocation L, unsigned D, unsigned P, |
| IdentifierInfo *Id, |
| TemplateParameterList *Params, |
| ArrayRef<TemplateParameterList *> Expansions) { |
| void *Mem = C.Allocate(sizeof(TemplateTemplateParmDecl) + |
| sizeof(TemplateParameterList*) * Expansions.size()); |
| return new (Mem) TemplateTemplateParmDecl(DC, L, D, P, Id, Params, |
| Expansions.size(), |
| Expansions.data()); |
| } |
| |
| TemplateTemplateParmDecl * |
| TemplateTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(TemplateTemplateParmDecl)); |
| return new (Mem) TemplateTemplateParmDecl(0, SourceLocation(), 0, 0, false, |
| 0, 0); |
| } |
| |
| TemplateTemplateParmDecl * |
| TemplateTemplateParmDecl::CreateDeserialized(ASTContext &C, unsigned ID, |
| unsigned NumExpansions) { |
| unsigned Size = sizeof(TemplateTemplateParmDecl) + |
| sizeof(TemplateParameterList*) * NumExpansions; |
| void *Mem = AllocateDeserializedDecl(C, ID, Size); |
| return new (Mem) TemplateTemplateParmDecl(0, SourceLocation(), 0, 0, 0, 0, |
| NumExpansions, 0); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // TemplateArgumentList Implementation |
| //===----------------------------------------------------------------------===// |
| TemplateArgumentList * |
| TemplateArgumentList::CreateCopy(ASTContext &Context, |
| const TemplateArgument *Args, |
| unsigned NumArgs) { |
| std::size_t Size = sizeof(TemplateArgumentList) |
| + NumArgs * sizeof(TemplateArgument); |
| void *Mem = Context.Allocate(Size); |
| TemplateArgument *StoredArgs |
| = reinterpret_cast<TemplateArgument *>( |
| static_cast<TemplateArgumentList *>(Mem) + 1); |
| std::uninitialized_copy(Args, Args + NumArgs, StoredArgs); |
| return new (Mem) TemplateArgumentList(StoredArgs, NumArgs, true); |
| } |
| |
| FunctionTemplateSpecializationInfo * |
| FunctionTemplateSpecializationInfo::Create(ASTContext &C, FunctionDecl *FD, |
| FunctionTemplateDecl *Template, |
| TemplateSpecializationKind TSK, |
| const TemplateArgumentList *TemplateArgs, |
| const TemplateArgumentListInfo *TemplateArgsAsWritten, |
| SourceLocation POI) { |
| const ASTTemplateArgumentListInfo *ArgsAsWritten = 0; |
| if (TemplateArgsAsWritten) |
| ArgsAsWritten = ASTTemplateArgumentListInfo::Create(C, |
| *TemplateArgsAsWritten); |
| |
| return new (C) FunctionTemplateSpecializationInfo(FD, Template, TSK, |
| TemplateArgs, |
| ArgsAsWritten, |
| POI); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // TemplateDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| void TemplateDecl::anchor() { } |
| |
| //===----------------------------------------------------------------------===// |
| // ClassTemplateSpecializationDecl Implementation |
| //===----------------------------------------------------------------------===// |
| ClassTemplateSpecializationDecl:: |
| ClassTemplateSpecializationDecl(ASTContext &Context, Kind DK, TagKind TK, |
| DeclContext *DC, SourceLocation StartLoc, |
| SourceLocation IdLoc, |
| ClassTemplateDecl *SpecializedTemplate, |
| const TemplateArgument *Args, |
| unsigned NumArgs, |
| ClassTemplateSpecializationDecl *PrevDecl) |
| : CXXRecordDecl(DK, TK, DC, StartLoc, IdLoc, |
| SpecializedTemplate->getIdentifier(), |
| PrevDecl), |
| SpecializedTemplate(SpecializedTemplate), |
| ExplicitInfo(0), |
| TemplateArgs(TemplateArgumentList::CreateCopy(Context, Args, NumArgs)), |
| SpecializationKind(TSK_Undeclared) { |
| } |
| |
| ClassTemplateSpecializationDecl::ClassTemplateSpecializationDecl(Kind DK) |
| : CXXRecordDecl(DK, TTK_Struct, 0, SourceLocation(), SourceLocation(), 0, 0), |
| ExplicitInfo(0), |
| SpecializationKind(TSK_Undeclared) { |
| } |
| |
| ClassTemplateSpecializationDecl * |
| ClassTemplateSpecializationDecl::Create(ASTContext &Context, TagKind TK, |
| DeclContext *DC, |
| SourceLocation StartLoc, |
| SourceLocation IdLoc, |
| ClassTemplateDecl *SpecializedTemplate, |
| const TemplateArgument *Args, |
| unsigned NumArgs, |
| ClassTemplateSpecializationDecl *PrevDecl) { |
| ClassTemplateSpecializationDecl *Result |
| = new (Context)ClassTemplateSpecializationDecl(Context, |
| ClassTemplateSpecialization, |
| TK, DC, StartLoc, IdLoc, |
| SpecializedTemplate, |
| Args, NumArgs, |
| PrevDecl); |
| Result->MayHaveOutOfDateDef = false; |
| |
| Context.getTypeDeclType(Result, PrevDecl); |
| return Result; |
| } |
| |
| ClassTemplateSpecializationDecl * |
| ClassTemplateSpecializationDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, |
| sizeof(ClassTemplateSpecializationDecl)); |
| ClassTemplateSpecializationDecl *Result = |
| new (Mem) ClassTemplateSpecializationDecl(ClassTemplateSpecialization); |
| Result->MayHaveOutOfDateDef = false; |
| return Result; |
| } |
| |
| void ClassTemplateSpecializationDecl::getNameForDiagnostic( |
| raw_ostream &OS, const PrintingPolicy &Policy, bool Qualified) const { |
| NamedDecl::getNameForDiagnostic(OS, Policy, Qualified); |
| |
| const TemplateArgumentList &TemplateArgs = getTemplateArgs(); |
| TemplateSpecializationType::PrintTemplateArgumentList( |
| OS, TemplateArgs.data(), TemplateArgs.size(), Policy); |
| } |
| |
| ClassTemplateDecl * |
| ClassTemplateSpecializationDecl::getSpecializedTemplate() const { |
| if (SpecializedPartialSpecialization *PartialSpec |
| = SpecializedTemplate.dyn_cast<SpecializedPartialSpecialization*>()) |
| return PartialSpec->PartialSpecialization->getSpecializedTemplate(); |
| return SpecializedTemplate.get<ClassTemplateDecl*>(); |
| } |
| |
| SourceRange |
| ClassTemplateSpecializationDecl::getSourceRange() const { |
| if (ExplicitInfo) { |
| SourceLocation Begin = getTemplateKeywordLoc(); |
| if (Begin.isValid()) { |
| // Here we have an explicit (partial) specialization or instantiation. |
| assert(getSpecializationKind() == TSK_ExplicitSpecialization || |
| getSpecializationKind() == TSK_ExplicitInstantiationDeclaration || |
| getSpecializationKind() == TSK_ExplicitInstantiationDefinition); |
| if (getExternLoc().isValid()) |
| Begin = getExternLoc(); |
| SourceLocation End = getRBraceLoc(); |
| if (End.isInvalid()) |
| End = getTypeAsWritten()->getTypeLoc().getEndLoc(); |
| return SourceRange(Begin, End); |
| } |
| // An implicit instantiation of a class template partial specialization |
| // uses ExplicitInfo to record the TypeAsWritten, but the source |
| // locations should be retrieved from the instantiation pattern. |
| typedef ClassTemplatePartialSpecializationDecl CTPSDecl; |
| CTPSDecl *ctpsd = const_cast<CTPSDecl*>(cast<CTPSDecl>(this)); |
| CTPSDecl *inst_from = ctpsd->getInstantiatedFromMember(); |
| assert(inst_from != 0); |
| return inst_from->getSourceRange(); |
| } |
| else { |
| // No explicit info available. |
| llvm::PointerUnion<ClassTemplateDecl *, |
| ClassTemplatePartialSpecializationDecl *> |
| inst_from = getInstantiatedFrom(); |
| if (inst_from.isNull()) |
| return getSpecializedTemplate()->getSourceRange(); |
| if (ClassTemplateDecl *ctd = inst_from.dyn_cast<ClassTemplateDecl*>()) |
| return ctd->getSourceRange(); |
| return inst_from.get<ClassTemplatePartialSpecializationDecl*>() |
| ->getSourceRange(); |
| } |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // ClassTemplatePartialSpecializationDecl Implementation |
| //===----------------------------------------------------------------------===// |
| void ClassTemplatePartialSpecializationDecl::anchor() { } |
| |
| ClassTemplatePartialSpecializationDecl:: |
| ClassTemplatePartialSpecializationDecl(ASTContext &Context, TagKind TK, |
| DeclContext *DC, |
| SourceLocation StartLoc, |
| SourceLocation IdLoc, |
| TemplateParameterList *Params, |
| ClassTemplateDecl *SpecializedTemplate, |
| const TemplateArgument *Args, |
| unsigned NumArgs, |
| TemplateArgumentLoc *ArgInfos, |
| unsigned NumArgInfos, |
| ClassTemplatePartialSpecializationDecl *PrevDecl, |
| unsigned SequenceNumber) |
| : ClassTemplateSpecializationDecl(Context, |
| ClassTemplatePartialSpecialization, |
| TK, DC, StartLoc, IdLoc, |
| SpecializedTemplate, |
| Args, NumArgs, PrevDecl), |
| TemplateParams(Params), ArgsAsWritten(ArgInfos), |
| NumArgsAsWritten(NumArgInfos), SequenceNumber(SequenceNumber), |
| InstantiatedFromMember(0, false) |
| { |
| AdoptTemplateParameterList(Params, this); |
| } |
| |
| ClassTemplatePartialSpecializationDecl * |
| ClassTemplatePartialSpecializationDecl:: |
| Create(ASTContext &Context, TagKind TK,DeclContext *DC, |
| SourceLocation StartLoc, SourceLocation IdLoc, |
| TemplateParameterList *Params, |
| ClassTemplateDecl *SpecializedTemplate, |
| const TemplateArgument *Args, |
| unsigned NumArgs, |
| const TemplateArgumentListInfo &ArgInfos, |
| QualType CanonInjectedType, |
| ClassTemplatePartialSpecializationDecl *PrevDecl, |
| unsigned SequenceNumber) { |
| unsigned N = ArgInfos.size(); |
| TemplateArgumentLoc *ClonedArgs = new (Context) TemplateArgumentLoc[N]; |
| for (unsigned I = 0; I != N; ++I) |
| ClonedArgs[I] = ArgInfos[I]; |
| |
| ClassTemplatePartialSpecializationDecl *Result |
| = new (Context)ClassTemplatePartialSpecializationDecl(Context, TK, DC, |
| StartLoc, IdLoc, |
| Params, |
| SpecializedTemplate, |
| Args, NumArgs, |
| ClonedArgs, N, |
| PrevDecl, |
| SequenceNumber); |
| Result->setSpecializationKind(TSK_ExplicitSpecialization); |
| Result->MayHaveOutOfDateDef = false; |
| |
| Context.getInjectedClassNameType(Result, CanonInjectedType); |
| return Result; |
| } |
| |
| ClassTemplatePartialSpecializationDecl * |
| ClassTemplatePartialSpecializationDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, |
| sizeof(ClassTemplatePartialSpecializationDecl)); |
| ClassTemplatePartialSpecializationDecl *Result |
| = new (Mem) ClassTemplatePartialSpecializationDecl(); |
| Result->MayHaveOutOfDateDef = false; |
| return Result; |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // FriendTemplateDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| void FriendTemplateDecl::anchor() { } |
| |
| FriendTemplateDecl *FriendTemplateDecl::Create(ASTContext &Context, |
| DeclContext *DC, |
| SourceLocation L, |
| unsigned NParams, |
| TemplateParameterList **Params, |
| FriendUnion Friend, |
| SourceLocation FLoc) { |
| FriendTemplateDecl *Result |
| = new (Context) FriendTemplateDecl(DC, L, NParams, Params, Friend, FLoc); |
| return Result; |
| } |
| |
| FriendTemplateDecl *FriendTemplateDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(FriendTemplateDecl)); |
| return new (Mem) FriendTemplateDecl(EmptyShell()); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // TypeAliasTemplateDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| TypeAliasTemplateDecl *TypeAliasTemplateDecl::Create(ASTContext &C, |
| DeclContext *DC, |
| SourceLocation L, |
| DeclarationName Name, |
| TemplateParameterList *Params, |
| NamedDecl *Decl) { |
| AdoptTemplateParameterList(Params, DC); |
| return new (C) TypeAliasTemplateDecl(DC, L, Name, Params, Decl); |
| } |
| |
| TypeAliasTemplateDecl *TypeAliasTemplateDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, sizeof(TypeAliasTemplateDecl)); |
| return new (Mem) TypeAliasTemplateDecl(0, SourceLocation(), DeclarationName(), |
| 0, 0); |
| } |
| |
| void TypeAliasTemplateDecl::DeallocateCommon(void *Ptr) { |
| static_cast<Common *>(Ptr)->~Common(); |
| } |
| RedeclarableTemplateDecl::CommonBase * |
| TypeAliasTemplateDecl::newCommon(ASTContext &C) const { |
| Common *CommonPtr = new (C) Common; |
| C.AddDeallocation(DeallocateCommon, CommonPtr); |
| return CommonPtr; |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // ClassScopeFunctionSpecializationDecl Implementation |
| //===----------------------------------------------------------------------===// |
| |
| void ClassScopeFunctionSpecializationDecl::anchor() { } |
| |
| ClassScopeFunctionSpecializationDecl * |
| ClassScopeFunctionSpecializationDecl::CreateDeserialized(ASTContext &C, |
| unsigned ID) { |
| void *Mem = AllocateDeserializedDecl(C, ID, |
| sizeof(ClassScopeFunctionSpecializationDecl)); |
| return new (Mem) ClassScopeFunctionSpecializationDecl(0, SourceLocation(), 0, |
| false, TemplateArgumentListInfo()); |
| } |