blob: b55869b0826dd32749a717daeb487d1ed6dd28a7 [file] [log] [blame]
Chris Lattner4b009652007-07-25 00:24:17 +00001//===--- StmtPrinter.cpp - Printing implementation for Stmt ASTs ----------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner959e5be2007-12-29 19:59:25 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Chris Lattner4b009652007-07-25 00:24:17 +00007//
8//===----------------------------------------------------------------------===//
9//
Chris Lattner95578782007-08-08 22:51:59 +000010// This file implements the Stmt::dumpPretty/Stmt::printPretty methods, which
11// pretty print the AST back out to C code.
Chris Lattner4b009652007-07-25 00:24:17 +000012//
13//===----------------------------------------------------------------------===//
14
15#include "clang/AST/StmtVisitor.h"
16#include "clang/AST/Decl.h"
Ted Kremenek10364dd2007-10-17 18:36:42 +000017#include "clang/AST/DeclObjC.h"
Chris Lattner4b009652007-07-25 00:24:17 +000018#include "clang/AST/ExprCXX.h"
Steve Naroff9ed3e772008-05-29 21:12:08 +000019#include "clang/AST/ExprObjC.h"
Ted Kremenek08176a52007-08-31 21:30:12 +000020#include "clang/AST/PrettyPrinter.h"
Chris Lattner2fd1c652007-10-07 08:58:51 +000021#include "clang/Basic/IdentifierTable.h"
Chris Lattner4b009652007-07-25 00:24:17 +000022#include "llvm/Support/Compiler.h"
Ted Kremenekdf409eb2007-11-26 22:50:46 +000023#include "llvm/Support/Streams.h"
Chris Lattner4b009652007-07-25 00:24:17 +000024#include <iomanip>
25using namespace clang;
26
27//===----------------------------------------------------------------------===//
28// StmtPrinter Visitor
29//===----------------------------------------------------------------------===//
30
31namespace {
Chris Lattner7ba21fa2007-08-21 04:04:25 +000032 class VISIBILITY_HIDDEN StmtPrinter : public StmtVisitor<StmtPrinter> {
Chris Lattner4b009652007-07-25 00:24:17 +000033 std::ostream &OS;
34 unsigned IndentLevel;
Ted Kremenek08176a52007-08-31 21:30:12 +000035 clang::PrinterHelper* Helper;
Chris Lattner4b009652007-07-25 00:24:17 +000036 public:
Ted Kremenek08176a52007-08-31 21:30:12 +000037 StmtPrinter(std::ostream &os, PrinterHelper* helper) :
38 OS(os), IndentLevel(0), Helper(helper) {}
Chris Lattner4b009652007-07-25 00:24:17 +000039
40 void PrintStmt(Stmt *S, int SubIndent = 1) {
41 IndentLevel += SubIndent;
42 if (S && isa<Expr>(S)) {
43 // If this is an expr used in a stmt context, indent and newline it.
44 Indent();
Chris Lattner7ba21fa2007-08-21 04:04:25 +000045 Visit(S);
Chris Lattner4b009652007-07-25 00:24:17 +000046 OS << ";\n";
47 } else if (S) {
Chris Lattner7ba21fa2007-08-21 04:04:25 +000048 Visit(S);
Chris Lattner4b009652007-07-25 00:24:17 +000049 } else {
50 Indent() << "<<<NULL STATEMENT>>>\n";
51 }
52 IndentLevel -= SubIndent;
53 }
54
55 void PrintRawCompoundStmt(CompoundStmt *S);
56 void PrintRawDecl(Decl *D);
57 void PrintRawIfStmt(IfStmt *If);
58
59 void PrintExpr(Expr *E) {
60 if (E)
Chris Lattner7ba21fa2007-08-21 04:04:25 +000061 Visit(E);
Chris Lattner4b009652007-07-25 00:24:17 +000062 else
63 OS << "<null expr>";
64 }
65
66 std::ostream &Indent(int Delta = 0) const {
67 for (int i = 0, e = IndentLevel+Delta; i < e; ++i)
68 OS << " ";
69 return OS;
70 }
71
Chris Lattner2af6a802007-08-30 17:59:59 +000072 bool PrintOffsetOfDesignator(Expr *E);
73 void VisitUnaryOffsetOf(UnaryOperator *Node);
74
Ted Kremenek08176a52007-08-31 21:30:12 +000075 void Visit(Stmt* S) {
76 if (Helper && Helper->handledStmt(S,OS))
77 return;
78 else StmtVisitor<StmtPrinter>::Visit(S);
79 }
80
Chris Lattner7ba21fa2007-08-21 04:04:25 +000081 void VisitStmt(Stmt *Node);
Chris Lattner4b009652007-07-25 00:24:17 +000082#define STMT(N, CLASS, PARENT) \
Chris Lattner7ba21fa2007-08-21 04:04:25 +000083 void Visit##CLASS(CLASS *Node);
Chris Lattner4b009652007-07-25 00:24:17 +000084#include "clang/AST/StmtNodes.def"
85 };
86}
87
88//===----------------------------------------------------------------------===//
89// Stmt printing methods.
90//===----------------------------------------------------------------------===//
91
92void StmtPrinter::VisitStmt(Stmt *Node) {
93 Indent() << "<<unknown stmt type>>\n";
94}
95
96/// PrintRawCompoundStmt - Print a compound stmt without indenting the {, and
97/// with no newline after the }.
98void StmtPrinter::PrintRawCompoundStmt(CompoundStmt *Node) {
99 OS << "{\n";
100 for (CompoundStmt::body_iterator I = Node->body_begin(), E = Node->body_end();
101 I != E; ++I)
102 PrintStmt(*I);
103
104 Indent() << "}";
105}
106
107void StmtPrinter::PrintRawDecl(Decl *D) {
108 // FIXME: Need to complete/beautify this... this code simply shows the
109 // nodes are where they need to be.
110 if (TypedefDecl *localType = dyn_cast<TypedefDecl>(D)) {
111 OS << "typedef " << localType->getUnderlyingType().getAsString();
112 OS << " " << localType->getName();
113 } else if (ValueDecl *VD = dyn_cast<ValueDecl>(D)) {
114 // Emit storage class for vardecls.
115 if (VarDecl *V = dyn_cast<VarDecl>(VD)) {
116 switch (V->getStorageClass()) {
117 default: assert(0 && "Unknown storage class!");
118 case VarDecl::None: break;
119 case VarDecl::Extern: OS << "extern "; break;
120 case VarDecl::Static: OS << "static "; break;
121 case VarDecl::Auto: OS << "auto "; break;
122 case VarDecl::Register: OS << "register "; break;
123 }
124 }
125
126 std::string Name = VD->getName();
127 VD->getType().getAsStringInternal(Name);
128 OS << Name;
129
130 // If this is a vardecl with an initializer, emit it.
131 if (VarDecl *V = dyn_cast<VarDecl>(VD)) {
132 if (V->getInit()) {
133 OS << " = ";
134 PrintExpr(V->getInit());
135 }
136 }
Steve Naroff3340c232007-11-17 21:21:01 +0000137 } else if (TagDecl *TD = dyn_cast<TagDecl>(D)) {
138 // print a free standing tag decl (e.g. "struct x;").
139 OS << TD->getKindName();
140 OS << " ";
141 if (const IdentifierInfo *II = TD->getIdentifier())
142 OS << II->getName();
143 else
144 OS << "<anonymous>";
145 // FIXME: print tag bodies.
Chris Lattner4b009652007-07-25 00:24:17 +0000146 } else {
Chris Lattner4b009652007-07-25 00:24:17 +0000147 assert(0 && "Unexpected decl");
148 }
149}
150
151
152void StmtPrinter::VisitNullStmt(NullStmt *Node) {
153 Indent() << ";\n";
154}
155
156void StmtPrinter::VisitDeclStmt(DeclStmt *Node) {
Steve Naroff2591e1b2007-09-13 23:52:58 +0000157 for (ScopedDecl *D = Node->getDecl(); D; D = D->getNextDeclarator()) {
Chris Lattner4b009652007-07-25 00:24:17 +0000158 Indent();
159 PrintRawDecl(D);
160 OS << ";\n";
161 }
162}
163
164void StmtPrinter::VisitCompoundStmt(CompoundStmt *Node) {
165 Indent();
166 PrintRawCompoundStmt(Node);
167 OS << "\n";
168}
169
170void StmtPrinter::VisitCaseStmt(CaseStmt *Node) {
171 Indent(-1) << "case ";
172 PrintExpr(Node->getLHS());
173 if (Node->getRHS()) {
174 OS << " ... ";
175 PrintExpr(Node->getRHS());
176 }
177 OS << ":\n";
178
179 PrintStmt(Node->getSubStmt(), 0);
180}
181
182void StmtPrinter::VisitDefaultStmt(DefaultStmt *Node) {
183 Indent(-1) << "default:\n";
184 PrintStmt(Node->getSubStmt(), 0);
185}
186
187void StmtPrinter::VisitLabelStmt(LabelStmt *Node) {
188 Indent(-1) << Node->getName() << ":\n";
189 PrintStmt(Node->getSubStmt(), 0);
190}
191
192void StmtPrinter::PrintRawIfStmt(IfStmt *If) {
193 OS << "if ";
194 PrintExpr(If->getCond());
195
196 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(If->getThen())) {
197 OS << ' ';
198 PrintRawCompoundStmt(CS);
199 OS << (If->getElse() ? ' ' : '\n');
200 } else {
201 OS << '\n';
202 PrintStmt(If->getThen());
203 if (If->getElse()) Indent();
204 }
205
206 if (Stmt *Else = If->getElse()) {
207 OS << "else";
208
209 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Else)) {
210 OS << ' ';
211 PrintRawCompoundStmt(CS);
212 OS << '\n';
213 } else if (IfStmt *ElseIf = dyn_cast<IfStmt>(Else)) {
214 OS << ' ';
215 PrintRawIfStmt(ElseIf);
216 } else {
217 OS << '\n';
218 PrintStmt(If->getElse());
219 }
220 }
221}
222
223void StmtPrinter::VisitIfStmt(IfStmt *If) {
224 Indent();
225 PrintRawIfStmt(If);
226}
227
228void StmtPrinter::VisitSwitchStmt(SwitchStmt *Node) {
229 Indent() << "switch (";
230 PrintExpr(Node->getCond());
231 OS << ")";
232
233 // Pretty print compoundstmt bodies (very common).
234 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Node->getBody())) {
235 OS << " ";
236 PrintRawCompoundStmt(CS);
237 OS << "\n";
238 } else {
239 OS << "\n";
240 PrintStmt(Node->getBody());
241 }
242}
243
244void StmtPrinter::VisitSwitchCase(SwitchCase*) {
245 assert(0 && "SwitchCase is an abstract class");
246}
247
248void StmtPrinter::VisitWhileStmt(WhileStmt *Node) {
249 Indent() << "while (";
250 PrintExpr(Node->getCond());
251 OS << ")\n";
252 PrintStmt(Node->getBody());
253}
254
255void StmtPrinter::VisitDoStmt(DoStmt *Node) {
Chris Lattnercfe0ff02007-09-15 21:49:37 +0000256 Indent() << "do ";
257 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Node->getBody())) {
258 PrintRawCompoundStmt(CS);
259 OS << " ";
260 } else {
261 OS << "\n";
262 PrintStmt(Node->getBody());
263 Indent();
264 }
265
266 OS << "while ";
Chris Lattner4b009652007-07-25 00:24:17 +0000267 PrintExpr(Node->getCond());
268 OS << ";\n";
269}
270
271void StmtPrinter::VisitForStmt(ForStmt *Node) {
272 Indent() << "for (";
273 if (Node->getInit()) {
274 if (DeclStmt *DS = dyn_cast<DeclStmt>(Node->getInit()))
275 PrintRawDecl(DS->getDecl());
276 else
277 PrintExpr(cast<Expr>(Node->getInit()));
278 }
Chris Lattnercfe0ff02007-09-15 21:49:37 +0000279 OS << ";";
280 if (Node->getCond()) {
281 OS << " ";
Chris Lattner4b009652007-07-25 00:24:17 +0000282 PrintExpr(Node->getCond());
Chris Lattnercfe0ff02007-09-15 21:49:37 +0000283 }
284 OS << ";";
285 if (Node->getInc()) {
286 OS << " ";
Chris Lattner4b009652007-07-25 00:24:17 +0000287 PrintExpr(Node->getInc());
Chris Lattnercfe0ff02007-09-15 21:49:37 +0000288 }
289 OS << ") ";
290
291 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Node->getBody())) {
292 PrintRawCompoundStmt(CS);
293 OS << "\n";
294 } else {
295 OS << "\n";
296 PrintStmt(Node->getBody());
297 }
Chris Lattner4b009652007-07-25 00:24:17 +0000298}
299
Ted Kremenek42730c52008-01-07 19:49:32 +0000300void StmtPrinter::VisitObjCForCollectionStmt(ObjCForCollectionStmt *Node) {
Fariborz Jahanian9e920f32008-01-02 22:54:34 +0000301 Indent() << "for (";
302 if (DeclStmt *DS = dyn_cast<DeclStmt>(Node->getElement()))
303 PrintRawDecl(DS->getDecl());
304 else
305 PrintExpr(cast<Expr>(Node->getElement()));
306 OS << " in ";
307 PrintExpr(Node->getCollection());
308 OS << ") ";
309
310 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Node->getBody())) {
311 PrintRawCompoundStmt(CS);
312 OS << "\n";
313 } else {
314 OS << "\n";
315 PrintStmt(Node->getBody());
316 }
317}
318
Chris Lattner4b009652007-07-25 00:24:17 +0000319void StmtPrinter::VisitGotoStmt(GotoStmt *Node) {
320 Indent() << "goto " << Node->getLabel()->getName() << ";\n";
321}
322
323void StmtPrinter::VisitIndirectGotoStmt(IndirectGotoStmt *Node) {
324 Indent() << "goto *";
325 PrintExpr(Node->getTarget());
326 OS << ";\n";
327}
328
329void StmtPrinter::VisitContinueStmt(ContinueStmt *Node) {
330 Indent() << "continue;\n";
331}
332
333void StmtPrinter::VisitBreakStmt(BreakStmt *Node) {
334 Indent() << "break;\n";
335}
336
337
338void StmtPrinter::VisitReturnStmt(ReturnStmt *Node) {
339 Indent() << "return";
340 if (Node->getRetValue()) {
341 OS << " ";
342 PrintExpr(Node->getRetValue());
343 }
344 OS << ";\n";
345}
346
Chris Lattner8a40a832007-10-29 04:04:16 +0000347
348void StmtPrinter::VisitAsmStmt(AsmStmt *Node) {
Anders Carlsson759f45d2007-11-23 23:12:25 +0000349 Indent() << "asm ";
350
351 if (Node->isVolatile())
352 OS << "volatile ";
353
354 OS << "(";
Anders Carlsson076c1112007-11-20 19:21:03 +0000355 VisitStringLiteral(Node->getAsmString());
Anders Carlsson965d5202007-11-22 01:36:19 +0000356
357 // Outputs
358 if (Node->getNumOutputs() != 0 || Node->getNumInputs() != 0 ||
359 Node->getNumClobbers() != 0)
360 OS << " : ";
361
362 for (unsigned i = 0, e = Node->getNumOutputs(); i != e; ++i) {
363 if (i != 0)
364 OS << ", ";
365
366 if (!Node->getOutputName(i).empty()) {
367 OS << '[';
368 OS << Node->getOutputName(i);
369 OS << "] ";
370 }
371
372 VisitStringLiteral(Node->getOutputConstraint(i));
373 OS << " ";
374 Visit(Node->getOutputExpr(i));
375 }
376
377 // Inputs
378 if (Node->getNumInputs() != 0 || Node->getNumClobbers() != 0)
379 OS << " : ";
380
381 for (unsigned i = 0, e = Node->getNumInputs(); i != e; ++i) {
382 if (i != 0)
383 OS << ", ";
384
385 if (!Node->getInputName(i).empty()) {
386 OS << '[';
387 OS << Node->getInputName(i);
388 OS << "] ";
389 }
390
391 VisitStringLiteral(Node->getInputConstraint(i));
392 OS << " ";
393 Visit(Node->getInputExpr(i));
394 }
395
396 // Clobbers
397 if (Node->getNumClobbers() != 0)
398 OS << " : ";
399
400 for (unsigned i = 0, e = Node->getNumClobbers(); i != e; ++i) {
401 if (i != 0)
402 OS << ", ";
403
404 VisitStringLiteral(Node->getClobber(i));
405 }
406
Anders Carlsson076c1112007-11-20 19:21:03 +0000407 OS << ");\n";
Chris Lattner8a40a832007-10-29 04:04:16 +0000408}
409
Ted Kremenek42730c52008-01-07 19:49:32 +0000410void StmtPrinter::VisitObjCAtTryStmt(ObjCAtTryStmt *Node) {
Fariborz Jahanian59e9e042007-11-02 18:16:07 +0000411 Indent() << "@try";
412 if (CompoundStmt *TS = dyn_cast<CompoundStmt>(Node->getTryBody())) {
413 PrintRawCompoundStmt(TS);
414 OS << "\n";
415 }
416
Ted Kremenek42730c52008-01-07 19:49:32 +0000417 for (ObjCAtCatchStmt *catchStmt =
418 static_cast<ObjCAtCatchStmt *>(Node->getCatchStmts());
Fariborz Jahanian59e9e042007-11-02 18:16:07 +0000419 catchStmt;
420 catchStmt =
Ted Kremenek42730c52008-01-07 19:49:32 +0000421 static_cast<ObjCAtCatchStmt *>(catchStmt->getNextCatchStmt())) {
Fariborz Jahanian59e9e042007-11-02 18:16:07 +0000422 Indent() << "@catch(";
423 if (catchStmt->getCatchParamStmt()) {
424 if (DeclStmt *DS = dyn_cast<DeclStmt>(catchStmt->getCatchParamStmt()))
425 PrintRawDecl(DS->getDecl());
426 }
427 OS << ")";
428 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(catchStmt->getCatchBody()))
429 {
430 PrintRawCompoundStmt(CS);
431 OS << "\n";
432 }
433 }
434
Ted Kremenek42730c52008-01-07 19:49:32 +0000435 if (ObjCAtFinallyStmt *FS =static_cast<ObjCAtFinallyStmt *>(
Fariborz Jahanian50b47922007-11-07 00:46:42 +0000436 Node->getFinallyStmt())) {
437 Indent() << "@finally";
438 PrintRawCompoundStmt(dyn_cast<CompoundStmt>(FS->getFinallyBody()));
Fariborz Jahanian59e9e042007-11-02 18:16:07 +0000439 OS << "\n";
440 }
Fariborz Jahanian70952482007-11-01 21:12:44 +0000441}
442
Ted Kremenek42730c52008-01-07 19:49:32 +0000443void StmtPrinter::VisitObjCAtFinallyStmt(ObjCAtFinallyStmt *Node) {
Fariborz Jahanian70952482007-11-01 21:12:44 +0000444}
445
Ted Kremenek42730c52008-01-07 19:49:32 +0000446void StmtPrinter::VisitObjCAtCatchStmt (ObjCAtCatchStmt *Node) {
Fariborz Jahanian70952482007-11-01 21:12:44 +0000447 Indent() << "@catch (...) { /* todo */ } \n";
448}
449
Fariborz Jahanian5f5d6222008-01-30 17:38:29 +0000450void StmtPrinter::VisitObjCAtThrowStmt(ObjCAtThrowStmt *Node) {
Fariborz Jahanian08df2c62007-11-07 02:00:49 +0000451 Indent() << "@throw";
452 if (Node->getThrowExpr()) {
453 OS << " ";
454 PrintExpr(Node->getThrowExpr());
455 }
456 OS << ";\n";
457}
458
Fariborz Jahanian5f5d6222008-01-30 17:38:29 +0000459void StmtPrinter::VisitObjCAtSynchronizedStmt(ObjCAtSynchronizedStmt *Node) {
Fariborz Jahanian993360a2008-01-29 18:21:32 +0000460 Indent() << "@synchronized (";
461 PrintExpr(Node->getSynchExpr());
462 OS << ")";
Fariborz Jahanian5f5d6222008-01-30 17:38:29 +0000463 PrintRawCompoundStmt(Node->getSynchBody());
464 OS << "\n";
Fariborz Jahanian993360a2008-01-29 18:21:32 +0000465}
466
Chris Lattner4b009652007-07-25 00:24:17 +0000467//===----------------------------------------------------------------------===//
468// Expr printing methods.
469//===----------------------------------------------------------------------===//
470
471void StmtPrinter::VisitExpr(Expr *Node) {
472 OS << "<<unknown expr type>>";
473}
474
475void StmtPrinter::VisitDeclRefExpr(DeclRefExpr *Node) {
476 OS << Node->getDecl()->getName();
477}
478
Steve Naroff5eb2a4a2007-11-12 14:29:37 +0000479void StmtPrinter::VisitObjCIvarRefExpr(ObjCIvarRefExpr *Node) {
Fariborz Jahanian4af72492007-11-12 22:29:28 +0000480 if (Node->getBase()) {
481 PrintExpr(Node->getBase());
482 OS << (Node->isArrow() ? "->" : ".");
483 }
Steve Naroff5eb2a4a2007-11-12 14:29:37 +0000484 OS << Node->getDecl()->getName();
485}
486
Chris Lattner4b009652007-07-25 00:24:17 +0000487void StmtPrinter::VisitPreDefinedExpr(PreDefinedExpr *Node) {
488 switch (Node->getIdentType()) {
489 default:
490 assert(0 && "unknown case");
491 case PreDefinedExpr::Func:
492 OS << "__func__";
493 break;
494 case PreDefinedExpr::Function:
495 OS << "__FUNCTION__";
496 break;
497 case PreDefinedExpr::PrettyFunction:
498 OS << "__PRETTY_FUNCTION__";
499 break;
500 }
501}
502
503void StmtPrinter::VisitCharacterLiteral(CharacterLiteral *Node) {
504 // FIXME should print an L for wchar_t constants
505 unsigned value = Node->getValue();
506 switch (value) {
507 case '\\':
508 OS << "'\\\\'";
509 break;
510 case '\'':
511 OS << "'\\''";
512 break;
513 case '\a':
514 // TODO: K&R: the meaning of '\\a' is different in traditional C
515 OS << "'\\a'";
516 break;
517 case '\b':
518 OS << "'\\b'";
519 break;
520 // Nonstandard escape sequence.
521 /*case '\e':
522 OS << "'\\e'";
523 break;*/
524 case '\f':
525 OS << "'\\f'";
526 break;
527 case '\n':
528 OS << "'\\n'";
529 break;
530 case '\r':
531 OS << "'\\r'";
532 break;
533 case '\t':
534 OS << "'\\t'";
535 break;
536 case '\v':
537 OS << "'\\v'";
538 break;
539 default:
Ted Kremenekf91bc1d2008-02-23 00:52:04 +0000540 if (value < 256 && isprint(value)) {
Chris Lattner4b009652007-07-25 00:24:17 +0000541 OS << "'" << (char)value << "'";
542 } else if (value < 256) {
543 OS << "'\\x" << std::hex << value << std::dec << "'";
544 } else {
545 // FIXME what to really do here?
546 OS << value;
547 }
548 }
549}
550
551void StmtPrinter::VisitIntegerLiteral(IntegerLiteral *Node) {
552 bool isSigned = Node->getType()->isSignedIntegerType();
553 OS << Node->getValue().toString(10, isSigned);
554
555 // Emit suffixes. Integer literals are always a builtin integer type.
556 switch (cast<BuiltinType>(Node->getType().getCanonicalType())->getKind()) {
557 default: assert(0 && "Unexpected type for integer literal!");
558 case BuiltinType::Int: break; // no suffix.
559 case BuiltinType::UInt: OS << 'U'; break;
560 case BuiltinType::Long: OS << 'L'; break;
561 case BuiltinType::ULong: OS << "UL"; break;
562 case BuiltinType::LongLong: OS << "LL"; break;
563 case BuiltinType::ULongLong: OS << "ULL"; break;
564 }
565}
566void StmtPrinter::VisitFloatingLiteral(FloatingLiteral *Node) {
Chris Lattner6a390402007-08-01 00:23:58 +0000567 // FIXME: print value more precisely.
Chris Lattner7f298762007-09-22 18:47:25 +0000568 OS << Node->getValueAsDouble();
Chris Lattner4b009652007-07-25 00:24:17 +0000569}
Chris Lattner1de66eb2007-08-26 03:42:43 +0000570
571void StmtPrinter::VisitImaginaryLiteral(ImaginaryLiteral *Node) {
572 PrintExpr(Node->getSubExpr());
573 OS << "i";
574}
575
Chris Lattner4b009652007-07-25 00:24:17 +0000576void StmtPrinter::VisitStringLiteral(StringLiteral *Str) {
577 if (Str->isWide()) OS << 'L';
578 OS << '"';
Anders Carlsson55bfe0d2007-10-15 02:50:23 +0000579
Chris Lattner4b009652007-07-25 00:24:17 +0000580 // FIXME: this doesn't print wstrings right.
581 for (unsigned i = 0, e = Str->getByteLength(); i != e; ++i) {
582 switch (Str->getStrData()[i]) {
583 default: OS << Str->getStrData()[i]; break;
584 // Handle some common ones to make dumps prettier.
585 case '\\': OS << "\\\\"; break;
586 case '"': OS << "\\\""; break;
587 case '\n': OS << "\\n"; break;
588 case '\t': OS << "\\t"; break;
589 case '\a': OS << "\\a"; break;
590 case '\b': OS << "\\b"; break;
591 }
592 }
593 OS << '"';
594}
595void StmtPrinter::VisitParenExpr(ParenExpr *Node) {
596 OS << "(";
597 PrintExpr(Node->getSubExpr());
598 OS << ")";
599}
600void StmtPrinter::VisitUnaryOperator(UnaryOperator *Node) {
Chris Lattner2b97b092007-08-23 21:46:40 +0000601 if (!Node->isPostfix()) {
Chris Lattner4b009652007-07-25 00:24:17 +0000602 OS << UnaryOperator::getOpcodeStr(Node->getOpcode());
Chris Lattner2b97b092007-08-23 21:46:40 +0000603
604 // Print a space if this is an "identifier operator" like sizeof or __real.
605 switch (Node->getOpcode()) {
606 default: break;
607 case UnaryOperator::SizeOf:
608 case UnaryOperator::AlignOf:
609 case UnaryOperator::Real:
610 case UnaryOperator::Imag:
611 case UnaryOperator::Extension:
612 OS << ' ';
613 break;
614 }
615 }
Chris Lattner4b009652007-07-25 00:24:17 +0000616 PrintExpr(Node->getSubExpr());
617
618 if (Node->isPostfix())
619 OS << UnaryOperator::getOpcodeStr(Node->getOpcode());
Chris Lattner4b009652007-07-25 00:24:17 +0000620}
Chris Lattner2af6a802007-08-30 17:59:59 +0000621
622bool StmtPrinter::PrintOffsetOfDesignator(Expr *E) {
623 if (isa<CompoundLiteralExpr>(E)) {
624 // Base case, print the type and comma.
625 OS << E->getType().getAsString() << ", ";
626 return true;
627 } else if (ArraySubscriptExpr *ASE = dyn_cast<ArraySubscriptExpr>(E)) {
628 PrintOffsetOfDesignator(ASE->getLHS());
629 OS << "[";
630 PrintExpr(ASE->getRHS());
631 OS << "]";
632 return false;
633 } else {
634 MemberExpr *ME = cast<MemberExpr>(E);
635 bool IsFirst = PrintOffsetOfDesignator(ME->getBase());
636 OS << (IsFirst ? "" : ".") << ME->getMemberDecl()->getName();
637 return false;
638 }
639}
640
641void StmtPrinter::VisitUnaryOffsetOf(UnaryOperator *Node) {
642 OS << "__builtin_offsetof(";
643 PrintOffsetOfDesignator(Node->getSubExpr());
644 OS << ")";
645}
646
Chris Lattner4b009652007-07-25 00:24:17 +0000647void StmtPrinter::VisitSizeOfAlignOfTypeExpr(SizeOfAlignOfTypeExpr *Node) {
648 OS << (Node->isSizeOf() ? "sizeof(" : "__alignof(");
649 OS << Node->getArgumentType().getAsString() << ")";
650}
651void StmtPrinter::VisitArraySubscriptExpr(ArraySubscriptExpr *Node) {
Ted Kremenek1c1700f2007-08-20 16:18:38 +0000652 PrintExpr(Node->getLHS());
Chris Lattner4b009652007-07-25 00:24:17 +0000653 OS << "[";
Ted Kremenek1c1700f2007-08-20 16:18:38 +0000654 PrintExpr(Node->getRHS());
Chris Lattner4b009652007-07-25 00:24:17 +0000655 OS << "]";
656}
657
658void StmtPrinter::VisitCallExpr(CallExpr *Call) {
659 PrintExpr(Call->getCallee());
660 OS << "(";
661 for (unsigned i = 0, e = Call->getNumArgs(); i != e; ++i) {
Chris Lattner3e254fb2008-04-08 04:40:51 +0000662 if (isa<CXXDefaultArgExpr>(Call->getArg(i))) {
663 // Don't print any defaulted arguments
664 break;
665 }
666
Chris Lattner4b009652007-07-25 00:24:17 +0000667 if (i) OS << ", ";
668 PrintExpr(Call->getArg(i));
669 }
670 OS << ")";
671}
672void StmtPrinter::VisitMemberExpr(MemberExpr *Node) {
673 PrintExpr(Node->getBase());
674 OS << (Node->isArrow() ? "->" : ".");
675
676 FieldDecl *Field = Node->getMemberDecl();
677 assert(Field && "MemberExpr should alway reference a field!");
678 OS << Field->getName();
679}
Nate Begemanaf6ed502008-04-18 23:10:10 +0000680void StmtPrinter::VisitExtVectorElementExpr(ExtVectorElementExpr *Node) {
Steve Naroffc11705f2007-07-28 23:10:27 +0000681 PrintExpr(Node->getBase());
682 OS << ".";
683 OS << Node->getAccessor().getName();
684}
Chris Lattner4b009652007-07-25 00:24:17 +0000685void StmtPrinter::VisitCastExpr(CastExpr *Node) {
686 OS << "(" << Node->getType().getAsString() << ")";
687 PrintExpr(Node->getSubExpr());
688}
689void StmtPrinter::VisitCompoundLiteralExpr(CompoundLiteralExpr *Node) {
690 OS << "(" << Node->getType().getAsString() << ")";
691 PrintExpr(Node->getInitializer());
692}
693void StmtPrinter::VisitImplicitCastExpr(ImplicitCastExpr *Node) {
694 // No need to print anything, simply forward to the sub expression.
695 PrintExpr(Node->getSubExpr());
696}
697void StmtPrinter::VisitBinaryOperator(BinaryOperator *Node) {
698 PrintExpr(Node->getLHS());
699 OS << " " << BinaryOperator::getOpcodeStr(Node->getOpcode()) << " ";
700 PrintExpr(Node->getRHS());
701}
Chris Lattner06078d22007-08-25 02:00:02 +0000702void StmtPrinter::VisitCompoundAssignOperator(CompoundAssignOperator *Node) {
703 PrintExpr(Node->getLHS());
704 OS << " " << BinaryOperator::getOpcodeStr(Node->getOpcode()) << " ";
705 PrintExpr(Node->getRHS());
706}
Chris Lattner4b009652007-07-25 00:24:17 +0000707void StmtPrinter::VisitConditionalOperator(ConditionalOperator *Node) {
708 PrintExpr(Node->getCond());
Ted Kremenekeecd9e82007-11-26 18:27:54 +0000709
710 if (Node->getLHS()) {
711 OS << " ? ";
712 PrintExpr(Node->getLHS());
713 OS << " : ";
714 }
715 else { // Handle GCC extention where LHS can be NULL.
716 OS << " ?: ";
717 }
718
Chris Lattner4b009652007-07-25 00:24:17 +0000719 PrintExpr(Node->getRHS());
720}
721
722// GNU extensions.
723
Chris Lattnera0d03a72007-08-03 17:31:20 +0000724void StmtPrinter::VisitAddrLabelExpr(AddrLabelExpr *Node) {
Chris Lattner4b009652007-07-25 00:24:17 +0000725 OS << "&&" << Node->getLabel()->getName();
726}
727
728void StmtPrinter::VisitStmtExpr(StmtExpr *E) {
729 OS << "(";
730 PrintRawCompoundStmt(E->getSubStmt());
731 OS << ")";
732}
733
Steve Naroff63bad2d2007-08-01 22:05:33 +0000734void StmtPrinter::VisitTypesCompatibleExpr(TypesCompatibleExpr *Node) {
735 OS << "__builtin_types_compatible_p(";
736 OS << Node->getArgType1().getAsString() << ",";
737 OS << Node->getArgType2().getAsString() << ")";
738}
739
Steve Naroff93c53012007-08-03 21:21:27 +0000740void StmtPrinter::VisitChooseExpr(ChooseExpr *Node) {
741 OS << "__builtin_choose_expr(";
742 PrintExpr(Node->getCond());
Chris Lattner44fcf4f2007-08-04 00:20:15 +0000743 OS << ", ";
Steve Naroff93c53012007-08-03 21:21:27 +0000744 PrintExpr(Node->getLHS());
Chris Lattner44fcf4f2007-08-04 00:20:15 +0000745 OS << ", ";
Steve Naroff93c53012007-08-03 21:21:27 +0000746 PrintExpr(Node->getRHS());
747 OS << ")";
748}
Chris Lattner4b009652007-07-25 00:24:17 +0000749
Nate Begeman9f3bfb72008-01-17 17:46:27 +0000750void StmtPrinter::VisitOverloadExpr(OverloadExpr *Node) {
751 OS << "__builtin_overload(";
Nate Begemanbd881ef2008-01-30 20:50:20 +0000752 for (unsigned i = 0, e = Node->getNumSubExprs(); i != e; ++i) {
Nate Begeman9f3bfb72008-01-17 17:46:27 +0000753 if (i) OS << ", ";
Nate Begemanbd881ef2008-01-30 20:50:20 +0000754 PrintExpr(Node->getExpr(i));
Nate Begeman9f3bfb72008-01-17 17:46:27 +0000755 }
756 OS << ")";
757}
758
Eli Friedmand0e9d092008-05-14 19:38:39 +0000759void StmtPrinter::VisitShuffleVectorExpr(ShuffleVectorExpr *Node) {
760 OS << "__builtin_shufflevector(";
761 for (unsigned i = 0, e = Node->getNumSubExprs(); i != e; ++i) {
762 if (i) OS << ", ";
763 PrintExpr(Node->getExpr(i));
764 }
765 OS << ")";
766}
767
Anders Carlsson762b7c72007-08-31 04:56:16 +0000768void StmtPrinter::VisitInitListExpr(InitListExpr* Node) {
769 OS << "{ ";
770 for (unsigned i = 0, e = Node->getNumInits(); i != e; ++i) {
771 if (i) OS << ", ";
772 PrintExpr(Node->getInit(i));
773 }
774 OS << " }";
775}
776
Anders Carlsson36760332007-10-15 20:28:48 +0000777void StmtPrinter::VisitVAArgExpr(VAArgExpr *Node) {
778 OS << "va_arg(";
779 PrintExpr(Node->getSubExpr());
780 OS << ", ";
781 OS << Node->getType().getAsString();
782 OS << ")";
783}
784
Chris Lattner4b009652007-07-25 00:24:17 +0000785// C++
786
787void StmtPrinter::VisitCXXCastExpr(CXXCastExpr *Node) {
Chris Lattnerac485c12007-08-09 17:34:19 +0000788 OS << CXXCastExpr::getOpcodeStr(Node->getOpcode()) << '<';
Chris Lattner4b009652007-07-25 00:24:17 +0000789 OS << Node->getDestType().getAsString() << ">(";
790 PrintExpr(Node->getSubExpr());
791 OS << ")";
792}
793
794void StmtPrinter::VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *Node) {
795 OS << (Node->getValue() ? "true" : "false");
796}
797
Chris Lattnera7447ba2008-02-26 00:51:44 +0000798void StmtPrinter::VisitCXXThrowExpr(CXXThrowExpr *Node) {
799 if (Node->getSubExpr() == 0)
800 OS << "throw";
801 else {
802 OS << "throw ";
803 PrintExpr(Node->getSubExpr());
804 }
805}
806
Chris Lattner3e254fb2008-04-08 04:40:51 +0000807void StmtPrinter::VisitCXXDefaultArgExpr(CXXDefaultArgExpr *Node) {
808 // Nothing to print: we picked up the default argument
809}
810
Anders Carlssona66cad42007-08-21 17:43:55 +0000811// Obj-C
812
813void StmtPrinter::VisitObjCStringLiteral(ObjCStringLiteral *Node) {
814 OS << "@";
815 VisitStringLiteral(Node->getString());
816}
Chris Lattner4b009652007-07-25 00:24:17 +0000817
Anders Carlsson8be1d402007-08-22 15:14:15 +0000818void StmtPrinter::VisitObjCEncodeExpr(ObjCEncodeExpr *Node) {
Chris Lattnerc17ccab2007-10-18 00:39:29 +0000819 OS << "@encode(" << Node->getEncodedType().getAsString() << ")";
Anders Carlsson8be1d402007-08-22 15:14:15 +0000820}
821
Fariborz Jahanianf807c202007-10-16 20:40:23 +0000822void StmtPrinter::VisitObjCSelectorExpr(ObjCSelectorExpr *Node) {
Chris Lattnerc17ccab2007-10-18 00:39:29 +0000823 OS << "@selector(" << Node->getSelector().getName() << ")";
Fariborz Jahanianf807c202007-10-16 20:40:23 +0000824}
825
Fariborz Jahanianb391e6e2007-10-17 16:58:11 +0000826void StmtPrinter::VisitObjCProtocolExpr(ObjCProtocolExpr *Node) {
Chris Lattnerc17ccab2007-10-18 00:39:29 +0000827 OS << "@protocol(" << Node->getProtocol()->getName() << ")";
Fariborz Jahanianb391e6e2007-10-17 16:58:11 +0000828}
829
Steve Naroffc39ca262007-09-18 23:55:05 +0000830void StmtPrinter::VisitObjCMessageExpr(ObjCMessageExpr *Mess) {
831 OS << "[";
Steve Narofffa465d12007-10-02 20:01:56 +0000832 Expr *receiver = Mess->getReceiver();
833 if (receiver) PrintExpr(receiver);
834 else OS << Mess->getClassName()->getName();
Ted Kremenekbfec9492008-05-02 17:32:38 +0000835 OS << ' ';
Ted Kremenek2287cee2008-04-16 04:30:16 +0000836 Selector selector = Mess->getSelector();
Steve Narofffa465d12007-10-02 20:01:56 +0000837 if (selector.isUnarySelector()) {
Ted Kremenekbfec9492008-05-02 17:32:38 +0000838 OS << selector.getIdentifierInfoForSlot(0)->getName();
Steve Narofffa465d12007-10-02 20:01:56 +0000839 } else {
840 for (unsigned i = 0, e = Mess->getNumArgs(); i != e; ++i) {
Ted Kremenekbfec9492008-05-02 17:32:38 +0000841 if (i < selector.getNumArgs()) {
842 if (i > 0) OS << ' ';
843 if (selector.getIdentifierInfoForSlot(i))
844 OS << selector.getIdentifierInfoForSlot(i)->getName() << ":";
845 else
846 OS << ":";
847 }
848 else OS << ", "; // Handle variadic methods.
849
Steve Narofffa465d12007-10-02 20:01:56 +0000850 PrintExpr(Mess->getArg(i));
851 }
Steve Naroffc39ca262007-09-18 23:55:05 +0000852 }
853 OS << "]";
854}
855
Chris Lattner4b009652007-07-25 00:24:17 +0000856//===----------------------------------------------------------------------===//
857// Stmt method implementations
858//===----------------------------------------------------------------------===//
859
Chris Lattner95578782007-08-08 22:51:59 +0000860void Stmt::dumpPretty() const {
Ted Kremenekdf409eb2007-11-26 22:50:46 +0000861 printPretty(*llvm::cerr.stream());
Chris Lattner4b009652007-07-25 00:24:17 +0000862}
863
Ted Kremenek08176a52007-08-31 21:30:12 +0000864void Stmt::printPretty(std::ostream &OS, PrinterHelper* Helper) const {
Chris Lattner4b009652007-07-25 00:24:17 +0000865 if (this == 0) {
866 OS << "<NULL>";
867 return;
868 }
869
Ted Kremenek08176a52007-08-31 21:30:12 +0000870 StmtPrinter P(OS, Helper);
Chris Lattner7ba21fa2007-08-21 04:04:25 +0000871 P.Visit(const_cast<Stmt*>(this));
Chris Lattner4b009652007-07-25 00:24:17 +0000872}
Ted Kremenek08176a52007-08-31 21:30:12 +0000873
874//===----------------------------------------------------------------------===//
875// PrinterHelper
876//===----------------------------------------------------------------------===//
877
878// Implement virtual destructor.
Gabor Greif61ce98c2007-09-11 15:32:40 +0000879PrinterHelper::~PrinterHelper() {}