From: Douglas Gregor Date: Fri, 17 Apr 2009 19:05:30 +0000 (+0000) Subject: PCH support for GNU statement expressions X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=6a2dd55b0b3ae376d449a4b07bbb6b2d30b26330;p=clang PCH support for GNU statement expressions git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@69370 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/clang/AST/Expr.h b/include/clang/AST/Expr.h index 8059ad2f2e..ccfc9c71e0 100644 --- a/include/clang/AST/Expr.h +++ b/include/clang/AST/Expr.h @@ -1641,15 +1641,21 @@ public: SourceLocation lp, SourceLocation rp) : Expr(StmtExprClass, T), SubStmt(substmt), LParenLoc(lp), RParenLoc(rp) { } + /// \brief Build an empty statement expression. + explicit StmtExpr(EmptyShell Empty) : Expr(StmtExprClass, Empty) { } + CompoundStmt *getSubStmt() { return cast(SubStmt); } const CompoundStmt *getSubStmt() const { return cast(SubStmt); } - + void setSubStmt(CompoundStmt *S) { SubStmt = S; } + virtual SourceRange getSourceRange() const { return SourceRange(LParenLoc, RParenLoc); } SourceLocation getLParenLoc() const { return LParenLoc; } + void setLParenLoc(SourceLocation L) { LParenLoc = L; } SourceLocation getRParenLoc() const { return RParenLoc; } + void setRParenLoc(SourceLocation L) { RParenLoc = L; } static bool classof(const Stmt *T) { return T->getStmtClass() == StmtExprClass; diff --git a/include/clang/Frontend/PCHBitCodes.h b/include/clang/Frontend/PCHBitCodes.h index 93b78f100e..04849ddba1 100644 --- a/include/clang/Frontend/PCHBitCodes.h +++ b/include/clang/Frontend/PCHBitCodes.h @@ -456,9 +456,10 @@ namespace clang { EXPR_IMPLICIT_VALUE_INIT, /// \brief A VAArgExpr record. EXPR_VA_ARG, - // An AddrLabelExpr record. + /// \brief An AddrLabelExpr record. EXPR_ADDR_LABEL, - // FIXME: StmtExpr + /// \brief A StmtExpr record. + EXPR_STMT, /// \brief A TypesCompatibleExpr record. EXPR_TYPES_COMPATIBLE, /// \brief A ChooseExpr record. diff --git a/lib/Frontend/PCHReader.cpp b/lib/Frontend/PCHReader.cpp index 8928b063cf..e474dc7a70 100644 --- a/lib/Frontend/PCHReader.cpp +++ b/lib/Frontend/PCHReader.cpp @@ -289,6 +289,7 @@ namespace { unsigned VisitImplicitValueInitExpr(ImplicitValueInitExpr *E); unsigned VisitVAArgExpr(VAArgExpr *E); unsigned VisitAddrLabelExpr(AddrLabelExpr *E); + unsigned VisitStmtExpr(StmtExpr *E); unsigned VisitTypesCompatibleExpr(TypesCompatibleExpr *E); unsigned VisitChooseExpr(ChooseExpr *E); unsigned VisitGNUNullExpr(GNUNullExpr *E); @@ -749,6 +750,14 @@ unsigned PCHStmtReader::VisitAddrLabelExpr(AddrLabelExpr *E) { return 0; } +unsigned PCHStmtReader::VisitStmtExpr(StmtExpr *E) { + VisitExpr(E); + E->setLParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); + E->setRParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++])); + E->setSubStmt(cast_or_null(StmtStack.back())); + return 1; +} + unsigned PCHStmtReader::VisitTypesCompatibleExpr(TypesCompatibleExpr *E) { VisitExpr(E); E->setArgType1(Reader.GetType(Record[Idx++])); @@ -2359,6 +2368,10 @@ Stmt *PCHReader::ReadStmt() { S = new (Context) AddrLabelExpr(Empty); break; + case pch::EXPR_STMT: + S = new (Context) StmtExpr(Empty); + break; + case pch::EXPR_TYPES_COMPATIBLE: S = new (Context) TypesCompatibleExpr(Empty); break; diff --git a/lib/Frontend/PCHWriter.cpp b/lib/Frontend/PCHWriter.cpp index 4bbaaa0c76..14e979f2bc 100644 --- a/lib/Frontend/PCHWriter.cpp +++ b/lib/Frontend/PCHWriter.cpp @@ -491,6 +491,7 @@ namespace { void VisitImplicitValueInitExpr(ImplicitValueInitExpr *E); void VisitVAArgExpr(VAArgExpr *E); void VisitAddrLabelExpr(AddrLabelExpr *E); + void VisitStmtExpr(StmtExpr *E); void VisitTypesCompatibleExpr(TypesCompatibleExpr *E); void VisitChooseExpr(ChooseExpr *E); void VisitGNUNullExpr(GNUNullExpr *E); @@ -896,6 +897,14 @@ void PCHStmtWriter::VisitAddrLabelExpr(AddrLabelExpr *E) { Code = pch::EXPR_ADDR_LABEL; } +void PCHStmtWriter::VisitStmtExpr(StmtExpr *E) { + VisitExpr(E); + Writer.WriteSubStmt(E->getSubStmt()); + Writer.AddSourceLocation(E->getLParenLoc(), Record); + Writer.AddSourceLocation(E->getRParenLoc(), Record); + Code = pch::EXPR_STMT; +} + void PCHStmtWriter::VisitTypesCompatibleExpr(TypesCompatibleExpr *E) { VisitExpr(E); Writer.AddTypeRef(E->getArgType1(), Record); diff --git a/test/PCH/stmts.c b/test/PCH/stmts.c index fc6cfc29a2..c8fbc83a0f 100644 --- a/test/PCH/stmts.c +++ b/test/PCH/stmts.c @@ -10,3 +10,5 @@ int g1(int x) { return f1(x); } const char* query_name(void) { return what_is_my_name(); } int use_computed_goto(int x) { return computed_goto(x); } + +int get_weird_max(int x, int y) { return weird_max(x, y); } diff --git a/test/PCH/stmts.h b/test/PCH/stmts.h index 10842e8b29..367a2837cf 100644 --- a/test/PCH/stmts.h +++ b/test/PCH/stmts.h @@ -86,3 +86,8 @@ int computed_goto(int x) { done: return 5; } + +#define maxint(a,b) ({int _a = (a), _b = (b); _a > _b ? _a : _b; }) +int weird_max(int x, int y) { + return maxint(++x, --y); +}