diff options
Diffstat (limited to 'clang')
| -rw-r--r-- | clang/include/clang/AST/ExprCXX.h | 54 | ||||
| -rw-r--r-- | clang/include/clang/AST/StmtNodes.def | 1 | ||||
| -rw-r--r-- | clang/lib/AST/Expr.cpp | 4 | ||||
| -rw-r--r-- | clang/lib/AST/ExprCXX.cpp | 22 | ||||
| -rw-r--r-- | clang/lib/AST/StmtPrinter.cpp | 4 | ||||
| -rw-r--r-- | clang/lib/AST/StmtProfile.cpp | 4 | ||||
| -rw-r--r-- | clang/lib/CodeGen/CGExpr.cpp | 2 | ||||
| -rw-r--r-- | clang/lib/Sema/TreeTransform.h | 10 | ||||
| -rw-r--r-- | clang/tools/CIndex/CXCursor.cpp | 1 | 
9 files changed, 102 insertions, 0 deletions
diff --git a/clang/include/clang/AST/ExprCXX.h b/clang/include/clang/AST/ExprCXX.h index 9dd3bc96ef3..798927237b2 100644 --- a/clang/include/clang/AST/ExprCXX.h +++ b/clang/include/clang/AST/ExprCXX.h @@ -544,6 +544,60 @@ public:    virtual child_iterator child_end();  }; +/// CXXBindReferenceExpr - Represents binding an expression to a reference. +/// In the example: +/// +/// const int &i = 10; +/// +/// a bind reference expression is inserted to indicate that 10 is bound to +/// a reference. (Ans also that a temporary needs to be created to hold the +/// value). +class CXXBindReferenceExpr : public Expr { +  // SubExpr - The expression being bound. +  Stmt *SubExpr; +   +  // ExtendsLifetime - Whether binding this reference extends the lifetime of +  // the expression being bound. FIXME: Add C++ reference. +  bool ExtendsLifetime; + +  /// RequiresTemporaryCopy - Whether binding the subexpression requires a +  /// temporary copy. +  bool RequiresTemporaryCopy; +   +  CXXBindReferenceExpr(Expr *subexpr, bool ExtendsLifetime,  +                       bool RequiresTemporaryCopy) +  : Expr(CXXBindReferenceExprClass, subexpr->getType(), false, false), +    SubExpr(subexpr), ExtendsLifetime(ExtendsLifetime),  +    RequiresTemporaryCopy(RequiresTemporaryCopy) { } +  ~CXXBindReferenceExpr() { } + +protected: +  virtual void DoDestroy(ASTContext &C); + +public: +  static CXXBindReferenceExpr *Create(ASTContext &C, Expr *SubExpr, +                                      bool ExtendsLifetime,  +                                      bool RequiresTemporaryCopy); + +  const Expr *getSubExpr() const { return cast<Expr>(SubExpr); } +  Expr *getSubExpr() { return cast<Expr>(SubExpr); } +  void setSubExpr(Expr *E) { SubExpr = E; } + +  virtual SourceRange getSourceRange() const {  +    return SubExpr->getSourceRange(); +  } + +  // Implement isa/cast/dyncast/etc. +  static bool classof(const Stmt *T) { +    return T->getStmtClass() == CXXBindReferenceExprClass; +  } +  static bool classof(const CXXBindReferenceExpr *) { return true; } + +  // Iterators +  virtual child_iterator child_begin(); +  virtual child_iterator child_end(); +}; +  /// CXXConstructExpr - Represents a call to a C++ constructor.  class CXXConstructExpr : public Expr {    CXXConstructorDecl *Constructor; diff --git a/clang/include/clang/AST/StmtNodes.def b/clang/include/clang/AST/StmtNodes.def index 7102336180c..301d6baf54d 100644 --- a/clang/include/clang/AST/StmtNodes.def +++ b/clang/include/clang/AST/StmtNodes.def @@ -129,6 +129,7 @@ EXPR(UnaryTypeTraitExpr     , Expr)  EXPR(DependentScopeDeclRefExpr  , Expr)  EXPR(CXXConstructExpr       , Expr)  EXPR(CXXBindTemporaryExpr   , Expr) +EXPR(CXXBindReferenceExpr   , Expr)  EXPR(CXXExprWithTemporaries , Expr)  EXPR(CXXTemporaryObjectExpr , CXXConstructExpr)  EXPR(CXXUnresolvedConstructExpr, Expr) diff --git a/clang/lib/AST/Expr.cpp b/clang/lib/AST/Expr.cpp index fa44b510e99..4e05a850270 100644 --- a/clang/lib/AST/Expr.cpp +++ b/clang/lib/AST/Expr.cpp @@ -1205,6 +1205,9 @@ Expr::isLvalueResult Expr::isLvalueInternal(ASTContext &Ctx) const {    case CXXBindTemporaryExprClass:      return cast<CXXBindTemporaryExpr>(this)->getSubExpr()->        isLvalueInternal(Ctx); +  case CXXBindReferenceExprClass: +    // Something that's bound to a reference is always an lvalue. +    return LV_Valid;    case ConditionalOperatorClass: {      // Complicated handling is only for C++.      if (!Ctx.getLangOptions().CPlusPlus) @@ -1594,6 +1597,7 @@ static ICEDiag CheckICE(const Expr* E, ASTContext &Ctx) {    case Expr::DependentScopeDeclRefExprClass:    case Expr::CXXConstructExprClass:    case Expr::CXXBindTemporaryExprClass: +  case Expr::CXXBindReferenceExprClass:    case Expr::CXXExprWithTemporariesClass:    case Expr::CXXTemporaryObjectExprClass:    case Expr::CXXUnresolvedConstructExprClass: diff --git a/clang/lib/AST/ExprCXX.cpp b/clang/lib/AST/ExprCXX.cpp index 475dcf20481..3931bbd6dae 100644 --- a/clang/lib/AST/ExprCXX.cpp +++ b/clang/lib/AST/ExprCXX.cpp @@ -395,6 +395,19 @@ void CXXBindTemporaryExpr::DoDestroy(ASTContext &C) {    C.Deallocate(this);  } +CXXBindReferenceExpr *CXXBindReferenceExpr::Create(ASTContext &C, Expr *SubExpr, +                                                   bool ExtendsLifetime,  +                                                   bool RequiresTemporaryCopy) { +  return new (C) CXXBindReferenceExpr(SubExpr,  +                                      ExtendsLifetime, +                                      RequiresTemporaryCopy); +} + +void CXXBindReferenceExpr::DoDestroy(ASTContext &C) { +  this->~CXXBindReferenceExpr(); +  C.Deallocate(this); +} +  CXXTemporaryObjectExpr::CXXTemporaryObjectExpr(ASTContext &C,                                                 CXXConstructorDecl *Cons,                                                 QualType writtenTy, @@ -493,6 +506,15 @@ Stmt::child_iterator CXXBindTemporaryExpr::child_end() {    return &SubExpr + 1;  } +// CXXBindReferenceExpr +Stmt::child_iterator CXXBindReferenceExpr::child_begin() { +  return &SubExpr; +} + +Stmt::child_iterator CXXBindReferenceExpr::child_end() { +  return &SubExpr + 1; +} +  // CXXConstructExpr  Stmt::child_iterator CXXConstructExpr::child_begin() {    return &Args[0]; diff --git a/clang/lib/AST/StmtPrinter.cpp b/clang/lib/AST/StmtPrinter.cpp index bbb904de79b..3ae306d3c7a 100644 --- a/clang/lib/AST/StmtPrinter.cpp +++ b/clang/lib/AST/StmtPrinter.cpp @@ -1038,6 +1038,10 @@ void StmtPrinter::VisitCXXBindTemporaryExpr(CXXBindTemporaryExpr *Node) {    PrintExpr(Node->getSubExpr());  } +void StmtPrinter::VisitCXXBindReferenceExpr(CXXBindReferenceExpr *Node) { +  PrintExpr(Node->getSubExpr()); +} +  void StmtPrinter::VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *Node) {    OS << Node->getType().getAsString();    OS << "("; diff --git a/clang/lib/AST/StmtProfile.cpp b/clang/lib/AST/StmtProfile.cpp index b74e1ef0bab..3a19ec212c1 100644 --- a/clang/lib/AST/StmtProfile.cpp +++ b/clang/lib/AST/StmtProfile.cpp @@ -465,6 +465,10 @@ void StmtProfiler::VisitCXXBindTemporaryExpr(CXXBindTemporaryExpr *S) {           const_cast<CXXDestructorDecl *>(S->getTemporary()->getDestructor()));  } +void StmtProfiler::VisitCXXBindReferenceExpr(CXXBindReferenceExpr *S) { +  VisitExpr(S); +} +  void StmtProfiler::VisitCXXConstructExpr(CXXConstructExpr *S) {    VisitExpr(S);    VisitDecl(S->getConstructor()); diff --git a/clang/lib/CodeGen/CGExpr.cpp b/clang/lib/CodeGen/CGExpr.cpp index b2ecd55f441..f90443e2a4d 100644 --- a/clang/lib/CodeGen/CGExpr.cpp +++ b/clang/lib/CodeGen/CGExpr.cpp @@ -456,6 +456,8 @@ LValue CodeGenFunction::EmitLValue(const Expr *E) {      return EmitCXXConstructLValue(cast<CXXConstructExpr>(E));    case Expr::CXXBindTemporaryExprClass:      return EmitCXXBindTemporaryLValue(cast<CXXBindTemporaryExpr>(E)); +  case Expr::CXXBindReferenceExprClass: +    return EmitLValue(cast<CXXBindReferenceExpr>(E)->getSubExpr());    case Expr::CXXExprWithTemporariesClass:      return EmitCXXExprWithTemporariesLValue(cast<CXXExprWithTemporaries>(E));    case Expr::CXXZeroInitValueExprClass: diff --git a/clang/lib/Sema/TreeTransform.h b/clang/lib/Sema/TreeTransform.h index bd1b6e683d7..4e29c3ad85f 100644 --- a/clang/lib/Sema/TreeTransform.h +++ b/clang/lib/Sema/TreeTransform.h @@ -4865,6 +4865,16 @@ TreeTransform<Derived>::TransformCXXBindTemporaryExpr(CXXBindTemporaryExpr *E) {    return getDerived().TransformExpr(E->getSubExpr());  } +/// \brief Transform a C++ reference-binding expression. +/// +/// Since CXXBindReferenceExpr nodes are implicitly generated, we just +/// transform the subexpression and return that. +template<typename Derived> +Sema::OwningExprResult +TreeTransform<Derived>::TransformCXXBindReferenceExpr(CXXBindReferenceExpr *E) { +  return getDerived().TransformExpr(E->getSubExpr()); +} +  /// \brief Transform a C++ expression that contains temporaries that should  /// be destroyed after the expression is evaluated.  /// diff --git a/clang/tools/CIndex/CXCursor.cpp b/clang/tools/CIndex/CXCursor.cpp index 64ff4939469..03b050db313 100644 --- a/clang/tools/CIndex/CXCursor.cpp +++ b/clang/tools/CIndex/CXCursor.cpp @@ -165,6 +165,7 @@ CXCursor cxcursor::MakeCXCursor(Stmt *S, Decl *Parent, ASTUnit *TU) {    case Stmt::UnaryTypeTraitExprClass:         case Stmt::DependentScopeDeclRefExprClass:      case Stmt::CXXBindTemporaryExprClass:    +  case Stmt::CXXBindReferenceExprClass:       case Stmt::CXXExprWithTemporariesClass:     case Stmt::CXXUnresolvedConstructExprClass:    case Stmt::CXXDependentScopeMemberExprClass:  | 

