From: Fariborz Jahanian Date: Wed, 5 Aug 2009 00:26:10 +0000 (+0000) Subject: Support for use of default argument in constructors. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=2eeed7bc4fd457ac57ff32ab3b02674588545f65;p=clang Support for use of default argument in constructors. work in progress. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@78132 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/clang/AST/ExprCXX.h b/include/clang/AST/ExprCXX.h index ae1c04a175..bee9b29896 100644 --- a/include/clang/AST/ExprCXX.h +++ b/include/clang/AST/ExprCXX.h @@ -511,6 +511,12 @@ public: unsigned getNumArgs() const { return NumArgs; } + /// setArg - Set the specified argument. + void setArg(unsigned Arg, Expr *ArgExpr) { + assert(Arg < NumArgs && "Arg access out of range!"); + Args[Arg] = ArgExpr; + } + virtual SourceRange getSourceRange() const { return SourceRange(); } static bool classof(const Stmt *T) { diff --git a/lib/AST/ExprCXX.cpp b/lib/AST/ExprCXX.cpp index df67bea6a6..d8e069d000 100644 --- a/lib/AST/ExprCXX.cpp +++ b/lib/AST/ExprCXX.cpp @@ -395,9 +395,13 @@ CXXConstructExpr::CXXConstructExpr(ASTContext &C, StmtClass SC, QualType T, (T->isDependentType() || CallExpr::hasAnyValueDependentArguments(args, numargs))), Constructor(D), Elidable(elidable), Args(0), NumArgs(numargs) { + // leave room for default arguments; + FunctionDecl *FDecl = cast(D); + unsigned NumArgsInProto = FDecl->param_size(); + NumArgs += (NumArgsInProto - numargs); if (NumArgs > 0) { Args = new (C) Stmt*[NumArgs]; - for (unsigned i = 0; i < NumArgs; ++i) + for (unsigned i = 0; i < numargs; ++i) Args[i] = args[i]; } } diff --git a/lib/Sema/SemaDeclCXX.cpp b/lib/Sema/SemaDeclCXX.cpp index a550e662c4..76bd71cdce 100644 --- a/lib/Sema/SemaDeclCXX.cpp +++ b/lib/Sema/SemaDeclCXX.cpp @@ -2360,8 +2360,29 @@ void Sema::InitializeVarWithConstructor(VarDecl *VD, CXXConstructorDecl *Constructor, QualType DeclInitType, Expr **Exprs, unsigned NumExprs) { - Expr *Temp = CXXConstructExpr::Create(Context, DeclInitType, Constructor, - false, Exprs, NumExprs); + CXXConstructExpr *Temp = CXXConstructExpr::Create(Context, DeclInitType, + Constructor, + false, Exprs, NumExprs); + // default arguments must be added to constructor call expression. + FunctionDecl *FDecl = cast(Constructor); + unsigned NumArgsInProto = FDecl->param_size(); + for (unsigned j = NumExprs; j != NumArgsInProto; j++) { + Expr *DefaultExpr = FDecl->getParamDecl(j)->getDefaultArg(); + + // If the default expression creates temporaries, we need to + // push them to the current stack of expression temporaries so they'll + // be properly destroyed. + if (CXXExprWithTemporaries *E + = dyn_cast_or_null(DefaultExpr)) { + assert(!E->shouldDestroyTemporaries() && + "Can't destroy temporaries in a default argument expr!"); + for (unsigned I = 0, N = E->getNumTemporaries(); I != N; ++I) + ExprTemporaries.push_back(E->getTemporary(I)); + } + Expr *Arg = new (Context) CXXDefaultArgExpr(FDecl->getParamDecl(j)); + Temp->setArg(j, Arg); + } + MarkDeclarationReferenced(VD->getLocation(), Constructor); VD->setInit(Context, Temp); }