aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/clang/AST/Expr.h12
-rw-r--r--include/clang/Frontend/PCHBitCodes.h9
-rw-r--r--include/clang/Frontend/PCHWriter.h16
-rw-r--r--lib/Frontend/PCHReader.cpp170
-rw-r--r--lib/Frontend/PCHWriter.cpp48
-rw-r--r--test/PCH/enum.h2
-rw-r--r--tools/clang-cc/ASTConsumers.cpp9
7 files changed, 207 insertions, 59 deletions
diff --git a/include/clang/AST/Expr.h b/include/clang/AST/Expr.h
index f7893d75d7..cc6a0a6d61 100644
--- a/include/clang/AST/Expr.h
+++ b/include/clang/AST/Expr.h
@@ -1116,10 +1116,15 @@ protected:
ty->isDependentType() || (op && op->isValueDependent())),
Op(op) {}
+ /// \brief Construct an empty cast.
+ CastExpr(StmtClass SC, EmptyShell Empty)
+ : Expr(SC, Empty) { }
+
public:
Expr *getSubExpr() { return cast<Expr>(Op); }
const Expr *getSubExpr() const { return cast<Expr>(Op); }
-
+ void setSubExpr(Expr *E) { Op = E; }
+
static bool classof(const Stmt *T) {
StmtClass SC = T->getStmtClass();
if (SC >= CXXNamedCastExprClass && SC <= CXXFunctionalCastExprClass)
@@ -1161,6 +1166,11 @@ public:
ImplicitCastExpr(QualType ty, Expr *op, bool Lvalue) :
CastExpr(ImplicitCastExprClass, ty, op), LvalueCast(Lvalue) { }
+ /// \brief Construct an empty implicit cast.
+ explicit ImplicitCastExpr(EmptyShell Shell)
+ : CastExpr(ImplicitCastExprClass, Shell) { }
+
+
virtual SourceRange getSourceRange() const {
return getSubExpr()->getSourceRange();
}
diff --git a/include/clang/Frontend/PCHBitCodes.h b/include/clang/Frontend/PCHBitCodes.h
index 0184a7e9b1..66b800c577 100644
--- a/include/clang/Frontend/PCHBitCodes.h
+++ b/include/clang/Frontend/PCHBitCodes.h
@@ -368,8 +368,11 @@ namespace clang {
/// 100. Each constant describes a record for a specific
/// statement or expression class in the AST.
enum StmtCode {
+ /// \brief A marker record that indicates that we are at the end
+ /// of an expression.
+ EXPR_STOP,
/// \brief A NULL expression.
- EXPR_NULL = 100,
+ EXPR_NULL,
/// \brief A PredefinedExpr record.
EXPR_PREDEFINED,
/// \brief A DeclRefExpr record.
@@ -379,7 +382,9 @@ namespace clang {
/// \brief A FloatingLiteral record.
EXPR_FLOATING_LITERAL,
/// \brief A CharacterLiteral record.
- EXPR_CHARACTER_LITERAL
+ EXPR_CHARACTER_LITERAL,
+ /// \brief An ImplicitCastExpr record.
+ EXPR_IMPLICIT_CAST
};
/// @}
}
diff --git a/include/clang/Frontend/PCHWriter.h b/include/clang/Frontend/PCHWriter.h
index edac6bc001..d95dc431c1 100644
--- a/include/clang/Frontend/PCHWriter.h
+++ b/include/clang/Frontend/PCHWriter.h
@@ -106,10 +106,7 @@ private:
/// \brief Expressions that we've encountered while serializing a
/// declaration or type.
- ///
- /// The expressions in this queue will be emitted following the
- /// declaration or type.
- std::queue<Expr *> ExprsToEmit;
+ llvm::SmallVector<Expr *, 8> ExprsToEmit;
void WriteTargetTriple(const TargetInfo &Target);
void WriteLanguageOptions(const LangOptions &LangOpts);
@@ -156,7 +153,16 @@ public:
/// \brief Add the given expression to the queue of expressions to
/// emit.
- void AddExpr(Expr *E) { ExprsToEmit.push(E); }
+ ///
+ /// This routine should be used when emitting types and declarations
+ /// that have expressions as part of their formulation. Once the
+ /// type or declaration has been written, call FlushExprs() to write
+ /// the corresponding expressions just after the type or
+ /// declaration.
+ void AddExpr(Expr *E) { ExprsToEmit.push_back(E); }
+
+ /// \brief Write the given subexpression to the bitstream.
+ void WriteSubExpr(Expr *E);
/// \brief Flush all of the expressions that have been added to the
/// queue via AddExpr().
diff --git a/lib/Frontend/PCHReader.cpp b/lib/Frontend/PCHReader.cpp
index b0b35e491b..057e66a6f7 100644
--- a/lib/Frontend/PCHReader.cpp
+++ b/lib/Frontend/PCHReader.cpp
@@ -215,61 +215,87 @@ PCHDeclReader::VisitDeclContext(DeclContext *DC) {
//===----------------------------------------------------------------------===//
namespace {
class VISIBILITY_HIDDEN PCHStmtReader
- : public StmtVisitor<PCHStmtReader, void> {
+ : public StmtVisitor<PCHStmtReader, unsigned> {
PCHReader &Reader;
const PCHReader::RecordData &Record;
unsigned &Idx;
+ llvm::SmallVectorImpl<Expr *> &ExprStack;
public:
PCHStmtReader(PCHReader &Reader, const PCHReader::RecordData &Record,
- unsigned &Idx)
- : Reader(Reader), Record(Record), Idx(Idx) { }
-
- void VisitExpr(Expr *E);
- void VisitPredefinedExpr(PredefinedExpr *E);
- void VisitDeclRefExpr(DeclRefExpr *E);
- void VisitIntegerLiteral(IntegerLiteral *E);
- void VisitFloatingLiteral(FloatingLiteral *E);
- void VisitCharacterLiteral(CharacterLiteral *E);
+ unsigned &Idx, llvm::SmallVectorImpl<Expr *> &ExprStack)
+ : Reader(Reader), Record(Record), Idx(Idx), ExprStack(ExprStack) { }
+
+ // Each of the Visit* functions reads in part of the expression
+ // from the given record and the current expression stack, then
+ // return the total number of operands that it read from the
+ // expression stack.
+
+ unsigned VisitExpr(Expr *E);
+ unsigned VisitPredefinedExpr(PredefinedExpr *E);
+ unsigned VisitDeclRefExpr(DeclRefExpr *E);
+ unsigned VisitIntegerLiteral(IntegerLiteral *E);
+ unsigned VisitFloatingLiteral(FloatingLiteral *E);
+ unsigned VisitCharacterLiteral(CharacterLiteral *E);
+ unsigned VisitCastExpr(CastExpr *E);
+ unsigned VisitImplicitCastExpr(ImplicitCastExpr *E);
};
}
-void PCHStmtReader::VisitExpr(Expr *E) {
+unsigned PCHStmtReader::VisitExpr(Expr *E) {
E->setType(Reader.GetType(Record[Idx++]));
E->setTypeDependent(Record[Idx++]);
E->setValueDependent(Record[Idx++]);
+ return 0;
}
-void PCHStmtReader::VisitPredefinedExpr(PredefinedExpr *E) {
+unsigned PCHStmtReader::VisitPredefinedExpr(PredefinedExpr *E) {
VisitExpr(E);
E->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
E->setIdentType((PredefinedExpr::IdentType)Record[Idx++]);
+ return 0;
}
-void PCHStmtReader::VisitDeclRefExpr(DeclRefExpr *E) {
+unsigned PCHStmtReader::VisitDeclRefExpr(DeclRefExpr *E) {
VisitExpr(E);
E->setDecl(cast<NamedDecl>(Reader.GetDecl(Record[Idx++])));
E->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
+ return 0;
}
-void PCHStmtReader::VisitIntegerLiteral(IntegerLiteral *E) {
+unsigned PCHStmtReader::VisitIntegerLiteral(IntegerLiteral *E) {
VisitExpr(E);
E->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
E->setValue(Reader.ReadAPInt(Record, Idx));
+ return 0;
}
-void PCHStmtReader::VisitFloatingLiteral(FloatingLiteral *E) {
+unsigned PCHStmtReader::VisitFloatingLiteral(FloatingLiteral *E) {
VisitExpr(E);
E->setValue(Reader.ReadAPFloat(Record, Idx));
E->setExact(Record[Idx++]);
E->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
+ return 0;
}
-void PCHStmtReader::VisitCharacterLiteral(CharacterLiteral *E) {
+unsigned PCHStmtReader::VisitCharacterLiteral(CharacterLiteral *E) {
VisitExpr(E);
E->setValue(Record[Idx++]);
E->setLocation(SourceLocation::getFromRawEncoding(Record[Idx++]));
E->setWide(Record[Idx++]);
+ return 0;
+}
+
+unsigned PCHStmtReader::VisitCastExpr(CastExpr *E) {
+ VisitExpr(E);
+ E->setSubExpr(ExprStack.back());
+ return 1;
+}
+
+unsigned PCHStmtReader::VisitImplicitCastExpr(ImplicitCastExpr *E) {
+ VisitCastExpr(E);
+ E->setLvalueCast(Record[Idx++]);
+ return 1;
}
// FIXME: use the diagnostics machinery
@@ -1507,46 +1533,100 @@ llvm::APFloat PCHReader::ReadAPFloat(const RecordData &Record, unsigned &Idx) {
}
Expr *PCHReader::ReadExpr() {
+ // Within the bitstream, expressions are stored in Reverse Polish
+ // Notation, with each of the subexpressions preceding the
+ // expression they are stored in. To evaluate expressions, we
+ // continue reading expressions and placing them on the stack, with
+ // expressions having operands removing those operands from the
+ // stack. Evaluation terminates when we see a EXPR_STOP record, and
+ // the single remaining expression on the stack is our result.
RecordData Record;
- unsigned Code = Stream.ReadCode();
- unsigned Idx = 0;
- PCHStmtReader Reader(*this, Record, Idx);
+ unsigned Idx;
+ llvm::SmallVector<Expr *, 16> ExprStack;
+ PCHStmtReader Reader(*this, Record, Idx, ExprStack);
Stmt::EmptyShell Empty;
- Expr *E = 0;
- switch ((pch::StmtCode)Stream.ReadRecord(Code, Record)) {
- case pch::EXPR_NULL:
- E = 0;
- break;
+ while (true) {
+ unsigned Code = Stream.ReadCode();
+ if (Code == llvm::bitc::END_BLOCK) {
+ if (Stream.ReadBlockEnd()) {
+ Error("Error at end of Source Manager block");
+ return 0;
+ }
+ break;
+ }
- case pch::EXPR_PREDEFINED:
- // FIXME: untested (until we can serialize function bodies).
- E = new (Context) PredefinedExpr(Empty);
- break;
+ if (Code == llvm::bitc::ENTER_SUBBLOCK) {
+ // No known subblocks, always skip them.
+ Stream.ReadSubBlockID();
+ if (Stream.SkipBlock()) {
+ Error("Malformed block record");
+ return 0;
+ }
+ continue;
+ }
- case pch::EXPR_DECL_REF:
- E = new (Context) DeclRefExpr(Empty);
- break;
+ if (Code == llvm::bitc::DEFINE_ABBREV) {
+ Stream.ReadAbbrevRecord();
+ continue;
+ }
- case pch::EXPR_INTEGER_LITERAL:
- E = new (Context) IntegerLiteral(Empty);
- break;
+ Expr *E = 0;
+ Idx = 0;
+ Record.clear();
+ bool Finished = false;
+ switch ((pch::StmtCode)Stream.ReadRecord(Code, Record)) {
+ case pch::EXPR_STOP:
+ Finished = true;
+ break;
- case pch::EXPR_FLOATING_LITERAL:
- E = new (Context) FloatingLiteral(Empty);
- break;
+ case pch::EXPR_NULL:
+ E = 0;
+ break;
- case pch::EXPR_CHARACTER_LITERAL:
- E = new (Context) CharacterLiteral(Empty);
- break;
- }
+ case pch::EXPR_PREDEFINED:
+ // FIXME: untested (until we can serialize function bodies).
+ E = new (Context) PredefinedExpr(Empty);
+ break;
+
+ case pch::EXPR_DECL_REF:
+ E = new (Context) DeclRefExpr(Empty);
+ break;
+
+ case pch::EXPR_INTEGER_LITERAL:
+ E = new (Context) IntegerLiteral(Empty);
+ break;
+
+ case pch::EXPR_FLOATING_LITERAL:
+ E = new (Context) FloatingLiteral(Empty);
+ break;
+
+ case pch::EXPR_CHARACTER_LITERAL:
+ E = new (Context) CharacterLiteral(Empty);
+ break;
- if (E)
- Reader.Visit(E);
+ case pch::EXPR_IMPLICIT_CAST:
+ E = new (Context) ImplicitCastExpr(Empty);
+ break;
+ }
- assert(Idx == Record.size() && "Invalid deserialization of expression");
+ // We hit an EXPR_STOP, so we're done with this expression.
+ if (Finished)
+ break;
- return E;
+ if (E) {
+ unsigned NumSubExprs = Reader.Visit(E);
+ while (NumSubExprs > 0) {
+ ExprStack.pop_back();
+ --NumSubExprs;
+ }
+ }
+
+ assert(Idx == Record.size() && "Invalid deserialization of expression");
+ ExprStack.push_back(E);
+ }
+ assert(ExprStack.size() == 1 && "Extra expressions on stack!");
+ return ExprStack.back();
}
DiagnosticBuilder PCHReader::Diag(unsigned DiagID) {
diff --git a/lib/Frontend/PCHWriter.cpp b/lib/Frontend/PCHWriter.cpp
index 20aee9c441..4ca3c8d0ae 100644
--- a/lib/Frontend/PCHWriter.cpp
+++ b/lib/Frontend/PCHWriter.cpp
@@ -449,6 +449,8 @@ namespace {
void VisitIntegerLiteral(IntegerLiteral *E);
void VisitFloatingLiteral(FloatingLiteral *E);
void VisitCharacterLiteral(CharacterLiteral *E);
+ void VisitCastExpr(CastExpr *E);
+ void VisitImplicitCastExpr(ImplicitCastExpr *E);
};
}
@@ -495,6 +497,17 @@ void PCHStmtWriter::VisitCharacterLiteral(CharacterLiteral *E) {
Code = pch::EXPR_CHARACTER_LITERAL;
}
+void PCHStmtWriter::VisitCastExpr(CastExpr *E) {
+ VisitExpr(E);
+ Writer.WriteSubExpr(E->getSubExpr());
+}
+
+void PCHStmtWriter::VisitImplicitCastExpr(ImplicitCastExpr *E) {
+ VisitCastExpr(E);
+ Record.push_back(E->isLvalueCast());
+ Code = pch::EXPR_IMPLICIT_CAST;
+}
+
//===----------------------------------------------------------------------===//
// PCHWriter Implementation
//===----------------------------------------------------------------------===//
@@ -1245,16 +1258,32 @@ void PCHWriter::AddDeclarationName(DeclarationName Name, RecordData &Record) {
}
}
+/// \brief Write the given subexpression to the bitstream.
+void PCHWriter::WriteSubExpr(Expr *E) {
+ RecordData Record;
+ PCHStmtWriter Writer(*this, Record);
+
+ if (!E) {
+ S.EmitRecord(pch::EXPR_NULL, Record);
+ return;
+ }
+
+ Writer.Code = pch::EXPR_NULL;
+ Writer.Visit(E);
+ assert(Writer.Code != pch::EXPR_NULL &&
+ "Unhandled expression writing PCH file");
+ S.EmitRecord(Writer.Code, Record);
+}
+
/// \brief Flush all of the expressions that have been added to the
/// queue via AddExpr().
void PCHWriter::FlushExprs() {
RecordData Record;
PCHStmtWriter Writer(*this, Record);
- while (!ExprsToEmit.empty()) {
- Expr *E = ExprsToEmit.front();
- ExprsToEmit.pop();
- Record.clear();
+ for (unsigned I = 0, N = ExprsToEmit.size(); I != N; ++I) {
+ Expr *E = ExprsToEmit[I];
+
if (!E) {
S.EmitRecord(pch::EXPR_NULL, Record);
continue;
@@ -1265,5 +1294,16 @@ void PCHWriter::FlushExprs() {
assert(Writer.Code != pch::EXPR_NULL &&
"Unhandled expression writing PCH file");
S.EmitRecord(Writer.Code, Record);
+
+ assert(N == ExprsToEmit.size() &&
+ "Subexpression writen via AddExpr rather than WriteSubExpr!");
+
+ // Note that we are at the end of a full expression. Any
+ // expression records that follow this one are part of a different
+ // expression.
+ Record.clear();
+ S.EmitRecord(pch::EXPR_STOP, Record);
}
+
+ ExprsToEmit.clear();
}
diff --git a/test/PCH/enum.h b/test/PCH/enum.h
index c94e314eb0..7dc4e631d6 100644
--- a/test/PCH/enum.h
+++ b/test/PCH/enum.h
@@ -13,4 +13,4 @@ enum Shape {
Circle
};
-enum Shape aRoundShape;// FIXME: = Circle;
+enum Shape aRoundShape = Circle;
diff --git a/tools/clang-cc/ASTConsumers.cpp b/tools/clang-cc/ASTConsumers.cpp
index 74683e3031..61a9e4460a 100644
--- a/tools/clang-cc/ASTConsumers.cpp
+++ b/tools/clang-cc/ASTConsumers.cpp
@@ -190,7 +190,14 @@ void DeclPrinter::Print(NamedDecl *ND) {
std::string Name = ND->getNameAsString();
// This forms: "int a".
dyn_cast<ValueDecl>(ND)->getType().getAsStringInternal(Name);
- Out << Name << ";\n";
+ Out << Name;
+ if (VarDecl *Var = dyn_cast<VarDecl>(ND)) {
+ if (Var->getInit()) {
+ Out << " = ";
+ Var->getInit()->printPretty(Out);
+ }
+ }
+ Out << ";\n";
break;
}
case Decl::Namespace: