| //===- ExprCXX.cpp - (C++) Expression 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 subclesses of Expr class declared in ExprCXX.h | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | #include "clang/AST/ExprCXX.h" | 
 | #include "clang/AST/ASTContext.h" | 
 | #include "clang/AST/Attr.h" | 
 | #include "clang/AST/Decl.h" | 
 | #include "clang/AST/DeclAccessPair.h" | 
 | #include "clang/AST/DeclBase.h" | 
 | #include "clang/AST/DeclCXX.h" | 
 | #include "clang/AST/DeclarationName.h" | 
 | #include "clang/AST/Expr.h" | 
 | #include "clang/AST/LambdaCapture.h" | 
 | #include "clang/AST/NestedNameSpecifier.h" | 
 | #include "clang/AST/TemplateBase.h" | 
 | #include "clang/AST/Type.h" | 
 | #include "clang/AST/TypeLoc.h" | 
 | #include "clang/Basic/LLVM.h" | 
 | #include "clang/Basic/OperatorKinds.h" | 
 | #include "clang/Basic/SourceLocation.h" | 
 | #include "clang/Basic/Specifiers.h" | 
 | #include "llvm/ADT/ArrayRef.h" | 
 | #include "llvm/Support/Casting.h" | 
 | #include "llvm/Support/ErrorHandling.h" | 
 | #include <cassert> | 
 | #include <cstddef> | 
 | #include <cstring> | 
 | #include <memory> | 
 |  | 
 | using namespace clang; | 
 |  | 
 | //===----------------------------------------------------------------------===// | 
 | //  Child Iterators for iterating over subexpressions/substatements | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | bool CXXOperatorCallExpr::isInfixBinaryOp() const { | 
 |   // An infix binary operator is any operator with two arguments other than | 
 |   // operator() and operator[]. Note that none of these operators can have | 
 |   // default arguments, so it suffices to check the number of argument | 
 |   // expressions. | 
 |   if (getNumArgs() != 2) | 
 |     return false; | 
 |  | 
 |   switch (getOperator()) { | 
 |   case OO_Call: case OO_Subscript: | 
 |     return false; | 
 |   default: | 
 |     return true; | 
 |   } | 
 | } | 
 |  | 
 | bool CXXTypeidExpr::isPotentiallyEvaluated() const { | 
 |   if (isTypeOperand()) | 
 |     return false; | 
 |  | 
 |   // C++11 [expr.typeid]p3: | 
 |   //   When typeid is applied to an expression other than a glvalue of | 
 |   //   polymorphic class type, [...] the expression is an unevaluated operand. | 
 |   const Expr *E = getExprOperand(); | 
 |   if (const CXXRecordDecl *RD = E->getType()->getAsCXXRecordDecl()) | 
 |     if (RD->isPolymorphic() && E->isGLValue()) | 
 |       return true; | 
 |  | 
 |   return false; | 
 | } | 
 |  | 
 | QualType CXXTypeidExpr::getTypeOperand(ASTContext &Context) const { | 
 |   assert(isTypeOperand() && "Cannot call getTypeOperand for typeid(expr)"); | 
 |   Qualifiers Quals; | 
 |   return Context.getUnqualifiedArrayType( | 
 |       Operand.get<TypeSourceInfo *>()->getType().getNonReferenceType(), Quals); | 
 | } | 
 |  | 
 | QualType CXXUuidofExpr::getTypeOperand(ASTContext &Context) const { | 
 |   assert(isTypeOperand() && "Cannot call getTypeOperand for __uuidof(expr)"); | 
 |   Qualifiers Quals; | 
 |   return Context.getUnqualifiedArrayType( | 
 |       Operand.get<TypeSourceInfo *>()->getType().getNonReferenceType(), Quals); | 
 | } | 
 |  | 
 | // CXXScalarValueInitExpr | 
 | SourceLocation CXXScalarValueInitExpr::getBeginLoc() const { | 
 |   return TypeInfo ? TypeInfo->getTypeLoc().getBeginLoc() : RParenLoc; | 
 | } | 
 |  | 
 | // CXXNewExpr | 
 | CXXNewExpr::CXXNewExpr(const ASTContext &C, bool globalNew, | 
 |                        FunctionDecl *operatorNew, FunctionDecl *operatorDelete, | 
 |                        bool PassAlignment, bool usualArrayDeleteWantsSize, | 
 |                        ArrayRef<Expr*> placementArgs, | 
 |                        SourceRange typeIdParens, Expr *arraySize, | 
 |                        InitializationStyle initializationStyle, | 
 |                        Expr *initializer, QualType ty, | 
 |                        TypeSourceInfo *allocatedTypeInfo, | 
 |                        SourceRange Range, SourceRange directInitRange) | 
 |     : Expr(CXXNewExprClass, ty, VK_RValue, OK_Ordinary, ty->isDependentType(), | 
 |            ty->isDependentType(), ty->isInstantiationDependentType(), | 
 |            ty->containsUnexpandedParameterPack()), | 
 |       OperatorNew(operatorNew), OperatorDelete(operatorDelete), | 
 |       AllocatedTypeInfo(allocatedTypeInfo), TypeIdParens(typeIdParens), | 
 |       Range(Range), DirectInitRange(directInitRange), GlobalNew(globalNew), | 
 |       PassAlignment(PassAlignment), | 
 |       UsualArrayDeleteWantsSize(usualArrayDeleteWantsSize) { | 
 |   assert((initializer != nullptr || initializationStyle == NoInit) && | 
 |          "Only NoInit can have no initializer."); | 
 |   StoredInitializationStyle = initializer ? initializationStyle + 1 : 0; | 
 |   AllocateArgsArray(C, arraySize != nullptr, placementArgs.size(), | 
 |                     initializer != nullptr); | 
 |   unsigned i = 0; | 
 |   if (Array) { | 
 |     if (arraySize->isInstantiationDependent()) | 
 |       ExprBits.InstantiationDependent = true; | 
 |  | 
 |     if (arraySize->containsUnexpandedParameterPack()) | 
 |       ExprBits.ContainsUnexpandedParameterPack = true; | 
 |  | 
 |     SubExprs[i++] = arraySize; | 
 |   } | 
 |  | 
 |   if (initializer) { | 
 |     if (initializer->isInstantiationDependent()) | 
 |       ExprBits.InstantiationDependent = true; | 
 |  | 
 |     if (initializer->containsUnexpandedParameterPack()) | 
 |       ExprBits.ContainsUnexpandedParameterPack = true; | 
 |  | 
 |     SubExprs[i++] = initializer; | 
 |   } | 
 |  | 
 |   for (unsigned j = 0; j != placementArgs.size(); ++j) { | 
 |     if (placementArgs[j]->isInstantiationDependent()) | 
 |       ExprBits.InstantiationDependent = true; | 
 |     if (placementArgs[j]->containsUnexpandedParameterPack()) | 
 |       ExprBits.ContainsUnexpandedParameterPack = true; | 
 |  | 
 |     SubExprs[i++] = placementArgs[j]; | 
 |   } | 
 |  | 
 |   switch (getInitializationStyle()) { | 
 |   case CallInit: | 
 |     this->Range.setEnd(DirectInitRange.getEnd()); break; | 
 |   case ListInit: | 
 |     this->Range.setEnd(getInitializer()->getSourceRange().getEnd()); break; | 
 |   default: | 
 |     if (TypeIdParens.isValid()) | 
 |       this->Range.setEnd(TypeIdParens.getEnd()); | 
 |     break; | 
 |   } | 
 | } | 
 |  | 
 | void CXXNewExpr::AllocateArgsArray(const ASTContext &C, bool isArray, | 
 |                                    unsigned numPlaceArgs, bool hasInitializer){ | 
 |   assert(SubExprs == nullptr && "SubExprs already allocated"); | 
 |   Array = isArray; | 
 |   NumPlacementArgs = numPlaceArgs; | 
 |  | 
 |   unsigned TotalSize = Array + hasInitializer + NumPlacementArgs; | 
 |   SubExprs = new (C) Stmt*[TotalSize]; | 
 | } | 
 |  | 
 | bool CXXNewExpr::shouldNullCheckAllocation(const ASTContext &Ctx) const { | 
 |   return getOperatorNew()->getType()->castAs<FunctionProtoType>() | 
 |                                           ->isNothrow() && | 
 |          !getOperatorNew()->isReservedGlobalPlacementOperator(); | 
 | } | 
 |  | 
 | // CXXDeleteExpr | 
 | QualType CXXDeleteExpr::getDestroyedType() const { | 
 |   const Expr *Arg = getArgument(); | 
 |  | 
 |   // For a destroying operator delete, we may have implicitly converted the | 
 |   // pointer type to the type of the parameter of the 'operator delete' | 
 |   // function. | 
 |   while (const auto *ICE = dyn_cast<ImplicitCastExpr>(Arg)) { | 
 |     if (ICE->getCastKind() == CK_DerivedToBase || | 
 |         ICE->getCastKind() == CK_UncheckedDerivedToBase || | 
 |         ICE->getCastKind() == CK_NoOp) { | 
 |       assert((ICE->getCastKind() == CK_NoOp || | 
 |               getOperatorDelete()->isDestroyingOperatorDelete()) && | 
 |              "only a destroying operator delete can have a converted arg"); | 
 |       Arg = ICE->getSubExpr(); | 
 |     } else | 
 |       break; | 
 |   } | 
 |  | 
 |   // The type-to-delete may not be a pointer if it's a dependent type. | 
 |   const QualType ArgType = Arg->getType(); | 
 |  | 
 |   if (ArgType->isDependentType() && !ArgType->isPointerType()) | 
 |     return QualType(); | 
 |  | 
 |   return ArgType->getAs<PointerType>()->getPointeeType(); | 
 | } | 
 |  | 
 | // CXXPseudoDestructorExpr | 
 | PseudoDestructorTypeStorage::PseudoDestructorTypeStorage(TypeSourceInfo *Info) | 
 |     : Type(Info) { | 
 |   Location = Info->getTypeLoc().getLocalSourceRange().getBegin(); | 
 | } | 
 |  | 
 | CXXPseudoDestructorExpr::CXXPseudoDestructorExpr(const ASTContext &Context, | 
 |                 Expr *Base, bool isArrow, SourceLocation OperatorLoc, | 
 |                 NestedNameSpecifierLoc QualifierLoc, TypeSourceInfo *ScopeType, | 
 |                 SourceLocation ColonColonLoc, SourceLocation TildeLoc, | 
 |                 PseudoDestructorTypeStorage DestroyedType) | 
 |   : Expr(CXXPseudoDestructorExprClass, | 
 |          Context.BoundMemberTy, | 
 |          VK_RValue, OK_Ordinary, | 
 |          /*isTypeDependent=*/(Base->isTypeDependent() || | 
 |            (DestroyedType.getTypeSourceInfo() && | 
 |             DestroyedType.getTypeSourceInfo()->getType()->isDependentType())), | 
 |          /*isValueDependent=*/Base->isValueDependent(), | 
 |          (Base->isInstantiationDependent() || | 
 |           (QualifierLoc && | 
 |            QualifierLoc.getNestedNameSpecifier()->isInstantiationDependent()) || | 
 |           (ScopeType && | 
 |            ScopeType->getType()->isInstantiationDependentType()) || | 
 |           (DestroyedType.getTypeSourceInfo() && | 
 |            DestroyedType.getTypeSourceInfo()->getType() | 
 |                                              ->isInstantiationDependentType())), | 
 |          // ContainsUnexpandedParameterPack | 
 |          (Base->containsUnexpandedParameterPack() || | 
 |           (QualifierLoc && | 
 |            QualifierLoc.getNestedNameSpecifier() | 
 |                                         ->containsUnexpandedParameterPack()) || | 
 |           (ScopeType && | 
 |            ScopeType->getType()->containsUnexpandedParameterPack()) || | 
 |           (DestroyedType.getTypeSourceInfo() && | 
 |            DestroyedType.getTypeSourceInfo()->getType() | 
 |                                    ->containsUnexpandedParameterPack()))), | 
 |     Base(static_cast<Stmt *>(Base)), IsArrow(isArrow), | 
 |     OperatorLoc(OperatorLoc), QualifierLoc(QualifierLoc), | 
 |     ScopeType(ScopeType), ColonColonLoc(ColonColonLoc), TildeLoc(TildeLoc), | 
 |     DestroyedType(DestroyedType) {} | 
 |  | 
 | QualType CXXPseudoDestructorExpr::getDestroyedType() const { | 
 |   if (TypeSourceInfo *TInfo = DestroyedType.getTypeSourceInfo()) | 
 |     return TInfo->getType(); | 
 |  | 
 |   return QualType(); | 
 | } | 
 |  | 
 | SourceLocation CXXPseudoDestructorExpr::getEndLoc() const { | 
 |   SourceLocation End = DestroyedType.getLocation(); | 
 |   if (TypeSourceInfo *TInfo = DestroyedType.getTypeSourceInfo()) | 
 |     End = TInfo->getTypeLoc().getLocalSourceRange().getEnd(); | 
 |   return End; | 
 | } | 
 |  | 
 | // UnresolvedLookupExpr | 
 | UnresolvedLookupExpr * | 
 | UnresolvedLookupExpr::Create(const ASTContext &C, | 
 |                              CXXRecordDecl *NamingClass, | 
 |                              NestedNameSpecifierLoc QualifierLoc, | 
 |                              SourceLocation TemplateKWLoc, | 
 |                              const DeclarationNameInfo &NameInfo, | 
 |                              bool ADL, | 
 |                              const TemplateArgumentListInfo *Args, | 
 |                              UnresolvedSetIterator Begin, | 
 |                              UnresolvedSetIterator End) { | 
 |   assert(Args || TemplateKWLoc.isValid()); | 
 |   unsigned num_args = Args ? Args->size() : 0; | 
 |  | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>(1, | 
 |                                                                       num_args); | 
 |   void *Mem = C.Allocate(Size, alignof(UnresolvedLookupExpr)); | 
 |   return new (Mem) UnresolvedLookupExpr(C, NamingClass, QualifierLoc, | 
 |                                         TemplateKWLoc, NameInfo, | 
 |                                         ADL, /*Overload*/ true, Args, | 
 |                                         Begin, End); | 
 | } | 
 |  | 
 | UnresolvedLookupExpr * | 
 | UnresolvedLookupExpr::CreateEmpty(const ASTContext &C, | 
 |                                   bool HasTemplateKWAndArgsInfo, | 
 |                                   unsigned NumTemplateArgs) { | 
 |   assert(NumTemplateArgs == 0 || HasTemplateKWAndArgsInfo); | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, NumTemplateArgs); | 
 |   void *Mem = C.Allocate(Size, alignof(UnresolvedLookupExpr)); | 
 |   auto *E = new (Mem) UnresolvedLookupExpr(EmptyShell()); | 
 |   E->HasTemplateKWAndArgsInfo = HasTemplateKWAndArgsInfo; | 
 |   return E; | 
 | } | 
 |  | 
 | OverloadExpr::OverloadExpr(StmtClass K, const ASTContext &C, | 
 |                            NestedNameSpecifierLoc QualifierLoc, | 
 |                            SourceLocation TemplateKWLoc, | 
 |                            const DeclarationNameInfo &NameInfo, | 
 |                            const TemplateArgumentListInfo *TemplateArgs, | 
 |                            UnresolvedSetIterator Begin, | 
 |                            UnresolvedSetIterator End, | 
 |                            bool KnownDependent, | 
 |                            bool KnownInstantiationDependent, | 
 |                            bool KnownContainsUnexpandedParameterPack) | 
 |     : Expr(K, C.OverloadTy, VK_LValue, OK_Ordinary, KnownDependent, | 
 |            KnownDependent, | 
 |            (KnownInstantiationDependent || | 
 |             NameInfo.isInstantiationDependent() || | 
 |             (QualifierLoc && | 
 |             QualifierLoc.getNestedNameSpecifier()->isInstantiationDependent())), | 
 |            (KnownContainsUnexpandedParameterPack || | 
 |             NameInfo.containsUnexpandedParameterPack() || | 
 |             (QualifierLoc && | 
 |              QualifierLoc.getNestedNameSpecifier() | 
 |                                         ->containsUnexpandedParameterPack()))), | 
 |       NameInfo(NameInfo), QualifierLoc(QualifierLoc), NumResults(End - Begin), | 
 |       HasTemplateKWAndArgsInfo(TemplateArgs != nullptr || | 
 |                                TemplateKWLoc.isValid()) { | 
 |   NumResults = End - Begin; | 
 |   if (NumResults) { | 
 |     // Determine whether this expression is type-dependent. | 
 |     for (UnresolvedSetImpl::const_iterator I = Begin; I != End; ++I) { | 
 |       if ((*I)->getDeclContext()->isDependentContext() || | 
 |           isa<UnresolvedUsingValueDecl>(*I)) { | 
 |         ExprBits.TypeDependent = true; | 
 |         ExprBits.ValueDependent = true; | 
 |         ExprBits.InstantiationDependent = true; | 
 |       } | 
 |     } | 
 |  | 
 |     Results = static_cast<DeclAccessPair *>(C.Allocate( | 
 |         sizeof(DeclAccessPair) * NumResults, alignof(DeclAccessPair))); | 
 |     memcpy(Results, Begin.I, NumResults * sizeof(DeclAccessPair)); | 
 |   } | 
 |  | 
 |   // If we have explicit template arguments, check for dependent | 
 |   // template arguments and whether they contain any unexpanded pack | 
 |   // expansions. | 
 |   if (TemplateArgs) { | 
 |     bool Dependent = false; | 
 |     bool InstantiationDependent = false; | 
 |     bool ContainsUnexpandedParameterPack = false; | 
 |     getTrailingASTTemplateKWAndArgsInfo()->initializeFrom( | 
 |         TemplateKWLoc, *TemplateArgs, getTrailingTemplateArgumentLoc(), | 
 |         Dependent, InstantiationDependent, ContainsUnexpandedParameterPack); | 
 |  | 
 |     if (Dependent) { | 
 |       ExprBits.TypeDependent = true; | 
 |       ExprBits.ValueDependent = true; | 
 |     } | 
 |     if (InstantiationDependent) | 
 |       ExprBits.InstantiationDependent = true; | 
 |     if (ContainsUnexpandedParameterPack) | 
 |       ExprBits.ContainsUnexpandedParameterPack = true; | 
 |   } else if (TemplateKWLoc.isValid()) { | 
 |     getTrailingASTTemplateKWAndArgsInfo()->initializeFrom(TemplateKWLoc); | 
 |   } | 
 |  | 
 |   if (isTypeDependent()) | 
 |     setType(C.DependentTy); | 
 | } | 
 |  | 
 | void OverloadExpr::initializeResults(const ASTContext &C, | 
 |                                      UnresolvedSetIterator Begin, | 
 |                                      UnresolvedSetIterator End) { | 
 |   assert(!Results && "Results already initialized!"); | 
 |   NumResults = End - Begin; | 
 |   if (NumResults) { | 
 |     Results = static_cast<DeclAccessPair *>( | 
 |         C.Allocate(sizeof(DeclAccessPair) * NumResults, | 
 |  | 
 |                    alignof(DeclAccessPair))); | 
 |     memcpy(Results, Begin.I, NumResults * sizeof(DeclAccessPair)); | 
 |   } | 
 | } | 
 |  | 
 | CXXRecordDecl *OverloadExpr::getNamingClass() const { | 
 |   if (isa<UnresolvedLookupExpr>(this)) | 
 |     return cast<UnresolvedLookupExpr>(this)->getNamingClass(); | 
 |   else | 
 |     return cast<UnresolvedMemberExpr>(this)->getNamingClass(); | 
 | } | 
 |  | 
 | // DependentScopeDeclRefExpr | 
 | DependentScopeDeclRefExpr::DependentScopeDeclRefExpr(QualType T, | 
 |                             NestedNameSpecifierLoc QualifierLoc, | 
 |                             SourceLocation TemplateKWLoc, | 
 |                             const DeclarationNameInfo &NameInfo, | 
 |                             const TemplateArgumentListInfo *Args) | 
 |   : Expr(DependentScopeDeclRefExprClass, T, VK_LValue, OK_Ordinary, | 
 |          true, true, | 
 |          (NameInfo.isInstantiationDependent() || | 
 |           (QualifierLoc && | 
 |            QualifierLoc.getNestedNameSpecifier()->isInstantiationDependent())), | 
 |          (NameInfo.containsUnexpandedParameterPack() || | 
 |           (QualifierLoc && | 
 |            QualifierLoc.getNestedNameSpecifier() | 
 |                             ->containsUnexpandedParameterPack()))), | 
 |     QualifierLoc(QualifierLoc), NameInfo(NameInfo), | 
 |     HasTemplateKWAndArgsInfo(Args != nullptr || TemplateKWLoc.isValid()) | 
 | { | 
 |   if (Args) { | 
 |     bool Dependent = true; | 
 |     bool InstantiationDependent = true; | 
 |     bool ContainsUnexpandedParameterPack | 
 |       = ExprBits.ContainsUnexpandedParameterPack; | 
 |     getTrailingObjects<ASTTemplateKWAndArgsInfo>()->initializeFrom( | 
 |         TemplateKWLoc, *Args, getTrailingObjects<TemplateArgumentLoc>(), | 
 |         Dependent, InstantiationDependent, ContainsUnexpandedParameterPack); | 
 |     ExprBits.ContainsUnexpandedParameterPack = ContainsUnexpandedParameterPack; | 
 |   } else if (TemplateKWLoc.isValid()) { | 
 |     getTrailingObjects<ASTTemplateKWAndArgsInfo>()->initializeFrom( | 
 |         TemplateKWLoc); | 
 |   } | 
 | } | 
 |  | 
 | DependentScopeDeclRefExpr * | 
 | DependentScopeDeclRefExpr::Create(const ASTContext &C, | 
 |                                   NestedNameSpecifierLoc QualifierLoc, | 
 |                                   SourceLocation TemplateKWLoc, | 
 |                                   const DeclarationNameInfo &NameInfo, | 
 |                                   const TemplateArgumentListInfo *Args) { | 
 |   assert(QualifierLoc && "should be created for dependent qualifiers"); | 
 |   bool HasTemplateKWAndArgsInfo = Args || TemplateKWLoc.isValid(); | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, Args ? Args->size() : 0); | 
 |   void *Mem = C.Allocate(Size); | 
 |   return new (Mem) DependentScopeDeclRefExpr(C.DependentTy, QualifierLoc, | 
 |                                              TemplateKWLoc, NameInfo, Args); | 
 | } | 
 |  | 
 | DependentScopeDeclRefExpr * | 
 | DependentScopeDeclRefExpr::CreateEmpty(const ASTContext &C, | 
 |                                        bool HasTemplateKWAndArgsInfo, | 
 |                                        unsigned NumTemplateArgs) { | 
 |   assert(NumTemplateArgs == 0 || HasTemplateKWAndArgsInfo); | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, NumTemplateArgs); | 
 |   void *Mem = C.Allocate(Size); | 
 |   auto *E = | 
 |       new (Mem) DependentScopeDeclRefExpr(QualType(), NestedNameSpecifierLoc(), | 
 |                                           SourceLocation(), | 
 |                                           DeclarationNameInfo(), nullptr); | 
 |   E->HasTemplateKWAndArgsInfo = HasTemplateKWAndArgsInfo; | 
 |   return E; | 
 | } | 
 |  | 
 | SourceLocation CXXConstructExpr::getBeginLoc() const { | 
 |   if (isa<CXXTemporaryObjectExpr>(this)) | 
 |     return cast<CXXTemporaryObjectExpr>(this)->getLocStart(); | 
 |   return Loc; | 
 | } | 
 |  | 
 | SourceLocation CXXConstructExpr::getEndLoc() const { | 
 |   if (isa<CXXTemporaryObjectExpr>(this)) | 
 |     return cast<CXXTemporaryObjectExpr>(this)->getLocEnd(); | 
 |  | 
 |   if (ParenOrBraceRange.isValid()) | 
 |     return ParenOrBraceRange.getEnd(); | 
 |  | 
 |   SourceLocation End = Loc; | 
 |   for (unsigned I = getNumArgs(); I > 0; --I) { | 
 |     const Expr *Arg = getArg(I-1); | 
 |     if (!Arg->isDefaultArgument()) { | 
 |       SourceLocation NewEnd = Arg->getLocEnd(); | 
 |       if (NewEnd.isValid()) { | 
 |         End = NewEnd; | 
 |         break; | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   return End; | 
 | } | 
 |  | 
 | SourceRange CXXOperatorCallExpr::getSourceRangeImpl() const { | 
 |   OverloadedOperatorKind Kind = getOperator(); | 
 |   if (Kind == OO_PlusPlus || Kind == OO_MinusMinus) { | 
 |     if (getNumArgs() == 1) | 
 |       // Prefix operator | 
 |       return SourceRange(getOperatorLoc(), getArg(0)->getLocEnd()); | 
 |     else | 
 |       // Postfix operator | 
 |       return SourceRange(getArg(0)->getLocStart(), getOperatorLoc()); | 
 |   } else if (Kind == OO_Arrow) { | 
 |     return getArg(0)->getSourceRange(); | 
 |   } else if (Kind == OO_Call) { | 
 |     return SourceRange(getArg(0)->getLocStart(), getRParenLoc()); | 
 |   } else if (Kind == OO_Subscript) { | 
 |     return SourceRange(getArg(0)->getLocStart(), getRParenLoc()); | 
 |   } else if (getNumArgs() == 1) { | 
 |     return SourceRange(getOperatorLoc(), getArg(0)->getLocEnd()); | 
 |   } else if (getNumArgs() == 2) { | 
 |     return SourceRange(getArg(0)->getLocStart(), getArg(1)->getLocEnd()); | 
 |   } else { | 
 |     return getOperatorLoc(); | 
 |   } | 
 | } | 
 |  | 
 | Expr *CXXMemberCallExpr::getImplicitObjectArgument() const { | 
 |   const Expr *Callee = getCallee()->IgnoreParens(); | 
 |   if (const auto *MemExpr = dyn_cast<MemberExpr>(Callee)) | 
 |     return MemExpr->getBase(); | 
 |   if (const auto *BO = dyn_cast<BinaryOperator>(Callee)) | 
 |     if (BO->getOpcode() == BO_PtrMemD || BO->getOpcode() == BO_PtrMemI) | 
 |       return BO->getLHS(); | 
 |  | 
 |   // FIXME: Will eventually need to cope with member pointers. | 
 |   return nullptr; | 
 | } | 
 |  | 
 | CXXMethodDecl *CXXMemberCallExpr::getMethodDecl() const { | 
 |   if (const auto *MemExpr = dyn_cast<MemberExpr>(getCallee()->IgnoreParens())) | 
 |     return cast<CXXMethodDecl>(MemExpr->getMemberDecl()); | 
 |  | 
 |   // FIXME: Will eventually need to cope with member pointers. | 
 |   return nullptr; | 
 | } | 
 |  | 
 | CXXRecordDecl *CXXMemberCallExpr::getRecordDecl() const { | 
 |   Expr* ThisArg = getImplicitObjectArgument(); | 
 |   if (!ThisArg) | 
 |     return nullptr; | 
 |  | 
 |   if (ThisArg->getType()->isAnyPointerType()) | 
 |     return ThisArg->getType()->getPointeeType()->getAsCXXRecordDecl(); | 
 |  | 
 |   return ThisArg->getType()->getAsCXXRecordDecl(); | 
 | } | 
 |  | 
 | //===----------------------------------------------------------------------===// | 
 | //  Named casts | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | /// getCastName - Get the name of the C++ cast being used, e.g., | 
 | /// "static_cast", "dynamic_cast", "reinterpret_cast", or | 
 | /// "const_cast". The returned pointer must not be freed. | 
 | const char *CXXNamedCastExpr::getCastName() const { | 
 |   switch (getStmtClass()) { | 
 |   case CXXStaticCastExprClass:      return "static_cast"; | 
 |   case CXXDynamicCastExprClass:     return "dynamic_cast"; | 
 |   case CXXReinterpretCastExprClass: return "reinterpret_cast"; | 
 |   case CXXConstCastExprClass:       return "const_cast"; | 
 |   default:                          return "<invalid cast>"; | 
 |   } | 
 | } | 
 |  | 
 | CXXStaticCastExpr *CXXStaticCastExpr::Create(const ASTContext &C, QualType T, | 
 |                                              ExprValueKind VK, | 
 |                                              CastKind K, Expr *Op, | 
 |                                              const CXXCastPath *BasePath, | 
 |                                              TypeSourceInfo *WrittenTy, | 
 |                                              SourceLocation L, | 
 |                                              SourceLocation RParenLoc, | 
 |                                              SourceRange AngleBrackets) { | 
 |   unsigned PathSize = (BasePath ? BasePath->size() : 0); | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   auto *E = | 
 |       new (Buffer) CXXStaticCastExpr(T, VK, K, Op, PathSize, WrittenTy, L, | 
 |                                      RParenLoc, AngleBrackets); | 
 |   if (PathSize) | 
 |     std::uninitialized_copy_n(BasePath->data(), BasePath->size(), | 
 |                               E->getTrailingObjects<CXXBaseSpecifier *>()); | 
 |   return E; | 
 | } | 
 |  | 
 | CXXStaticCastExpr *CXXStaticCastExpr::CreateEmpty(const ASTContext &C, | 
 |                                                   unsigned PathSize) { | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   return new (Buffer) CXXStaticCastExpr(EmptyShell(), PathSize); | 
 | } | 
 |  | 
 | CXXDynamicCastExpr *CXXDynamicCastExpr::Create(const ASTContext &C, QualType T, | 
 |                                                ExprValueKind VK, | 
 |                                                CastKind K, Expr *Op, | 
 |                                                const CXXCastPath *BasePath, | 
 |                                                TypeSourceInfo *WrittenTy, | 
 |                                                SourceLocation L, | 
 |                                                SourceLocation RParenLoc, | 
 |                                                SourceRange AngleBrackets) { | 
 |   unsigned PathSize = (BasePath ? BasePath->size() : 0); | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   auto *E = | 
 |       new (Buffer) CXXDynamicCastExpr(T, VK, K, Op, PathSize, WrittenTy, L, | 
 |                                       RParenLoc, AngleBrackets); | 
 |   if (PathSize) | 
 |     std::uninitialized_copy_n(BasePath->data(), BasePath->size(), | 
 |                               E->getTrailingObjects<CXXBaseSpecifier *>()); | 
 |   return E; | 
 | } | 
 |  | 
 | CXXDynamicCastExpr *CXXDynamicCastExpr::CreateEmpty(const ASTContext &C, | 
 |                                                     unsigned PathSize) { | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   return new (Buffer) CXXDynamicCastExpr(EmptyShell(), PathSize); | 
 | } | 
 |  | 
 | /// isAlwaysNull - Return whether the result of the dynamic_cast is proven | 
 | /// to always be null. For example: | 
 | /// | 
 | /// struct A { }; | 
 | /// struct B final : A { }; | 
 | /// struct C { }; | 
 | /// | 
 | /// C *f(B* b) { return dynamic_cast<C*>(b); } | 
 | bool CXXDynamicCastExpr::isAlwaysNull() const | 
 | { | 
 |   QualType SrcType = getSubExpr()->getType(); | 
 |   QualType DestType = getType(); | 
 |  | 
 |   if (const auto *SrcPTy = SrcType->getAs<PointerType>()) { | 
 |     SrcType = SrcPTy->getPointeeType(); | 
 |     DestType = DestType->castAs<PointerType>()->getPointeeType(); | 
 |   } | 
 |  | 
 |   if (DestType->isVoidType()) | 
 |     return false; | 
 |  | 
 |   const auto *SrcRD = | 
 |       cast<CXXRecordDecl>(SrcType->castAs<RecordType>()->getDecl()); | 
 |  | 
 |   if (!SrcRD->hasAttr<FinalAttr>()) | 
 |     return false; | 
 |  | 
 |   const auto *DestRD = | 
 |       cast<CXXRecordDecl>(DestType->castAs<RecordType>()->getDecl()); | 
 |  | 
 |   return !DestRD->isDerivedFrom(SrcRD); | 
 | } | 
 |  | 
 | CXXReinterpretCastExpr * | 
 | CXXReinterpretCastExpr::Create(const ASTContext &C, QualType T, | 
 |                                ExprValueKind VK, CastKind K, Expr *Op, | 
 |                                const CXXCastPath *BasePath, | 
 |                                TypeSourceInfo *WrittenTy, SourceLocation L, | 
 |                                SourceLocation RParenLoc, | 
 |                                SourceRange AngleBrackets) { | 
 |   unsigned PathSize = (BasePath ? BasePath->size() : 0); | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   auto *E = | 
 |       new (Buffer) CXXReinterpretCastExpr(T, VK, K, Op, PathSize, WrittenTy, L, | 
 |                                           RParenLoc, AngleBrackets); | 
 |   if (PathSize) | 
 |     std::uninitialized_copy_n(BasePath->data(), BasePath->size(), | 
 |                               E->getTrailingObjects<CXXBaseSpecifier *>()); | 
 |   return E; | 
 | } | 
 |  | 
 | CXXReinterpretCastExpr * | 
 | CXXReinterpretCastExpr::CreateEmpty(const ASTContext &C, unsigned PathSize) { | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   return new (Buffer) CXXReinterpretCastExpr(EmptyShell(), PathSize); | 
 | } | 
 |  | 
 | CXXConstCastExpr *CXXConstCastExpr::Create(const ASTContext &C, QualType T, | 
 |                                            ExprValueKind VK, Expr *Op, | 
 |                                            TypeSourceInfo *WrittenTy, | 
 |                                            SourceLocation L, | 
 |                                            SourceLocation RParenLoc, | 
 |                                            SourceRange AngleBrackets) { | 
 |   return new (C) CXXConstCastExpr(T, VK, Op, WrittenTy, L, RParenLoc, AngleBrackets); | 
 | } | 
 |  | 
 | CXXConstCastExpr *CXXConstCastExpr::CreateEmpty(const ASTContext &C) { | 
 |   return new (C) CXXConstCastExpr(EmptyShell()); | 
 | } | 
 |  | 
 | CXXFunctionalCastExpr * | 
 | CXXFunctionalCastExpr::Create(const ASTContext &C, QualType T, ExprValueKind VK, | 
 |                               TypeSourceInfo *Written, CastKind K, Expr *Op, | 
 |                               const CXXCastPath *BasePath, | 
 |                               SourceLocation L, SourceLocation R) { | 
 |   unsigned PathSize = (BasePath ? BasePath->size() : 0); | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   auto *E = | 
 |       new (Buffer) CXXFunctionalCastExpr(T, VK, Written, K, Op, PathSize, L, R); | 
 |   if (PathSize) | 
 |     std::uninitialized_copy_n(BasePath->data(), BasePath->size(), | 
 |                               E->getTrailingObjects<CXXBaseSpecifier *>()); | 
 |   return E; | 
 | } | 
 |  | 
 | CXXFunctionalCastExpr * | 
 | CXXFunctionalCastExpr::CreateEmpty(const ASTContext &C, unsigned PathSize) { | 
 |   void *Buffer = | 
 |       C.Allocate(totalSizeToAlloc<CastExpr::BasePathSizeTy, CXXBaseSpecifier *>( | 
 |           PathSize ? 1 : 0, PathSize)); | 
 |   return new (Buffer) CXXFunctionalCastExpr(EmptyShell(), PathSize); | 
 | } | 
 |  | 
 | SourceLocation CXXFunctionalCastExpr::getBeginLoc() const { | 
 |   return getTypeInfoAsWritten()->getTypeLoc().getLocStart(); | 
 | } | 
 |  | 
 | SourceLocation CXXFunctionalCastExpr::getEndLoc() const { | 
 |   return RParenLoc.isValid() ? RParenLoc : getSubExpr()->getLocEnd(); | 
 | } | 
 |  | 
 | UserDefinedLiteral::LiteralOperatorKind | 
 | UserDefinedLiteral::getLiteralOperatorKind() const { | 
 |   if (getNumArgs() == 0) | 
 |     return LOK_Template; | 
 |   if (getNumArgs() == 2) | 
 |     return LOK_String; | 
 |  | 
 |   assert(getNumArgs() == 1 && "unexpected #args in literal operator call"); | 
 |   QualType ParamTy = | 
 |     cast<FunctionDecl>(getCalleeDecl())->getParamDecl(0)->getType(); | 
 |   if (ParamTy->isPointerType()) | 
 |     return LOK_Raw; | 
 |   if (ParamTy->isAnyCharacterType()) | 
 |     return LOK_Character; | 
 |   if (ParamTy->isIntegerType()) | 
 |     return LOK_Integer; | 
 |   if (ParamTy->isFloatingType()) | 
 |     return LOK_Floating; | 
 |  | 
 |   llvm_unreachable("unknown kind of literal operator"); | 
 | } | 
 |  | 
 | Expr *UserDefinedLiteral::getCookedLiteral() { | 
 | #ifndef NDEBUG | 
 |   LiteralOperatorKind LOK = getLiteralOperatorKind(); | 
 |   assert(LOK != LOK_Template && LOK != LOK_Raw && "not a cooked literal"); | 
 | #endif | 
 |   return getArg(0); | 
 | } | 
 |  | 
 | const IdentifierInfo *UserDefinedLiteral::getUDSuffix() const { | 
 |   return cast<FunctionDecl>(getCalleeDecl())->getLiteralIdentifier(); | 
 | } | 
 |  | 
 | CXXDefaultInitExpr::CXXDefaultInitExpr(const ASTContext &C, SourceLocation Loc, | 
 |                                        FieldDecl *Field, QualType T) | 
 |     : Expr(CXXDefaultInitExprClass, T.getNonLValueExprType(C), | 
 |            T->isLValueReferenceType() ? VK_LValue : T->isRValueReferenceType() | 
 |                                                         ? VK_XValue | 
 |                                                         : VK_RValue, | 
 |            /*FIXME*/ OK_Ordinary, false, false, false, false), | 
 |       Field(Field), Loc(Loc) { | 
 |   assert(Field->hasInClassInitializer()); | 
 | } | 
 |  | 
 | CXXTemporary *CXXTemporary::Create(const ASTContext &C, | 
 |                                    const CXXDestructorDecl *Destructor) { | 
 |   return new (C) CXXTemporary(Destructor); | 
 | } | 
 |  | 
 | CXXBindTemporaryExpr *CXXBindTemporaryExpr::Create(const ASTContext &C, | 
 |                                                    CXXTemporary *Temp, | 
 |                                                    Expr* SubExpr) { | 
 |   assert((SubExpr->getType()->isRecordType() || | 
 |           SubExpr->getType()->isArrayType()) && | 
 |          "Expression bound to a temporary must have record or array type!"); | 
 |  | 
 |   return new (C) CXXBindTemporaryExpr(Temp, SubExpr); | 
 | } | 
 |  | 
 | CXXTemporaryObjectExpr::CXXTemporaryObjectExpr(const ASTContext &C, | 
 |                                                CXXConstructorDecl *Cons, | 
 |                                                QualType Type, | 
 |                                                TypeSourceInfo *TSI, | 
 |                                                ArrayRef<Expr*> Args, | 
 |                                                SourceRange ParenOrBraceRange, | 
 |                                                bool HadMultipleCandidates, | 
 |                                                bool ListInitialization, | 
 |                                                bool StdInitListInitialization, | 
 |                                                bool ZeroInitialization) | 
 |     : CXXConstructExpr(C, CXXTemporaryObjectExprClass, Type, | 
 |                        TSI->getTypeLoc().getBeginLoc(), Cons, false, Args, | 
 |                        HadMultipleCandidates, ListInitialization, | 
 |                        StdInitListInitialization,  ZeroInitialization, | 
 |                        CXXConstructExpr::CK_Complete, ParenOrBraceRange), | 
 |       Type(TSI) {} | 
 |  | 
 | SourceLocation CXXTemporaryObjectExpr::getBeginLoc() const { | 
 |   return Type->getTypeLoc().getBeginLoc(); | 
 | } | 
 |  | 
 | SourceLocation CXXTemporaryObjectExpr::getEndLoc() const { | 
 |   SourceLocation Loc = getParenOrBraceRange().getEnd(); | 
 |   if (Loc.isInvalid() && getNumArgs()) | 
 |     Loc = getArg(getNumArgs()-1)->getLocEnd(); | 
 |   return Loc; | 
 | } | 
 |  | 
 | CXXConstructExpr *CXXConstructExpr::Create(const ASTContext &C, QualType T, | 
 |                                            SourceLocation Loc, | 
 |                                            CXXConstructorDecl *Ctor, | 
 |                                            bool Elidable, | 
 |                                            ArrayRef<Expr*> Args, | 
 |                                            bool HadMultipleCandidates, | 
 |                                            bool ListInitialization, | 
 |                                            bool StdInitListInitialization, | 
 |                                            bool ZeroInitialization, | 
 |                                            ConstructionKind ConstructKind, | 
 |                                            SourceRange ParenOrBraceRange) { | 
 |   return new (C) CXXConstructExpr(C, CXXConstructExprClass, T, Loc, | 
 |                                   Ctor, Elidable, Args, | 
 |                                   HadMultipleCandidates, ListInitialization, | 
 |                                   StdInitListInitialization, | 
 |                                   ZeroInitialization, ConstructKind, | 
 |                                   ParenOrBraceRange); | 
 | } | 
 |  | 
 | CXXConstructExpr::CXXConstructExpr(const ASTContext &C, StmtClass SC, | 
 |                                    QualType T, SourceLocation Loc, | 
 |                                    CXXConstructorDecl *Ctor, | 
 |                                    bool Elidable, | 
 |                                    ArrayRef<Expr*> Args, | 
 |                                    bool HadMultipleCandidates, | 
 |                                    bool ListInitialization, | 
 |                                    bool StdInitListInitialization, | 
 |                                    bool ZeroInitialization, | 
 |                                    ConstructionKind ConstructKind, | 
 |                                    SourceRange ParenOrBraceRange) | 
 |     : Expr(SC, T, VK_RValue, OK_Ordinary, | 
 |            T->isDependentType(), T->isDependentType(), | 
 |            T->isInstantiationDependentType(), | 
 |            T->containsUnexpandedParameterPack()), | 
 |       Constructor(Ctor), Loc(Loc), ParenOrBraceRange(ParenOrBraceRange), | 
 |       NumArgs(Args.size()), Elidable(Elidable), | 
 |       HadMultipleCandidates(HadMultipleCandidates), | 
 |       ListInitialization(ListInitialization), | 
 |       StdInitListInitialization(StdInitListInitialization), | 
 |       ZeroInitialization(ZeroInitialization), ConstructKind(ConstructKind) { | 
 |   if (NumArgs) { | 
 |     this->Args = new (C) Stmt*[Args.size()]; | 
 |  | 
 |     for (unsigned i = 0; i != Args.size(); ++i) { | 
 |       assert(Args[i] && "NULL argument in CXXConstructExpr"); | 
 |  | 
 |       if (Args[i]->isValueDependent()) | 
 |         ExprBits.ValueDependent = true; | 
 |       if (Args[i]->isInstantiationDependent()) | 
 |         ExprBits.InstantiationDependent = true; | 
 |       if (Args[i]->containsUnexpandedParameterPack()) | 
 |         ExprBits.ContainsUnexpandedParameterPack = true; | 
 |  | 
 |       this->Args[i] = Args[i]; | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | LambdaCapture::LambdaCapture(SourceLocation Loc, bool Implicit, | 
 |                              LambdaCaptureKind Kind, VarDecl *Var, | 
 |                              SourceLocation EllipsisLoc) | 
 |     : DeclAndBits(Var, 0), Loc(Loc), EllipsisLoc(EllipsisLoc) { | 
 |   unsigned Bits = 0; | 
 |   if (Implicit) | 
 |     Bits |= Capture_Implicit; | 
 |  | 
 |   switch (Kind) { | 
 |   case LCK_StarThis: | 
 |     Bits |= Capture_ByCopy; | 
 |     LLVM_FALLTHROUGH; | 
 |   case LCK_This: | 
 |     assert(!Var && "'this' capture cannot have a variable!"); | 
 |     Bits |= Capture_This; | 
 |     break; | 
 |  | 
 |   case LCK_ByCopy: | 
 |     Bits |= Capture_ByCopy; | 
 |     LLVM_FALLTHROUGH; | 
 |   case LCK_ByRef: | 
 |     assert(Var && "capture must have a variable!"); | 
 |     break; | 
 |   case LCK_VLAType: | 
 |     assert(!Var && "VLA type capture cannot have a variable!"); | 
 |     break; | 
 |   } | 
 |   DeclAndBits.setInt(Bits); | 
 | } | 
 |  | 
 | LambdaCaptureKind LambdaCapture::getCaptureKind() const { | 
 |   if (capturesVLAType()) | 
 |     return LCK_VLAType; | 
 |   bool CapByCopy = DeclAndBits.getInt() & Capture_ByCopy; | 
 |   if (capturesThis()) | 
 |     return CapByCopy ? LCK_StarThis : LCK_This; | 
 |   return CapByCopy ? LCK_ByCopy : LCK_ByRef; | 
 | } | 
 |  | 
 | LambdaExpr::LambdaExpr(QualType T, SourceRange IntroducerRange, | 
 |                        LambdaCaptureDefault CaptureDefault, | 
 |                        SourceLocation CaptureDefaultLoc, | 
 |                        ArrayRef<LambdaCapture> Captures, bool ExplicitParams, | 
 |                        bool ExplicitResultType, ArrayRef<Expr *> CaptureInits, | 
 |                        SourceLocation ClosingBrace, | 
 |                        bool ContainsUnexpandedParameterPack) | 
 |     : Expr(LambdaExprClass, T, VK_RValue, OK_Ordinary, T->isDependentType(), | 
 |            T->isDependentType(), T->isDependentType(), | 
 |            ContainsUnexpandedParameterPack), | 
 |       IntroducerRange(IntroducerRange), CaptureDefaultLoc(CaptureDefaultLoc), | 
 |       NumCaptures(Captures.size()), CaptureDefault(CaptureDefault), | 
 |       ExplicitParams(ExplicitParams), ExplicitResultType(ExplicitResultType), | 
 |       ClosingBrace(ClosingBrace) { | 
 |   assert(CaptureInits.size() == Captures.size() && "Wrong number of arguments"); | 
 |   CXXRecordDecl *Class = getLambdaClass(); | 
 |   CXXRecordDecl::LambdaDefinitionData &Data = Class->getLambdaData(); | 
 |  | 
 |   // FIXME: Propagate "has unexpanded parameter pack" bit. | 
 |  | 
 |   // Copy captures. | 
 |   const ASTContext &Context = Class->getASTContext(); | 
 |   Data.NumCaptures = NumCaptures; | 
 |   Data.NumExplicitCaptures = 0; | 
 |   Data.Captures = | 
 |       (LambdaCapture *)Context.Allocate(sizeof(LambdaCapture) * NumCaptures); | 
 |   LambdaCapture *ToCapture = Data.Captures; | 
 |   for (unsigned I = 0, N = Captures.size(); I != N; ++I) { | 
 |     if (Captures[I].isExplicit()) | 
 |       ++Data.NumExplicitCaptures; | 
 |  | 
 |     *ToCapture++ = Captures[I]; | 
 |   } | 
 |  | 
 |   // Copy initialization expressions for the non-static data members. | 
 |   Stmt **Stored = getStoredStmts(); | 
 |   for (unsigned I = 0, N = CaptureInits.size(); I != N; ++I) | 
 |     *Stored++ = CaptureInits[I]; | 
 |  | 
 |   // Copy the body of the lambda. | 
 |   *Stored++ = getCallOperator()->getBody(); | 
 | } | 
 |  | 
 | LambdaExpr *LambdaExpr::Create( | 
 |     const ASTContext &Context, CXXRecordDecl *Class, | 
 |     SourceRange IntroducerRange, LambdaCaptureDefault CaptureDefault, | 
 |     SourceLocation CaptureDefaultLoc, ArrayRef<LambdaCapture> Captures, | 
 |     bool ExplicitParams, bool ExplicitResultType, ArrayRef<Expr *> CaptureInits, | 
 |     SourceLocation ClosingBrace, bool ContainsUnexpandedParameterPack) { | 
 |   // Determine the type of the expression (i.e., the type of the | 
 |   // function object we're creating). | 
 |   QualType T = Context.getTypeDeclType(Class); | 
 |  | 
 |   unsigned Size = totalSizeToAlloc<Stmt *>(Captures.size() + 1); | 
 |   void *Mem = Context.Allocate(Size); | 
 |   return new (Mem) | 
 |       LambdaExpr(T, IntroducerRange, CaptureDefault, CaptureDefaultLoc, | 
 |                  Captures, ExplicitParams, ExplicitResultType, CaptureInits, | 
 |                  ClosingBrace, ContainsUnexpandedParameterPack); | 
 | } | 
 |  | 
 | LambdaExpr *LambdaExpr::CreateDeserialized(const ASTContext &C, | 
 |                                            unsigned NumCaptures) { | 
 |   unsigned Size = totalSizeToAlloc<Stmt *>(NumCaptures + 1); | 
 |   void *Mem = C.Allocate(Size); | 
 |   return new (Mem) LambdaExpr(EmptyShell(), NumCaptures); | 
 | } | 
 |  | 
 | bool LambdaExpr::isInitCapture(const LambdaCapture *C) const { | 
 |   return (C->capturesVariable() && C->getCapturedVar()->isInitCapture() && | 
 |           (getCallOperator() == C->getCapturedVar()->getDeclContext())); | 
 | } | 
 |  | 
 | LambdaExpr::capture_iterator LambdaExpr::capture_begin() const { | 
 |   return getLambdaClass()->getLambdaData().Captures; | 
 | } | 
 |  | 
 | LambdaExpr::capture_iterator LambdaExpr::capture_end() const { | 
 |   return capture_begin() + NumCaptures; | 
 | } | 
 |  | 
 | LambdaExpr::capture_range LambdaExpr::captures() const { | 
 |   return capture_range(capture_begin(), capture_end()); | 
 | } | 
 |  | 
 | LambdaExpr::capture_iterator LambdaExpr::explicit_capture_begin() const { | 
 |   return capture_begin(); | 
 | } | 
 |  | 
 | LambdaExpr::capture_iterator LambdaExpr::explicit_capture_end() const { | 
 |   struct CXXRecordDecl::LambdaDefinitionData &Data | 
 |     = getLambdaClass()->getLambdaData(); | 
 |   return Data.Captures + Data.NumExplicitCaptures; | 
 | } | 
 |  | 
 | LambdaExpr::capture_range LambdaExpr::explicit_captures() const { | 
 |   return capture_range(explicit_capture_begin(), explicit_capture_end()); | 
 | } | 
 |  | 
 | LambdaExpr::capture_iterator LambdaExpr::implicit_capture_begin() const { | 
 |   return explicit_capture_end(); | 
 | } | 
 |  | 
 | LambdaExpr::capture_iterator LambdaExpr::implicit_capture_end() const { | 
 |   return capture_end(); | 
 | } | 
 |  | 
 | LambdaExpr::capture_range LambdaExpr::implicit_captures() const { | 
 |   return capture_range(implicit_capture_begin(), implicit_capture_end()); | 
 | } | 
 |  | 
 | CXXRecordDecl *LambdaExpr::getLambdaClass() const { | 
 |   return getType()->getAsCXXRecordDecl(); | 
 | } | 
 |  | 
 | CXXMethodDecl *LambdaExpr::getCallOperator() const { | 
 |   CXXRecordDecl *Record = getLambdaClass(); | 
 |   return Record->getLambdaCallOperator(); | 
 | } | 
 |  | 
 | TemplateParameterList *LambdaExpr::getTemplateParameterList() const { | 
 |   CXXRecordDecl *Record = getLambdaClass(); | 
 |   return Record->getGenericLambdaTemplateParameterList(); | 
 |  | 
 | } | 
 |  | 
 | CompoundStmt *LambdaExpr::getBody() const { | 
 |   // FIXME: this mutation in getBody is bogus. It should be | 
 |   // initialized in ASTStmtReader::VisitLambdaExpr, but for reasons I | 
 |   // don't understand, that doesn't work. | 
 |   if (!getStoredStmts()[NumCaptures]) | 
 |     *const_cast<Stmt **>(&getStoredStmts()[NumCaptures]) = | 
 |         getCallOperator()->getBody(); | 
 |  | 
 |   return static_cast<CompoundStmt *>(getStoredStmts()[NumCaptures]); | 
 | } | 
 |  | 
 | bool LambdaExpr::isMutable() const { | 
 |   return !getCallOperator()->isConst(); | 
 | } | 
 |  | 
 | ExprWithCleanups::ExprWithCleanups(Expr *subexpr, | 
 |                                    bool CleanupsHaveSideEffects, | 
 |                                    ArrayRef<CleanupObject> objects) | 
 |     : Expr(ExprWithCleanupsClass, subexpr->getType(), | 
 |            subexpr->getValueKind(), subexpr->getObjectKind(), | 
 |            subexpr->isTypeDependent(), subexpr->isValueDependent(), | 
 |            subexpr->isInstantiationDependent(), | 
 |            subexpr->containsUnexpandedParameterPack()), | 
 |       SubExpr(subexpr) { | 
 |   ExprWithCleanupsBits.CleanupsHaveSideEffects = CleanupsHaveSideEffects; | 
 |   ExprWithCleanupsBits.NumObjects = objects.size(); | 
 |   for (unsigned i = 0, e = objects.size(); i != e; ++i) | 
 |     getTrailingObjects<CleanupObject>()[i] = objects[i]; | 
 | } | 
 |  | 
 | ExprWithCleanups *ExprWithCleanups::Create(const ASTContext &C, Expr *subexpr, | 
 |                                            bool CleanupsHaveSideEffects, | 
 |                                            ArrayRef<CleanupObject> objects) { | 
 |   void *buffer = C.Allocate(totalSizeToAlloc<CleanupObject>(objects.size()), | 
 |                             alignof(ExprWithCleanups)); | 
 |   return new (buffer) | 
 |       ExprWithCleanups(subexpr, CleanupsHaveSideEffects, objects); | 
 | } | 
 |  | 
 | ExprWithCleanups::ExprWithCleanups(EmptyShell empty, unsigned numObjects) | 
 |     : Expr(ExprWithCleanupsClass, empty) { | 
 |   ExprWithCleanupsBits.NumObjects = numObjects; | 
 | } | 
 |  | 
 | ExprWithCleanups *ExprWithCleanups::Create(const ASTContext &C, | 
 |                                            EmptyShell empty, | 
 |                                            unsigned numObjects) { | 
 |   void *buffer = C.Allocate(totalSizeToAlloc<CleanupObject>(numObjects), | 
 |                             alignof(ExprWithCleanups)); | 
 |   return new (buffer) ExprWithCleanups(empty, numObjects); | 
 | } | 
 |  | 
 | CXXUnresolvedConstructExpr::CXXUnresolvedConstructExpr(TypeSourceInfo *Type, | 
 |                                                  SourceLocation LParenLoc, | 
 |                                                  ArrayRef<Expr*> Args, | 
 |                                                  SourceLocation RParenLoc) | 
 |     : Expr(CXXUnresolvedConstructExprClass, | 
 |            Type->getType().getNonReferenceType(), | 
 |            (Type->getType()->isLValueReferenceType() | 
 |                 ? VK_LValue | 
 |                 : Type->getType()->isRValueReferenceType() ? VK_XValue | 
 |                                                            : VK_RValue), | 
 |            OK_Ordinary, | 
 |            Type->getType()->isDependentType() || | 
 |                Type->getType()->getContainedDeducedType(), | 
 |            true, true, Type->getType()->containsUnexpandedParameterPack()), | 
 |       Type(Type), LParenLoc(LParenLoc), RParenLoc(RParenLoc), | 
 |       NumArgs(Args.size()) { | 
 |   auto **StoredArgs = getTrailingObjects<Expr *>(); | 
 |   for (unsigned I = 0; I != Args.size(); ++I) { | 
 |     if (Args[I]->containsUnexpandedParameterPack()) | 
 |       ExprBits.ContainsUnexpandedParameterPack = true; | 
 |  | 
 |     StoredArgs[I] = Args[I]; | 
 |   } | 
 | } | 
 |  | 
 | CXXUnresolvedConstructExpr * | 
 | CXXUnresolvedConstructExpr::Create(const ASTContext &C, | 
 |                                    TypeSourceInfo *Type, | 
 |                                    SourceLocation LParenLoc, | 
 |                                    ArrayRef<Expr*> Args, | 
 |                                    SourceLocation RParenLoc) { | 
 |   void *Mem = C.Allocate(totalSizeToAlloc<Expr *>(Args.size())); | 
 |   return new (Mem) CXXUnresolvedConstructExpr(Type, LParenLoc, Args, RParenLoc); | 
 | } | 
 |  | 
 | CXXUnresolvedConstructExpr * | 
 | CXXUnresolvedConstructExpr::CreateEmpty(const ASTContext &C, unsigned NumArgs) { | 
 |   Stmt::EmptyShell Empty; | 
 |   void *Mem = C.Allocate(totalSizeToAlloc<Expr *>(NumArgs)); | 
 |   return new (Mem) CXXUnresolvedConstructExpr(Empty, NumArgs); | 
 | } | 
 |  | 
 | SourceLocation CXXUnresolvedConstructExpr::getBeginLoc() const { | 
 |   return Type->getTypeLoc().getBeginLoc(); | 
 | } | 
 |  | 
 | CXXDependentScopeMemberExpr::CXXDependentScopeMemberExpr( | 
 |     const ASTContext &C, Expr *Base, QualType BaseType, bool IsArrow, | 
 |     SourceLocation OperatorLoc, NestedNameSpecifierLoc QualifierLoc, | 
 |     SourceLocation TemplateKWLoc, NamedDecl *FirstQualifierFoundInScope, | 
 |     DeclarationNameInfo MemberNameInfo, | 
 |     const TemplateArgumentListInfo *TemplateArgs) | 
 |     : Expr(CXXDependentScopeMemberExprClass, C.DependentTy, VK_LValue, | 
 |            OK_Ordinary, true, true, true, | 
 |            ((Base && Base->containsUnexpandedParameterPack()) || | 
 |             (QualifierLoc && | 
 |              QualifierLoc.getNestedNameSpecifier() | 
 |                  ->containsUnexpandedParameterPack()) || | 
 |             MemberNameInfo.containsUnexpandedParameterPack())), | 
 |       Base(Base), BaseType(BaseType), IsArrow(IsArrow), | 
 |       HasTemplateKWAndArgsInfo(TemplateArgs != nullptr || | 
 |                                TemplateKWLoc.isValid()), | 
 |       OperatorLoc(OperatorLoc), QualifierLoc(QualifierLoc), | 
 |       FirstQualifierFoundInScope(FirstQualifierFoundInScope), | 
 |       MemberNameInfo(MemberNameInfo) { | 
 |   if (TemplateArgs) { | 
 |     bool Dependent = true; | 
 |     bool InstantiationDependent = true; | 
 |     bool ContainsUnexpandedParameterPack = false; | 
 |     getTrailingObjects<ASTTemplateKWAndArgsInfo>()->initializeFrom( | 
 |         TemplateKWLoc, *TemplateArgs, getTrailingObjects<TemplateArgumentLoc>(), | 
 |         Dependent, InstantiationDependent, ContainsUnexpandedParameterPack); | 
 |     if (ContainsUnexpandedParameterPack) | 
 |       ExprBits.ContainsUnexpandedParameterPack = true; | 
 |   } else if (TemplateKWLoc.isValid()) { | 
 |     getTrailingObjects<ASTTemplateKWAndArgsInfo>()->initializeFrom( | 
 |         TemplateKWLoc); | 
 |   } | 
 | } | 
 |  | 
 | CXXDependentScopeMemberExpr * | 
 | CXXDependentScopeMemberExpr::Create(const ASTContext &C, | 
 |                                 Expr *Base, QualType BaseType, bool IsArrow, | 
 |                                 SourceLocation OperatorLoc, | 
 |                                 NestedNameSpecifierLoc QualifierLoc, | 
 |                                 SourceLocation TemplateKWLoc, | 
 |                                 NamedDecl *FirstQualifierFoundInScope, | 
 |                                 DeclarationNameInfo MemberNameInfo, | 
 |                                 const TemplateArgumentListInfo *TemplateArgs) { | 
 |   bool HasTemplateKWAndArgsInfo = TemplateArgs || TemplateKWLoc.isValid(); | 
 |   unsigned NumTemplateArgs = TemplateArgs ? TemplateArgs->size() : 0; | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, NumTemplateArgs); | 
 |  | 
 |   void *Mem = C.Allocate(Size, alignof(CXXDependentScopeMemberExpr)); | 
 |   return new (Mem) CXXDependentScopeMemberExpr(C, Base, BaseType, | 
 |                                                IsArrow, OperatorLoc, | 
 |                                                QualifierLoc, | 
 |                                                TemplateKWLoc, | 
 |                                                FirstQualifierFoundInScope, | 
 |                                                MemberNameInfo, TemplateArgs); | 
 | } | 
 |  | 
 | CXXDependentScopeMemberExpr * | 
 | CXXDependentScopeMemberExpr::CreateEmpty(const ASTContext &C, | 
 |                                          bool HasTemplateKWAndArgsInfo, | 
 |                                          unsigned NumTemplateArgs) { | 
 |   assert(NumTemplateArgs == 0 || HasTemplateKWAndArgsInfo); | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, NumTemplateArgs); | 
 |   void *Mem = C.Allocate(Size, alignof(CXXDependentScopeMemberExpr)); | 
 |   auto *E = | 
 |       new (Mem) CXXDependentScopeMemberExpr(C, nullptr, QualType(), | 
 |                                             false, SourceLocation(), | 
 |                                             NestedNameSpecifierLoc(), | 
 |                                             SourceLocation(), nullptr, | 
 |                                             DeclarationNameInfo(), nullptr); | 
 |   E->HasTemplateKWAndArgsInfo = HasTemplateKWAndArgsInfo; | 
 |   return E; | 
 | } | 
 |  | 
 | bool CXXDependentScopeMemberExpr::isImplicitAccess() const { | 
 |   if (!Base) | 
 |     return true; | 
 |  | 
 |   return cast<Expr>(Base)->isImplicitCXXThis(); | 
 | } | 
 |  | 
 | static bool hasOnlyNonStaticMemberFunctions(UnresolvedSetIterator begin, | 
 |                                             UnresolvedSetIterator end) { | 
 |   do { | 
 |     NamedDecl *decl = *begin; | 
 |     if (isa<UnresolvedUsingValueDecl>(decl)) | 
 |       return false; | 
 |  | 
 |     // Unresolved member expressions should only contain methods and | 
 |     // method templates. | 
 |     if (cast<CXXMethodDecl>(decl->getUnderlyingDecl()->getAsFunction()) | 
 |             ->isStatic()) | 
 |       return false; | 
 |   } while (++begin != end); | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | UnresolvedMemberExpr::UnresolvedMemberExpr(const ASTContext &C, | 
 |                                            bool HasUnresolvedUsing, | 
 |                                            Expr *Base, QualType BaseType, | 
 |                                            bool IsArrow, | 
 |                                            SourceLocation OperatorLoc, | 
 |                                            NestedNameSpecifierLoc QualifierLoc, | 
 |                                            SourceLocation TemplateKWLoc, | 
 |                                    const DeclarationNameInfo &MemberNameInfo, | 
 |                                    const TemplateArgumentListInfo *TemplateArgs, | 
 |                                            UnresolvedSetIterator Begin, | 
 |                                            UnresolvedSetIterator End) | 
 |     : OverloadExpr( | 
 |           UnresolvedMemberExprClass, C, QualifierLoc, TemplateKWLoc, | 
 |           MemberNameInfo, TemplateArgs, Begin, End, | 
 |           // Dependent | 
 |           ((Base && Base->isTypeDependent()) || BaseType->isDependentType()), | 
 |           ((Base && Base->isInstantiationDependent()) || | 
 |            BaseType->isInstantiationDependentType()), | 
 |           // Contains unexpanded parameter pack | 
 |           ((Base && Base->containsUnexpandedParameterPack()) || | 
 |            BaseType->containsUnexpandedParameterPack())), | 
 |       IsArrow(IsArrow), HasUnresolvedUsing(HasUnresolvedUsing), Base(Base), | 
 |       BaseType(BaseType), OperatorLoc(OperatorLoc) { | 
 |   // Check whether all of the members are non-static member functions, | 
 |   // and if so, mark give this bound-member type instead of overload type. | 
 |   if (hasOnlyNonStaticMemberFunctions(Begin, End)) | 
 |     setType(C.BoundMemberTy); | 
 | } | 
 |  | 
 | bool UnresolvedMemberExpr::isImplicitAccess() const { | 
 |   if (!Base) | 
 |     return true; | 
 |  | 
 |   return cast<Expr>(Base)->isImplicitCXXThis(); | 
 | } | 
 |  | 
 | UnresolvedMemberExpr *UnresolvedMemberExpr::Create( | 
 |     const ASTContext &C, bool HasUnresolvedUsing, Expr *Base, QualType BaseType, | 
 |     bool IsArrow, SourceLocation OperatorLoc, | 
 |     NestedNameSpecifierLoc QualifierLoc, SourceLocation TemplateKWLoc, | 
 |     const DeclarationNameInfo &MemberNameInfo, | 
 |     const TemplateArgumentListInfo *TemplateArgs, UnresolvedSetIterator Begin, | 
 |     UnresolvedSetIterator End) { | 
 |   bool HasTemplateKWAndArgsInfo = TemplateArgs || TemplateKWLoc.isValid(); | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, TemplateArgs ? TemplateArgs->size() : 0); | 
 |  | 
 |   void *Mem = C.Allocate(Size, alignof(UnresolvedMemberExpr)); | 
 |   return new (Mem) UnresolvedMemberExpr( | 
 |       C, HasUnresolvedUsing, Base, BaseType, IsArrow, OperatorLoc, QualifierLoc, | 
 |       TemplateKWLoc, MemberNameInfo, TemplateArgs, Begin, End); | 
 | } | 
 |  | 
 | UnresolvedMemberExpr * | 
 | UnresolvedMemberExpr::CreateEmpty(const ASTContext &C, | 
 |                                   bool HasTemplateKWAndArgsInfo, | 
 |                                   unsigned NumTemplateArgs) { | 
 |   assert(NumTemplateArgs == 0 || HasTemplateKWAndArgsInfo); | 
 |   std::size_t Size = | 
 |       totalSizeToAlloc<ASTTemplateKWAndArgsInfo, TemplateArgumentLoc>( | 
 |           HasTemplateKWAndArgsInfo, NumTemplateArgs); | 
 |  | 
 |   void *Mem = C.Allocate(Size, alignof(UnresolvedMemberExpr)); | 
 |   auto *E = new (Mem) UnresolvedMemberExpr(EmptyShell()); | 
 |   E->HasTemplateKWAndArgsInfo = HasTemplateKWAndArgsInfo; | 
 |   return E; | 
 | } | 
 |  | 
 | CXXRecordDecl *UnresolvedMemberExpr::getNamingClass() const { | 
 |   // Unlike for UnresolvedLookupExpr, it is very easy to re-derive this. | 
 |  | 
 |   // If there was a nested name specifier, it names the naming class. | 
 |   // It can't be dependent: after all, we were actually able to do the | 
 |   // lookup. | 
 |   CXXRecordDecl *Record = nullptr; | 
 |   auto *NNS = getQualifier(); | 
 |   if (NNS && NNS->getKind() != NestedNameSpecifier::Super) { | 
 |     const Type *T = getQualifier()->getAsType(); | 
 |     assert(T && "qualifier in member expression does not name type"); | 
 |     Record = T->getAsCXXRecordDecl(); | 
 |     assert(Record && "qualifier in member expression does not name record"); | 
 |   } | 
 |   // Otherwise the naming class must have been the base class. | 
 |   else { | 
 |     QualType BaseType = getBaseType().getNonReferenceType(); | 
 |     if (isArrow()) { | 
 |       const auto *PT = BaseType->getAs<PointerType>(); | 
 |       assert(PT && "base of arrow member access is not pointer"); | 
 |       BaseType = PT->getPointeeType(); | 
 |     } | 
 |  | 
 |     Record = BaseType->getAsCXXRecordDecl(); | 
 |     assert(Record && "base of member expression does not name record"); | 
 |   } | 
 |  | 
 |   return Record; | 
 | } | 
 |  | 
 | SizeOfPackExpr * | 
 | SizeOfPackExpr::Create(ASTContext &Context, SourceLocation OperatorLoc, | 
 |                        NamedDecl *Pack, SourceLocation PackLoc, | 
 |                        SourceLocation RParenLoc, | 
 |                        Optional<unsigned> Length, | 
 |                        ArrayRef<TemplateArgument> PartialArgs) { | 
 |   void *Storage = | 
 |       Context.Allocate(totalSizeToAlloc<TemplateArgument>(PartialArgs.size())); | 
 |   return new (Storage) SizeOfPackExpr(Context.getSizeType(), OperatorLoc, Pack, | 
 |                                       PackLoc, RParenLoc, Length, PartialArgs); | 
 | } | 
 |  | 
 | SizeOfPackExpr *SizeOfPackExpr::CreateDeserialized(ASTContext &Context, | 
 |                                                    unsigned NumPartialArgs) { | 
 |   void *Storage = | 
 |       Context.Allocate(totalSizeToAlloc<TemplateArgument>(NumPartialArgs)); | 
 |   return new (Storage) SizeOfPackExpr(EmptyShell(), NumPartialArgs); | 
 | } | 
 |  | 
 | SubstNonTypeTemplateParmPackExpr:: | 
 | SubstNonTypeTemplateParmPackExpr(QualType T, | 
 |                                  ExprValueKind ValueKind, | 
 |                                  NonTypeTemplateParmDecl *Param, | 
 |                                  SourceLocation NameLoc, | 
 |                                  const TemplateArgument &ArgPack) | 
 |     : Expr(SubstNonTypeTemplateParmPackExprClass, T, ValueKind, OK_Ordinary, | 
 |            true, true, true, true), | 
 |       Param(Param), Arguments(ArgPack.pack_begin()), | 
 |       NumArguments(ArgPack.pack_size()), NameLoc(NameLoc) {} | 
 |  | 
 | TemplateArgument SubstNonTypeTemplateParmPackExpr::getArgumentPack() const { | 
 |   return TemplateArgument(llvm::makeArrayRef(Arguments, NumArguments)); | 
 | } | 
 |  | 
 | FunctionParmPackExpr::FunctionParmPackExpr(QualType T, ParmVarDecl *ParamPack, | 
 |                                            SourceLocation NameLoc, | 
 |                                            unsigned NumParams, | 
 |                                            ParmVarDecl *const *Params) | 
 |     : Expr(FunctionParmPackExprClass, T, VK_LValue, OK_Ordinary, true, true, | 
 |            true, true), | 
 |       ParamPack(ParamPack), NameLoc(NameLoc), NumParameters(NumParams) { | 
 |   if (Params) | 
 |     std::uninitialized_copy(Params, Params + NumParams, | 
 |                             getTrailingObjects<ParmVarDecl *>()); | 
 | } | 
 |  | 
 | FunctionParmPackExpr * | 
 | FunctionParmPackExpr::Create(const ASTContext &Context, QualType T, | 
 |                              ParmVarDecl *ParamPack, SourceLocation NameLoc, | 
 |                              ArrayRef<ParmVarDecl *> Params) { | 
 |   return new (Context.Allocate(totalSizeToAlloc<ParmVarDecl *>(Params.size()))) | 
 |       FunctionParmPackExpr(T, ParamPack, NameLoc, Params.size(), Params.data()); | 
 | } | 
 |  | 
 | FunctionParmPackExpr * | 
 | FunctionParmPackExpr::CreateEmpty(const ASTContext &Context, | 
 |                                   unsigned NumParams) { | 
 |   return new (Context.Allocate(totalSizeToAlloc<ParmVarDecl *>(NumParams))) | 
 |       FunctionParmPackExpr(QualType(), nullptr, SourceLocation(), 0, nullptr); | 
 | } | 
 |  | 
 | void MaterializeTemporaryExpr::setExtendingDecl(const ValueDecl *ExtendedBy, | 
 |                                                 unsigned ManglingNumber) { | 
 |   // We only need extra state if we have to remember more than just the Stmt. | 
 |   if (!ExtendedBy) | 
 |     return; | 
 |  | 
 |   // We may need to allocate extra storage for the mangling number and the | 
 |   // extended-by ValueDecl. | 
 |   if (!State.is<ExtraState *>()) { | 
 |     auto *ES = new (ExtendedBy->getASTContext()) ExtraState; | 
 |     ES->Temporary = State.get<Stmt *>(); | 
 |     State = ES; | 
 |   } | 
 |  | 
 |   auto ES = State.get<ExtraState *>(); | 
 |   ES->ExtendingDecl = ExtendedBy; | 
 |   ES->ManglingNumber = ManglingNumber; | 
 | } | 
 |  | 
 | TypeTraitExpr::TypeTraitExpr(QualType T, SourceLocation Loc, TypeTrait Kind, | 
 |                              ArrayRef<TypeSourceInfo *> Args, | 
 |                              SourceLocation RParenLoc, | 
 |                              bool Value) | 
 |     : Expr(TypeTraitExprClass, T, VK_RValue, OK_Ordinary, | 
 |            /*TypeDependent=*/false, | 
 |            /*ValueDependent=*/false, | 
 |            /*InstantiationDependent=*/false, | 
 |            /*ContainsUnexpandedParameterPack=*/false), | 
 |       Loc(Loc), RParenLoc(RParenLoc) { | 
 |   TypeTraitExprBits.Kind = Kind; | 
 |   TypeTraitExprBits.Value = Value; | 
 |   TypeTraitExprBits.NumArgs = Args.size(); | 
 |  | 
 |   auto **ToArgs = getTrailingObjects<TypeSourceInfo *>(); | 
 |  | 
 |   for (unsigned I = 0, N = Args.size(); I != N; ++I) { | 
 |     if (Args[I]->getType()->isDependentType()) | 
 |       setValueDependent(true); | 
 |     if (Args[I]->getType()->isInstantiationDependentType()) | 
 |       setInstantiationDependent(true); | 
 |     if (Args[I]->getType()->containsUnexpandedParameterPack()) | 
 |       setContainsUnexpandedParameterPack(true); | 
 |  | 
 |     ToArgs[I] = Args[I]; | 
 |   } | 
 | } | 
 |  | 
 | TypeTraitExpr *TypeTraitExpr::Create(const ASTContext &C, QualType T, | 
 |                                      SourceLocation Loc, | 
 |                                      TypeTrait Kind, | 
 |                                      ArrayRef<TypeSourceInfo *> Args, | 
 |                                      SourceLocation RParenLoc, | 
 |                                      bool Value) { | 
 |   void *Mem = C.Allocate(totalSizeToAlloc<TypeSourceInfo *>(Args.size())); | 
 |   return new (Mem) TypeTraitExpr(T, Loc, Kind, Args, RParenLoc, Value); | 
 | } | 
 |  | 
 | TypeTraitExpr *TypeTraitExpr::CreateDeserialized(const ASTContext &C, | 
 |                                                  unsigned NumArgs) { | 
 |   void *Mem = C.Allocate(totalSizeToAlloc<TypeSourceInfo *>(NumArgs)); | 
 |   return new (Mem) TypeTraitExpr(EmptyShell()); | 
 | } | 
 |  | 
 | void ArrayTypeTraitExpr::anchor() {} |