blob: a8cc9d6f459175405eb31ab2130f19e80e67591b [file] [log] [blame]
Shih-wei Liaof8fd82b2010-02-10 11:10:31 -08001//===--- PCHWriterStmt.cpp - Statement and Expression Serialization -------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements serialization for Statements and Expressions.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/Frontend/PCHWriter.h"
15#include "clang/AST/DeclCXX.h"
16#include "clang/AST/DeclObjC.h"
17#include "clang/AST/StmtVisitor.h"
18#include "llvm/Bitcode/BitstreamWriter.h"
19using namespace clang;
20
21//===----------------------------------------------------------------------===//
22// Statement/expression serialization
23//===----------------------------------------------------------------------===//
24
25namespace {
26 class PCHStmtWriter : public StmtVisitor<PCHStmtWriter, void> {
27 PCHWriter &Writer;
28 PCHWriter::RecordData &Record;
29
30 public:
31 pch::StmtCode Code;
32
33 PCHStmtWriter(PCHWriter &Writer, PCHWriter::RecordData &Record)
34 : Writer(Writer), Record(Record) { }
35
36 void VisitStmt(Stmt *S);
37 void VisitNullStmt(NullStmt *S);
38 void VisitCompoundStmt(CompoundStmt *S);
39 void VisitSwitchCase(SwitchCase *S);
40 void VisitCaseStmt(CaseStmt *S);
41 void VisitDefaultStmt(DefaultStmt *S);
42 void VisitLabelStmt(LabelStmt *S);
43 void VisitIfStmt(IfStmt *S);
44 void VisitSwitchStmt(SwitchStmt *S);
45 void VisitWhileStmt(WhileStmt *S);
46 void VisitDoStmt(DoStmt *S);
47 void VisitForStmt(ForStmt *S);
48 void VisitGotoStmt(GotoStmt *S);
49 void VisitIndirectGotoStmt(IndirectGotoStmt *S);
50 void VisitContinueStmt(ContinueStmt *S);
51 void VisitBreakStmt(BreakStmt *S);
52 void VisitReturnStmt(ReturnStmt *S);
53 void VisitDeclStmt(DeclStmt *S);
54 void VisitAsmStmt(AsmStmt *S);
55 void VisitExpr(Expr *E);
56 void VisitPredefinedExpr(PredefinedExpr *E);
57 void VisitDeclRefExpr(DeclRefExpr *E);
58 void VisitIntegerLiteral(IntegerLiteral *E);
59 void VisitFloatingLiteral(FloatingLiteral *E);
60 void VisitImaginaryLiteral(ImaginaryLiteral *E);
61 void VisitStringLiteral(StringLiteral *E);
62 void VisitCharacterLiteral(CharacterLiteral *E);
63 void VisitParenExpr(ParenExpr *E);
64 void VisitUnaryOperator(UnaryOperator *E);
65 void VisitSizeOfAlignOfExpr(SizeOfAlignOfExpr *E);
66 void VisitArraySubscriptExpr(ArraySubscriptExpr *E);
67 void VisitCallExpr(CallExpr *E);
68 void VisitMemberExpr(MemberExpr *E);
69 void VisitCastExpr(CastExpr *E);
70 void VisitBinaryOperator(BinaryOperator *E);
71 void VisitCompoundAssignOperator(CompoundAssignOperator *E);
72 void VisitConditionalOperator(ConditionalOperator *E);
73 void VisitImplicitCastExpr(ImplicitCastExpr *E);
74 void VisitExplicitCastExpr(ExplicitCastExpr *E);
75 void VisitCStyleCastExpr(CStyleCastExpr *E);
76 void VisitCompoundLiteralExpr(CompoundLiteralExpr *E);
77 void VisitExtVectorElementExpr(ExtVectorElementExpr *E);
78 void VisitInitListExpr(InitListExpr *E);
79 void VisitDesignatedInitExpr(DesignatedInitExpr *E);
80 void VisitImplicitValueInitExpr(ImplicitValueInitExpr *E);
81 void VisitVAArgExpr(VAArgExpr *E);
82 void VisitAddrLabelExpr(AddrLabelExpr *E);
83 void VisitStmtExpr(StmtExpr *E);
84 void VisitTypesCompatibleExpr(TypesCompatibleExpr *E);
85 void VisitChooseExpr(ChooseExpr *E);
86 void VisitGNUNullExpr(GNUNullExpr *E);
87 void VisitShuffleVectorExpr(ShuffleVectorExpr *E);
88 void VisitBlockExpr(BlockExpr *E);
89 void VisitBlockDeclRefExpr(BlockDeclRefExpr *E);
90
91 // Objective-C Expressions
92 void VisitObjCStringLiteral(ObjCStringLiteral *E);
93 void VisitObjCEncodeExpr(ObjCEncodeExpr *E);
94 void VisitObjCSelectorExpr(ObjCSelectorExpr *E);
95 void VisitObjCProtocolExpr(ObjCProtocolExpr *E);
96 void VisitObjCIvarRefExpr(ObjCIvarRefExpr *E);
97 void VisitObjCPropertyRefExpr(ObjCPropertyRefExpr *E);
98 void VisitObjCImplicitSetterGetterRefExpr(
99 ObjCImplicitSetterGetterRefExpr *E);
100 void VisitObjCMessageExpr(ObjCMessageExpr *E);
101 void VisitObjCSuperExpr(ObjCSuperExpr *E);
102 void VisitObjCIsaExpr(ObjCIsaExpr *E);
103
104 // Objective-C Statements
105 void VisitObjCForCollectionStmt(ObjCForCollectionStmt *);
106 void VisitObjCAtCatchStmt(ObjCAtCatchStmt *);
107 void VisitObjCAtFinallyStmt(ObjCAtFinallyStmt *);
108 void VisitObjCAtTryStmt(ObjCAtTryStmt *);
109 void VisitObjCAtSynchronizedStmt(ObjCAtSynchronizedStmt *);
110 void VisitObjCAtThrowStmt(ObjCAtThrowStmt *);
111
112 // C++ Statements
113 void VisitCXXOperatorCallExpr(CXXOperatorCallExpr *E);
114 void VisitCXXConstructExpr(CXXConstructExpr *E);
115 void VisitCXXNamedCastExpr(CXXNamedCastExpr *E);
116 void VisitCXXStaticCastExpr(CXXStaticCastExpr *E);
117 void VisitCXXDynamicCastExpr(CXXDynamicCastExpr *E);
118 void VisitCXXReinterpretCastExpr(CXXReinterpretCastExpr *E);
119 void VisitCXXConstCastExpr(CXXConstCastExpr *E);
120 void VisitCXXFunctionalCastExpr(CXXFunctionalCastExpr *E);
121 void VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *E);
122 void VisitCXXNullPtrLiteralExpr(CXXNullPtrLiteralExpr *E);
123 };
124}
125
126void PCHStmtWriter::VisitStmt(Stmt *S) {
127}
128
129void PCHStmtWriter::VisitNullStmt(NullStmt *S) {
130 VisitStmt(S);
131 Writer.AddSourceLocation(S->getSemiLoc(), Record);
132 Code = pch::STMT_NULL;
133}
134
135void PCHStmtWriter::VisitCompoundStmt(CompoundStmt *S) {
136 VisitStmt(S);
137 Record.push_back(S->size());
138 for (CompoundStmt::body_iterator CS = S->body_begin(), CSEnd = S->body_end();
139 CS != CSEnd; ++CS)
140 Writer.WriteSubStmt(*CS);
141 Writer.AddSourceLocation(S->getLBracLoc(), Record);
142 Writer.AddSourceLocation(S->getRBracLoc(), Record);
143 Code = pch::STMT_COMPOUND;
144}
145
146void PCHStmtWriter::VisitSwitchCase(SwitchCase *S) {
147 VisitStmt(S);
148 Record.push_back(Writer.RecordSwitchCaseID(S));
149}
150
151void PCHStmtWriter::VisitCaseStmt(CaseStmt *S) {
152 VisitSwitchCase(S);
153 Writer.WriteSubStmt(S->getLHS());
154 Writer.WriteSubStmt(S->getRHS());
155 Writer.WriteSubStmt(S->getSubStmt());
156 Writer.AddSourceLocation(S->getCaseLoc(), Record);
157 Writer.AddSourceLocation(S->getEllipsisLoc(), Record);
158 Writer.AddSourceLocation(S->getColonLoc(), Record);
159 Code = pch::STMT_CASE;
160}
161
162void PCHStmtWriter::VisitDefaultStmt(DefaultStmt *S) {
163 VisitSwitchCase(S);
164 Writer.WriteSubStmt(S->getSubStmt());
165 Writer.AddSourceLocation(S->getDefaultLoc(), Record);
166 Writer.AddSourceLocation(S->getColonLoc(), Record);
167 Code = pch::STMT_DEFAULT;
168}
169
170void PCHStmtWriter::VisitLabelStmt(LabelStmt *S) {
171 VisitStmt(S);
172 Writer.AddIdentifierRef(S->getID(), Record);
173 Writer.WriteSubStmt(S->getSubStmt());
174 Writer.AddSourceLocation(S->getIdentLoc(), Record);
175 Record.push_back(Writer.GetLabelID(S));
176 Code = pch::STMT_LABEL;
177}
178
179void PCHStmtWriter::VisitIfStmt(IfStmt *S) {
180 VisitStmt(S);
181 Writer.AddDeclRef(S->getConditionVariable(), Record);
182 Writer.WriteSubStmt(S->getCond());
183 Writer.WriteSubStmt(S->getThen());
184 Writer.WriteSubStmt(S->getElse());
185 Writer.AddSourceLocation(S->getIfLoc(), Record);
186 Writer.AddSourceLocation(S->getElseLoc(), Record);
187 Code = pch::STMT_IF;
188}
189
190void PCHStmtWriter::VisitSwitchStmt(SwitchStmt *S) {
191 VisitStmt(S);
192 Writer.AddDeclRef(S->getConditionVariable(), Record);
193 Writer.WriteSubStmt(S->getCond());
194 Writer.WriteSubStmt(S->getBody());
195 Writer.AddSourceLocation(S->getSwitchLoc(), Record);
196 for (SwitchCase *SC = S->getSwitchCaseList(); SC;
197 SC = SC->getNextSwitchCase())
198 Record.push_back(Writer.getSwitchCaseID(SC));
199 Code = pch::STMT_SWITCH;
200}
201
202void PCHStmtWriter::VisitWhileStmt(WhileStmt *S) {
203 VisitStmt(S);
204 Writer.AddDeclRef(S->getConditionVariable(), Record);
205 Writer.WriteSubStmt(S->getCond());
206 Writer.WriteSubStmt(S->getBody());
207 Writer.AddSourceLocation(S->getWhileLoc(), Record);
208 Code = pch::STMT_WHILE;
209}
210
211void PCHStmtWriter::VisitDoStmt(DoStmt *S) {
212 VisitStmt(S);
213 Writer.WriteSubStmt(S->getCond());
214 Writer.WriteSubStmt(S->getBody());
215 Writer.AddSourceLocation(S->getDoLoc(), Record);
216 Writer.AddSourceLocation(S->getWhileLoc(), Record);
217 Writer.AddSourceLocation(S->getRParenLoc(), Record);
218 Code = pch::STMT_DO;
219}
220
221void PCHStmtWriter::VisitForStmt(ForStmt *S) {
222 VisitStmt(S);
223 Writer.WriteSubStmt(S->getInit());
224 Writer.WriteSubStmt(S->getCond());
225 Writer.AddDeclRef(S->getConditionVariable(), Record);
226 Writer.WriteSubStmt(S->getInc());
227 Writer.WriteSubStmt(S->getBody());
228 Writer.AddSourceLocation(S->getForLoc(), Record);
229 Writer.AddSourceLocation(S->getLParenLoc(), Record);
230 Writer.AddSourceLocation(S->getRParenLoc(), Record);
231 Code = pch::STMT_FOR;
232}
233
234void PCHStmtWriter::VisitGotoStmt(GotoStmt *S) {
235 VisitStmt(S);
236 Record.push_back(Writer.GetLabelID(S->getLabel()));
237 Writer.AddSourceLocation(S->getGotoLoc(), Record);
238 Writer.AddSourceLocation(S->getLabelLoc(), Record);
239 Code = pch::STMT_GOTO;
240}
241
242void PCHStmtWriter::VisitIndirectGotoStmt(IndirectGotoStmt *S) {
243 VisitStmt(S);
244 Writer.AddSourceLocation(S->getGotoLoc(), Record);
245 Writer.AddSourceLocation(S->getStarLoc(), Record);
246 Writer.WriteSubStmt(S->getTarget());
247 Code = pch::STMT_INDIRECT_GOTO;
248}
249
250void PCHStmtWriter::VisitContinueStmt(ContinueStmt *S) {
251 VisitStmt(S);
252 Writer.AddSourceLocation(S->getContinueLoc(), Record);
253 Code = pch::STMT_CONTINUE;
254}
255
256void PCHStmtWriter::VisitBreakStmt(BreakStmt *S) {
257 VisitStmt(S);
258 Writer.AddSourceLocation(S->getBreakLoc(), Record);
259 Code = pch::STMT_BREAK;
260}
261
262void PCHStmtWriter::VisitReturnStmt(ReturnStmt *S) {
263 VisitStmt(S);
264 Writer.WriteSubStmt(S->getRetValue());
265 Writer.AddSourceLocation(S->getReturnLoc(), Record);
266 Code = pch::STMT_RETURN;
267}
268
269void PCHStmtWriter::VisitDeclStmt(DeclStmt *S) {
270 VisitStmt(S);
271 Writer.AddSourceLocation(S->getStartLoc(), Record);
272 Writer.AddSourceLocation(S->getEndLoc(), Record);
273 DeclGroupRef DG = S->getDeclGroup();
274 for (DeclGroupRef::iterator D = DG.begin(), DEnd = DG.end(); D != DEnd; ++D)
275 Writer.AddDeclRef(*D, Record);
276 Code = pch::STMT_DECL;
277}
278
279void PCHStmtWriter::VisitAsmStmt(AsmStmt *S) {
280 VisitStmt(S);
281 Record.push_back(S->getNumOutputs());
282 Record.push_back(S->getNumInputs());
283 Record.push_back(S->getNumClobbers());
284 Writer.AddSourceLocation(S->getAsmLoc(), Record);
285 Writer.AddSourceLocation(S->getRParenLoc(), Record);
286 Record.push_back(S->isVolatile());
287 Record.push_back(S->isSimple());
288 Record.push_back(S->isMSAsm());
289 Writer.WriteSubStmt(S->getAsmString());
290
291 // Outputs
292 for (unsigned I = 0, N = S->getNumOutputs(); I != N; ++I) {
293 Writer.AddIdentifierRef(S->getOutputIdentifier(I), Record);
294 Writer.WriteSubStmt(S->getOutputConstraintLiteral(I));
295 Writer.WriteSubStmt(S->getOutputExpr(I));
296 }
297
298 // Inputs
299 for (unsigned I = 0, N = S->getNumInputs(); I != N; ++I) {
300 Writer.AddIdentifierRef(S->getInputIdentifier(I), Record);
301 Writer.WriteSubStmt(S->getInputConstraintLiteral(I));
302 Writer.WriteSubStmt(S->getInputExpr(I));
303 }
304
305 // Clobbers
306 for (unsigned I = 0, N = S->getNumClobbers(); I != N; ++I)
307 Writer.WriteSubStmt(S->getClobber(I));
308
309 Code = pch::STMT_ASM;
310}
311
312void PCHStmtWriter::VisitExpr(Expr *E) {
313 VisitStmt(E);
314 Writer.AddTypeRef(E->getType(), Record);
315 Record.push_back(E->isTypeDependent());
316 Record.push_back(E->isValueDependent());
317}
318
319void PCHStmtWriter::VisitPredefinedExpr(PredefinedExpr *E) {
320 VisitExpr(E);
321 Writer.AddSourceLocation(E->getLocation(), Record);
322 Record.push_back(E->getIdentType()); // FIXME: stable encoding
323 Code = pch::EXPR_PREDEFINED;
324}
325
326void PCHStmtWriter::VisitDeclRefExpr(DeclRefExpr *E) {
327 VisitExpr(E);
328 Writer.AddDeclRef(E->getDecl(), Record);
329 Writer.AddSourceLocation(E->getLocation(), Record);
330 // FIXME: write qualifier
331 // FIXME: write explicit template arguments
332 Code = pch::EXPR_DECL_REF;
333}
334
335void PCHStmtWriter::VisitIntegerLiteral(IntegerLiteral *E) {
336 VisitExpr(E);
337 Writer.AddSourceLocation(E->getLocation(), Record);
338 Writer.AddAPInt(E->getValue(), Record);
339 Code = pch::EXPR_INTEGER_LITERAL;
340}
341
342void PCHStmtWriter::VisitFloatingLiteral(FloatingLiteral *E) {
343 VisitExpr(E);
344 Writer.AddAPFloat(E->getValue(), Record);
345 Record.push_back(E->isExact());
346 Writer.AddSourceLocation(E->getLocation(), Record);
347 Code = pch::EXPR_FLOATING_LITERAL;
348}
349
350void PCHStmtWriter::VisitImaginaryLiteral(ImaginaryLiteral *E) {
351 VisitExpr(E);
352 Writer.WriteSubStmt(E->getSubExpr());
353 Code = pch::EXPR_IMAGINARY_LITERAL;
354}
355
356void PCHStmtWriter::VisitStringLiteral(StringLiteral *E) {
357 VisitExpr(E);
358 Record.push_back(E->getByteLength());
359 Record.push_back(E->getNumConcatenated());
360 Record.push_back(E->isWide());
361 // FIXME: String data should be stored as a blob at the end of the
362 // StringLiteral. However, we can't do so now because we have no
363 // provision for coping with abbreviations when we're jumping around
364 // the PCH file during deserialization.
365 Record.insert(Record.end(),
366 E->getStrData(), E->getStrData() + E->getByteLength());
367 for (unsigned I = 0, N = E->getNumConcatenated(); I != N; ++I)
368 Writer.AddSourceLocation(E->getStrTokenLoc(I), Record);
369 Code = pch::EXPR_STRING_LITERAL;
370}
371
372void PCHStmtWriter::VisitCharacterLiteral(CharacterLiteral *E) {
373 VisitExpr(E);
374 Record.push_back(E->getValue());
375 Writer.AddSourceLocation(E->getLocation(), Record);
376 Record.push_back(E->isWide());
377 Code = pch::EXPR_CHARACTER_LITERAL;
378}
379
380void PCHStmtWriter::VisitParenExpr(ParenExpr *E) {
381 VisitExpr(E);
382 Writer.AddSourceLocation(E->getLParen(), Record);
383 Writer.AddSourceLocation(E->getRParen(), Record);
384 Writer.WriteSubStmt(E->getSubExpr());
385 Code = pch::EXPR_PAREN;
386}
387
388void PCHStmtWriter::VisitUnaryOperator(UnaryOperator *E) {
389 VisitExpr(E);
390 Writer.WriteSubStmt(E->getSubExpr());
391 Record.push_back(E->getOpcode()); // FIXME: stable encoding
392 Writer.AddSourceLocation(E->getOperatorLoc(), Record);
393 Code = pch::EXPR_UNARY_OPERATOR;
394}
395
396void PCHStmtWriter::VisitSizeOfAlignOfExpr(SizeOfAlignOfExpr *E) {
397 VisitExpr(E);
398 Record.push_back(E->isSizeOf());
399 if (E->isArgumentType())
400 Writer.AddTypeSourceInfo(E->getArgumentTypeInfo(), Record);
401 else {
402 Record.push_back(0);
403 Writer.WriteSubStmt(E->getArgumentExpr());
404 }
405 Writer.AddSourceLocation(E->getOperatorLoc(), Record);
406 Writer.AddSourceLocation(E->getRParenLoc(), Record);
407 Code = pch::EXPR_SIZEOF_ALIGN_OF;
408}
409
410void PCHStmtWriter::VisitArraySubscriptExpr(ArraySubscriptExpr *E) {
411 VisitExpr(E);
412 Writer.WriteSubStmt(E->getLHS());
413 Writer.WriteSubStmt(E->getRHS());
414 Writer.AddSourceLocation(E->getRBracketLoc(), Record);
415 Code = pch::EXPR_ARRAY_SUBSCRIPT;
416}
417
418void PCHStmtWriter::VisitCallExpr(CallExpr *E) {
419 VisitExpr(E);
420 Record.push_back(E->getNumArgs());
421 Writer.AddSourceLocation(E->getRParenLoc(), Record);
422 Writer.WriteSubStmt(E->getCallee());
423 for (CallExpr::arg_iterator Arg = E->arg_begin(), ArgEnd = E->arg_end();
424 Arg != ArgEnd; ++Arg)
425 Writer.WriteSubStmt(*Arg);
426 Code = pch::EXPR_CALL;
427}
428
429void PCHStmtWriter::VisitMemberExpr(MemberExpr *E) {
430 VisitExpr(E);
431 Writer.WriteSubStmt(E->getBase());
432 Writer.AddDeclRef(E->getMemberDecl(), Record);
433 Writer.AddSourceLocation(E->getMemberLoc(), Record);
434 Record.push_back(E->isArrow());
435 // FIXME: C++ nested-name-specifier
436 // FIXME: C++ template argument list
437 Code = pch::EXPR_MEMBER;
438}
439
440void PCHStmtWriter::VisitObjCIsaExpr(ObjCIsaExpr *E) {
441 VisitExpr(E);
442 Writer.WriteSubStmt(E->getBase());
443 Writer.AddSourceLocation(E->getIsaMemberLoc(), Record);
444 Record.push_back(E->isArrow());
445 Code = pch::EXPR_OBJC_ISA;
446}
447
448void PCHStmtWriter::VisitCastExpr(CastExpr *E) {
449 VisitExpr(E);
450 Writer.WriteSubStmt(E->getSubExpr());
451 Record.push_back(E->getCastKind()); // FIXME: stable encoding
452}
453
454void PCHStmtWriter::VisitBinaryOperator(BinaryOperator *E) {
455 VisitExpr(E);
456 Writer.WriteSubStmt(E->getLHS());
457 Writer.WriteSubStmt(E->getRHS());
458 Record.push_back(E->getOpcode()); // FIXME: stable encoding
459 Writer.AddSourceLocation(E->getOperatorLoc(), Record);
460 Code = pch::EXPR_BINARY_OPERATOR;
461}
462
463void PCHStmtWriter::VisitCompoundAssignOperator(CompoundAssignOperator *E) {
464 VisitBinaryOperator(E);
465 Writer.AddTypeRef(E->getComputationLHSType(), Record);
466 Writer.AddTypeRef(E->getComputationResultType(), Record);
467 Code = pch::EXPR_COMPOUND_ASSIGN_OPERATOR;
468}
469
470void PCHStmtWriter::VisitConditionalOperator(ConditionalOperator *E) {
471 VisitExpr(E);
472 Writer.WriteSubStmt(E->getCond());
473 Writer.WriteSubStmt(E->getLHS());
474 Writer.WriteSubStmt(E->getRHS());
475 Writer.AddSourceLocation(E->getQuestionLoc(), Record);
476 Writer.AddSourceLocation(E->getColonLoc(), Record);
477 Code = pch::EXPR_CONDITIONAL_OPERATOR;
478}
479
480void PCHStmtWriter::VisitImplicitCastExpr(ImplicitCastExpr *E) {
481 VisitCastExpr(E);
482 Record.push_back(E->isLvalueCast());
483 Code = pch::EXPR_IMPLICIT_CAST;
484}
485
486void PCHStmtWriter::VisitExplicitCastExpr(ExplicitCastExpr *E) {
487 VisitCastExpr(E);
488 Writer.AddTypeSourceInfo(E->getTypeInfoAsWritten(), Record);
489}
490
491void PCHStmtWriter::VisitCStyleCastExpr(CStyleCastExpr *E) {
492 VisitExplicitCastExpr(E);
493 Writer.AddSourceLocation(E->getLParenLoc(), Record);
494 Writer.AddSourceLocation(E->getRParenLoc(), Record);
495 Code = pch::EXPR_CSTYLE_CAST;
496}
497
498void PCHStmtWriter::VisitCompoundLiteralExpr(CompoundLiteralExpr *E) {
499 VisitExpr(E);
500 Writer.AddSourceLocation(E->getLParenLoc(), Record);
501 Writer.AddTypeSourceInfo(E->getTypeSourceInfo(), Record);
502 Writer.WriteSubStmt(E->getInitializer());
503 Record.push_back(E->isFileScope());
504 Code = pch::EXPR_COMPOUND_LITERAL;
505}
506
507void PCHStmtWriter::VisitExtVectorElementExpr(ExtVectorElementExpr *E) {
508 VisitExpr(E);
509 Writer.WriteSubStmt(E->getBase());
510 Writer.AddIdentifierRef(&E->getAccessor(), Record);
511 Writer.AddSourceLocation(E->getAccessorLoc(), Record);
512 Code = pch::EXPR_EXT_VECTOR_ELEMENT;
513}
514
515void PCHStmtWriter::VisitInitListExpr(InitListExpr *E) {
516 VisitExpr(E);
517 Record.push_back(E->getNumInits());
518 for (unsigned I = 0, N = E->getNumInits(); I != N; ++I)
519 Writer.WriteSubStmt(E->getInit(I));
520 Writer.WriteSubStmt(E->getSyntacticForm());
521 Writer.AddSourceLocation(E->getLBraceLoc(), Record);
522 Writer.AddSourceLocation(E->getRBraceLoc(), Record);
523 Writer.AddDeclRef(E->getInitializedFieldInUnion(), Record);
524 Record.push_back(E->hadArrayRangeDesignator());
525 Code = pch::EXPR_INIT_LIST;
526}
527
528void PCHStmtWriter::VisitDesignatedInitExpr(DesignatedInitExpr *E) {
529 VisitExpr(E);
530 Record.push_back(E->getNumSubExprs());
531 for (unsigned I = 0, N = E->getNumSubExprs(); I != N; ++I)
532 Writer.WriteSubStmt(E->getSubExpr(I));
533 Writer.AddSourceLocation(E->getEqualOrColonLoc(), Record);
534 Record.push_back(E->usesGNUSyntax());
535 for (DesignatedInitExpr::designators_iterator D = E->designators_begin(),
536 DEnd = E->designators_end();
537 D != DEnd; ++D) {
538 if (D->isFieldDesignator()) {
539 if (FieldDecl *Field = D->getField()) {
540 Record.push_back(pch::DESIG_FIELD_DECL);
541 Writer.AddDeclRef(Field, Record);
542 } else {
543 Record.push_back(pch::DESIG_FIELD_NAME);
544 Writer.AddIdentifierRef(D->getFieldName(), Record);
545 }
546 Writer.AddSourceLocation(D->getDotLoc(), Record);
547 Writer.AddSourceLocation(D->getFieldLoc(), Record);
548 } else if (D->isArrayDesignator()) {
549 Record.push_back(pch::DESIG_ARRAY);
550 Record.push_back(D->getFirstExprIndex());
551 Writer.AddSourceLocation(D->getLBracketLoc(), Record);
552 Writer.AddSourceLocation(D->getRBracketLoc(), Record);
553 } else {
554 assert(D->isArrayRangeDesignator() && "Unknown designator");
555 Record.push_back(pch::DESIG_ARRAY_RANGE);
556 Record.push_back(D->getFirstExprIndex());
557 Writer.AddSourceLocation(D->getLBracketLoc(), Record);
558 Writer.AddSourceLocation(D->getEllipsisLoc(), Record);
559 Writer.AddSourceLocation(D->getRBracketLoc(), Record);
560 }
561 }
562 Code = pch::EXPR_DESIGNATED_INIT;
563}
564
565void PCHStmtWriter::VisitImplicitValueInitExpr(ImplicitValueInitExpr *E) {
566 VisitExpr(E);
567 Code = pch::EXPR_IMPLICIT_VALUE_INIT;
568}
569
570void PCHStmtWriter::VisitVAArgExpr(VAArgExpr *E) {
571 VisitExpr(E);
572 Writer.WriteSubStmt(E->getSubExpr());
573 Writer.AddSourceLocation(E->getBuiltinLoc(), Record);
574 Writer.AddSourceLocation(E->getRParenLoc(), Record);
575 Code = pch::EXPR_VA_ARG;
576}
577
578void PCHStmtWriter::VisitAddrLabelExpr(AddrLabelExpr *E) {
579 VisitExpr(E);
580 Writer.AddSourceLocation(E->getAmpAmpLoc(), Record);
581 Writer.AddSourceLocation(E->getLabelLoc(), Record);
582 Record.push_back(Writer.GetLabelID(E->getLabel()));
583 Code = pch::EXPR_ADDR_LABEL;
584}
585
586void PCHStmtWriter::VisitStmtExpr(StmtExpr *E) {
587 VisitExpr(E);
588 Writer.WriteSubStmt(E->getSubStmt());
589 Writer.AddSourceLocation(E->getLParenLoc(), Record);
590 Writer.AddSourceLocation(E->getRParenLoc(), Record);
591 Code = pch::EXPR_STMT;
592}
593
594void PCHStmtWriter::VisitTypesCompatibleExpr(TypesCompatibleExpr *E) {
595 VisitExpr(E);
596 Writer.AddTypeRef(E->getArgType1(), Record);
597 Writer.AddTypeRef(E->getArgType2(), Record);
598 Writer.AddSourceLocation(E->getBuiltinLoc(), Record);
599 Writer.AddSourceLocation(E->getRParenLoc(), Record);
600 Code = pch::EXPR_TYPES_COMPATIBLE;
601}
602
603void PCHStmtWriter::VisitChooseExpr(ChooseExpr *E) {
604 VisitExpr(E);
605 Writer.WriteSubStmt(E->getCond());
606 Writer.WriteSubStmt(E->getLHS());
607 Writer.WriteSubStmt(E->getRHS());
608 Writer.AddSourceLocation(E->getBuiltinLoc(), Record);
609 Writer.AddSourceLocation(E->getRParenLoc(), Record);
610 Code = pch::EXPR_CHOOSE;
611}
612
613void PCHStmtWriter::VisitGNUNullExpr(GNUNullExpr *E) {
614 VisitExpr(E);
615 Writer.AddSourceLocation(E->getTokenLocation(), Record);
616 Code = pch::EXPR_GNU_NULL;
617}
618
619void PCHStmtWriter::VisitShuffleVectorExpr(ShuffleVectorExpr *E) {
620 VisitExpr(E);
621 Record.push_back(E->getNumSubExprs());
622 for (unsigned I = 0, N = E->getNumSubExprs(); I != N; ++I)
623 Writer.WriteSubStmt(E->getExpr(I));
624 Writer.AddSourceLocation(E->getBuiltinLoc(), Record);
625 Writer.AddSourceLocation(E->getRParenLoc(), Record);
626 Code = pch::EXPR_SHUFFLE_VECTOR;
627}
628
629void PCHStmtWriter::VisitBlockExpr(BlockExpr *E) {
630 VisitExpr(E);
631 Writer.AddDeclRef(E->getBlockDecl(), Record);
632 Record.push_back(E->hasBlockDeclRefExprs());
633 Code = pch::EXPR_BLOCK;
634}
635
636void PCHStmtWriter::VisitBlockDeclRefExpr(BlockDeclRefExpr *E) {
637 VisitExpr(E);
638 Writer.AddDeclRef(E->getDecl(), Record);
639 Writer.AddSourceLocation(E->getLocation(), Record);
640 Record.push_back(E->isByRef());
641 Record.push_back(E->isConstQualAdded());
642 Code = pch::EXPR_BLOCK_DECL_REF;
643}
644
645//===----------------------------------------------------------------------===//
646// Objective-C Expressions and Statements.
647//===----------------------------------------------------------------------===//
648
649void PCHStmtWriter::VisitObjCStringLiteral(ObjCStringLiteral *E) {
650 VisitExpr(E);
651 Writer.WriteSubStmt(E->getString());
652 Writer.AddSourceLocation(E->getAtLoc(), Record);
653 Code = pch::EXPR_OBJC_STRING_LITERAL;
654}
655
656void PCHStmtWriter::VisitObjCEncodeExpr(ObjCEncodeExpr *E) {
657 VisitExpr(E);
658 Writer.AddTypeRef(E->getEncodedType(), Record);
659 Writer.AddSourceLocation(E->getAtLoc(), Record);
660 Writer.AddSourceLocation(E->getRParenLoc(), Record);
661 Code = pch::EXPR_OBJC_ENCODE;
662}
663
664void PCHStmtWriter::VisitObjCSelectorExpr(ObjCSelectorExpr *E) {
665 VisitExpr(E);
666 Writer.AddSelectorRef(E->getSelector(), Record);
667 Writer.AddSourceLocation(E->getAtLoc(), Record);
668 Writer.AddSourceLocation(E->getRParenLoc(), Record);
669 Code = pch::EXPR_OBJC_SELECTOR_EXPR;
670}
671
672void PCHStmtWriter::VisitObjCProtocolExpr(ObjCProtocolExpr *E) {
673 VisitExpr(E);
674 Writer.AddDeclRef(E->getProtocol(), Record);
675 Writer.AddSourceLocation(E->getAtLoc(), Record);
676 Writer.AddSourceLocation(E->getRParenLoc(), Record);
677 Code = pch::EXPR_OBJC_PROTOCOL_EXPR;
678}
679
680void PCHStmtWriter::VisitObjCIvarRefExpr(ObjCIvarRefExpr *E) {
681 VisitExpr(E);
682 Writer.AddDeclRef(E->getDecl(), Record);
683 Writer.AddSourceLocation(E->getLocation(), Record);
684 Writer.WriteSubStmt(E->getBase());
685 Record.push_back(E->isArrow());
686 Record.push_back(E->isFreeIvar());
687 Code = pch::EXPR_OBJC_IVAR_REF_EXPR;
688}
689
690void PCHStmtWriter::VisitObjCPropertyRefExpr(ObjCPropertyRefExpr *E) {
691 VisitExpr(E);
692 Writer.AddDeclRef(E->getProperty(), Record);
693 Writer.AddSourceLocation(E->getLocation(), Record);
694 Writer.WriteSubStmt(E->getBase());
695 Code = pch::EXPR_OBJC_PROPERTY_REF_EXPR;
696}
697
698void PCHStmtWriter::VisitObjCImplicitSetterGetterRefExpr(
699 ObjCImplicitSetterGetterRefExpr *E) {
700 VisitExpr(E);
701 Writer.AddDeclRef(E->getGetterMethod(), Record);
702 Writer.AddDeclRef(E->getSetterMethod(), Record);
703
704 // NOTE: InterfaceDecl and Base are mutually exclusive.
705 Writer.AddDeclRef(E->getInterfaceDecl(), Record);
706 Writer.WriteSubStmt(E->getBase());
707 Writer.AddSourceLocation(E->getLocation(), Record);
708 Writer.AddSourceLocation(E->getClassLoc(), Record);
709 Code = pch::EXPR_OBJC_KVC_REF_EXPR;
710}
711
712void PCHStmtWriter::VisitObjCMessageExpr(ObjCMessageExpr *E) {
713 VisitExpr(E);
714 Record.push_back(E->getNumArgs());
715 Writer.AddSourceLocation(E->getLeftLoc(), Record);
716 Writer.AddSourceLocation(E->getRightLoc(), Record);
717 Writer.AddSelectorRef(E->getSelector(), Record);
718 Writer.AddDeclRef(E->getMethodDecl(), Record); // optional
719 Writer.WriteSubStmt(E->getReceiver());
720
721 if (!E->getReceiver()) {
722 ObjCMessageExpr::ClassInfo CI = E->getClassInfo();
723 Writer.AddDeclRef(CI.first, Record);
724 Writer.AddIdentifierRef(CI.second, Record);
725 }
726
727 for (CallExpr::arg_iterator Arg = E->arg_begin(), ArgEnd = E->arg_end();
728 Arg != ArgEnd; ++Arg)
729 Writer.WriteSubStmt(*Arg);
730 Code = pch::EXPR_OBJC_MESSAGE_EXPR;
731}
732
733void PCHStmtWriter::VisitObjCSuperExpr(ObjCSuperExpr *E) {
734 VisitExpr(E);
735 Writer.AddSourceLocation(E->getLoc(), Record);
736 Code = pch::EXPR_OBJC_SUPER_EXPR;
737}
738
739void PCHStmtWriter::VisitObjCForCollectionStmt(ObjCForCollectionStmt *S) {
740 VisitStmt(S);
741 Writer.WriteSubStmt(S->getElement());
742 Writer.WriteSubStmt(S->getCollection());
743 Writer.WriteSubStmt(S->getBody());
744 Writer.AddSourceLocation(S->getForLoc(), Record);
745 Writer.AddSourceLocation(S->getRParenLoc(), Record);
746 Code = pch::STMT_OBJC_FOR_COLLECTION;
747}
748
749void PCHStmtWriter::VisitObjCAtCatchStmt(ObjCAtCatchStmt *S) {
750 Writer.WriteSubStmt(S->getCatchBody());
751 Writer.WriteSubStmt(S->getNextCatchStmt());
752 Writer.AddDeclRef(S->getCatchParamDecl(), Record);
753 Writer.AddSourceLocation(S->getAtCatchLoc(), Record);
754 Writer.AddSourceLocation(S->getRParenLoc(), Record);
755 Code = pch::STMT_OBJC_CATCH;
756}
757
758void PCHStmtWriter::VisitObjCAtFinallyStmt(ObjCAtFinallyStmt *S) {
759 Writer.WriteSubStmt(S->getFinallyBody());
760 Writer.AddSourceLocation(S->getAtFinallyLoc(), Record);
761 Code = pch::STMT_OBJC_FINALLY;
762}
763
764void PCHStmtWriter::VisitObjCAtTryStmt(ObjCAtTryStmt *S) {
765 Writer.WriteSubStmt(S->getTryBody());
766 Writer.WriteSubStmt(S->getCatchStmts());
767 Writer.WriteSubStmt(S->getFinallyStmt());
768 Writer.AddSourceLocation(S->getAtTryLoc(), Record);
769 Code = pch::STMT_OBJC_AT_TRY;
770}
771
772void PCHStmtWriter::VisitObjCAtSynchronizedStmt(ObjCAtSynchronizedStmt *S) {
773 Writer.WriteSubStmt(S->getSynchExpr());
774 Writer.WriteSubStmt(S->getSynchBody());
775 Writer.AddSourceLocation(S->getAtSynchronizedLoc(), Record);
776 Code = pch::STMT_OBJC_AT_SYNCHRONIZED;
777}
778
779void PCHStmtWriter::VisitObjCAtThrowStmt(ObjCAtThrowStmt *S) {
780 Writer.WriteSubStmt(S->getThrowExpr());
781 Writer.AddSourceLocation(S->getThrowLoc(), Record);
782 Code = pch::STMT_OBJC_AT_THROW;
783}
784
785//===----------------------------------------------------------------------===//
786// C++ Expressions and Statements.
787//===----------------------------------------------------------------------===//
788
789void PCHStmtWriter::VisitCXXOperatorCallExpr(CXXOperatorCallExpr *E) {
790 VisitCallExpr(E);
791 Record.push_back(E->getOperator());
792 Code = pch::EXPR_CXX_OPERATOR_CALL;
793}
794
795void PCHStmtWriter::VisitCXXConstructExpr(CXXConstructExpr *E) {
796 VisitExpr(E);
797 Writer.AddDeclRef(E->getConstructor(), Record);
798 Writer.AddSourceLocation(E->getLocation(), Record);
799 Record.push_back(E->isElidable());
800 Record.push_back(E->requiresZeroInitialization());
801 Record.push_back(E->getNumArgs());
802 for (unsigned I = 0, N = E->getNumArgs(); I != N; ++I)
803 Writer.WriteSubStmt(E->getArg(I));
804 Code = pch::EXPR_CXX_CONSTRUCT;
805}
806
807void PCHStmtWriter::VisitCXXNamedCastExpr(CXXNamedCastExpr *E) {
808 VisitExplicitCastExpr(E);
809 Writer.AddSourceLocation(E->getOperatorLoc(), Record);
810}
811
812void PCHStmtWriter::VisitCXXStaticCastExpr(CXXStaticCastExpr *E) {
813 VisitCXXNamedCastExpr(E);
814 Code = pch::EXPR_CXX_STATIC_CAST;
815}
816
817void PCHStmtWriter::VisitCXXDynamicCastExpr(CXXDynamicCastExpr *E) {
818 VisitCXXNamedCastExpr(E);
819 Code = pch::EXPR_CXX_DYNAMIC_CAST;
820}
821
822void PCHStmtWriter::VisitCXXReinterpretCastExpr(CXXReinterpretCastExpr *E) {
823 VisitCXXNamedCastExpr(E);
824 Code = pch::EXPR_CXX_REINTERPRET_CAST;
825}
826
827void PCHStmtWriter::VisitCXXConstCastExpr(CXXConstCastExpr *E) {
828 VisitCXXNamedCastExpr(E);
829 Code = pch::EXPR_CXX_CONST_CAST;
830}
831
832void PCHStmtWriter::VisitCXXFunctionalCastExpr(CXXFunctionalCastExpr *E) {
833 VisitExplicitCastExpr(E);
834 Writer.AddSourceLocation(E->getTypeBeginLoc(), Record);
835 Writer.AddSourceLocation(E->getRParenLoc(), Record);
836 Code = pch::EXPR_CXX_FUNCTIONAL_CAST;
837}
838
839void PCHStmtWriter::VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *E) {
840 VisitExpr(E);
841 Record.push_back(E->getValue());
842 Writer.AddSourceLocation(E->getLocation(), Record);
843 Code = pch::EXPR_CXX_BOOL_LITERAL;
844}
845
846void PCHStmtWriter::VisitCXXNullPtrLiteralExpr(CXXNullPtrLiteralExpr *E) {
847 VisitExpr(E);
848 Writer.AddSourceLocation(E->getLocation(), Record);
849 Code = pch::EXPR_CXX_NULL_PTR_LITERAL;
850}
851
852//===----------------------------------------------------------------------===//
853// PCHWriter Implementation
854//===----------------------------------------------------------------------===//
855
856unsigned PCHWriter::RecordSwitchCaseID(SwitchCase *S) {
857 assert(SwitchCaseIDs.find(S) == SwitchCaseIDs.end() &&
858 "SwitchCase recorded twice");
859 unsigned NextID = SwitchCaseIDs.size();
860 SwitchCaseIDs[S] = NextID;
861 return NextID;
862}
863
864unsigned PCHWriter::getSwitchCaseID(SwitchCase *S) {
865 assert(SwitchCaseIDs.find(S) != SwitchCaseIDs.end() &&
866 "SwitchCase hasn't been seen yet");
867 return SwitchCaseIDs[S];
868}
869
870/// \brief Retrieve the ID for the given label statement, which may
871/// or may not have been emitted yet.
872unsigned PCHWriter::GetLabelID(LabelStmt *S) {
873 std::map<LabelStmt *, unsigned>::iterator Pos = LabelIDs.find(S);
874 if (Pos != LabelIDs.end())
875 return Pos->second;
876
877 unsigned NextID = LabelIDs.size();
878 LabelIDs[S] = NextID;
879 return NextID;
880}
881
882/// \brief Write the given substatement or subexpression to the
883/// bitstream.
884void PCHWriter::WriteSubStmt(Stmt *S) {
885 RecordData Record;
886 PCHStmtWriter Writer(*this, Record);
887 ++NumStatements;
888
889 if (!S) {
890 Stream.EmitRecord(pch::STMT_NULL_PTR, Record);
891 return;
892 }
893
894 Writer.Code = pch::STMT_NULL_PTR;
895 Writer.Visit(S);
896 assert(Writer.Code != pch::STMT_NULL_PTR &&
897 "Unhandled expression writing PCH file");
898 Stream.EmitRecord(Writer.Code, Record);
899}
900
901/// \brief Flush all of the statements that have been added to the
902/// queue via AddStmt().
903void PCHWriter::FlushStmts() {
904 RecordData Record;
905 PCHStmtWriter Writer(*this, Record);
906
907 for (unsigned I = 0, N = StmtsToEmit.size(); I != N; ++I) {
908 ++NumStatements;
909 Stmt *S = StmtsToEmit[I];
910
911 if (!S) {
912 Stream.EmitRecord(pch::STMT_NULL_PTR, Record);
913 continue;
914 }
915
916 Writer.Code = pch::STMT_NULL_PTR;
917 Writer.Visit(S);
918 assert(Writer.Code != pch::STMT_NULL_PTR &&
919 "Unhandled expression writing PCH file");
920 Stream.EmitRecord(Writer.Code, Record);
921
922 assert(N == StmtsToEmit.size() &&
923 "Substatement writen via AddStmt rather than WriteSubStmt!");
924
925 // Note that we are at the end of a full expression. Any
926 // expression records that follow this one are part of a different
927 // expression.
928 Record.clear();
929 Stream.EmitRecord(pch::STMT_STOP, Record);
930 }
931
932 StmtsToEmit.clear();
933}