blob: 7ed761948db0045343a0a1decd0ffa11536d53b0 [file] [log] [blame]
Chris Lattnera3bcb7a2006-11-04 07:16:25 +00001//===--- StmtPrinter.cpp - Printing implementation for Stmt ASTs ----------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file was developed by Chris Lattner and is distributed under
6// the University of Illinois Open Source License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the Stmt::dump/Stmt::print methods.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/AST/StmtVisitor.h"
Chris Lattner5efbb332006-11-20 05:01:40 +000015#include "clang/AST/Decl.h"
Chris Lattner29375652006-12-04 18:06:35 +000016#include "clang/AST/ExprCXX.h"
Chris Lattner6c0ff132006-11-05 00:19:50 +000017#include "clang/Lex/IdentifierTable.h"
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000018#include "llvm/Support/Compiler.h"
19#include <iostream>
20using namespace llvm;
21using namespace clang;
22
Chris Lattner882f7882006-11-04 18:52:07 +000023namespace {
24 struct VISIBILITY_HIDDEN IsExprStmtVisitor : public StmtVisitor {
25 bool &Result;
26 IsExprStmtVisitor(bool &R) : Result(R) { Result = false; }
27
28 virtual void VisitExpr(Expr *Node) {
29 Result = true;
30 }
31 };
32}
33
34static bool isExpr(Stmt *S) {
35 bool Val = false;
36 IsExprStmtVisitor V(Val);
37 S->visit(V);
38 return Val;
39}
40
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000041//===----------------------------------------------------------------------===//
42// StmtPrinter Visitor
43//===----------------------------------------------------------------------===//
44
45namespace {
46 class VISIBILITY_HIDDEN StmtPrinter : public StmtVisitor {
47 std::ostream &OS;
48 unsigned IndentLevel;
49 public:
50 StmtPrinter(std::ostream &os) : OS(os), IndentLevel(0) {}
51
Chris Lattnerb9eb5a12007-05-20 22:52:15 +000052 void PrintStmt(Stmt *S, int SubIndent = 1) {
53 IndentLevel += SubIndent;
Chris Lattner882f7882006-11-04 18:52:07 +000054 if (S && isExpr(S)) {
55 // If this is an expr used in a stmt context, indent and newline it.
56 Indent();
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000057 S->visit(*this);
Chris Lattnerfc068c12007-05-30 17:57:36 +000058 OS << ";\n";
Chris Lattner882f7882006-11-04 18:52:07 +000059 } else if (S) {
60 S->visit(*this);
61 } else {
Chris Lattner2f6ac262007-05-28 01:47:47 +000062 Indent() << "<<<NULL STATEMENT>>>\n";
Chris Lattner882f7882006-11-04 18:52:07 +000063 }
Chris Lattnerb9eb5a12007-05-20 22:52:15 +000064 IndentLevel -= SubIndent;
Chris Lattner882f7882006-11-04 18:52:07 +000065 }
Chris Lattner073926e2007-05-20 23:04:55 +000066
67 void PrintRawCompoundStmt(CompoundStmt *S);
Chris Lattner882f7882006-11-04 18:52:07 +000068
69 void PrintExpr(Expr *E) {
70 if (E)
71 E->visit(*this);
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000072 else
Chris Lattner882f7882006-11-04 18:52:07 +000073 OS << "<null expr>";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000074 }
75
Chris Lattnerb9eb5a12007-05-20 22:52:15 +000076 std::ostream &Indent(int Delta = 0) const {
Chris Lattnerd5322cd2007-05-29 23:49:07 +000077 for (int i = 0, e = IndentLevel+Delta; i < e; ++i)
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000078 OS << " ";
79 return OS;
80 }
81
Chris Lattner882f7882006-11-04 18:52:07 +000082 virtual void VisitStmt(Stmt *Node);
Steve Naroff7f890eb2007-02-27 02:53:10 +000083#define STMT(N, CLASS, PARENT) \
Chris Lattnerf2174b62006-11-04 20:59:27 +000084 virtual void Visit##CLASS(CLASS *Node);
85#include "clang/AST/StmtNodes.def"
Chris Lattner71e23ce2006-11-04 20:18:38 +000086 };
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000087}
88
Chris Lattner71e23ce2006-11-04 20:18:38 +000089//===----------------------------------------------------------------------===//
90// Stmt printing methods.
91//===----------------------------------------------------------------------===//
92
Chris Lattner882f7882006-11-04 18:52:07 +000093void StmtPrinter::VisitStmt(Stmt *Node) {
94 Indent() << "<<unknown stmt type>>\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000095}
96
Chris Lattner073926e2007-05-20 23:04:55 +000097/// PrintRawCompoundStmt - Print a compound stmt without indenting the {, and
98/// with no newline after the }.
99void StmtPrinter::PrintRawCompoundStmt(CompoundStmt *Node) {
100 OS << "{\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000101 for (CompoundStmt::body_iterator I = Node->body_begin(), E = Node->body_end();
Chris Lattner882f7882006-11-04 18:52:07 +0000102 I != E; ++I)
103 PrintStmt(*I);
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000104
Chris Lattner073926e2007-05-20 23:04:55 +0000105 Indent() << "}";
106}
107
Chris Lattner012a6cf2007-05-28 01:41:58 +0000108void StmtPrinter::VisitNullStmt(NullStmt *Node) {
109 Indent() << ";\n";
110}
111
Steve Naroff2a8ad182007-05-29 22:59:26 +0000112void StmtPrinter::VisitDeclStmt(DeclStmt *Node) {
113 // FIXME: Need to complete/beautify this...this code simply shows the
114 // nodes are where they need to be.
115 if (BlockVarDecl *localVar = dyn_cast<BlockVarDecl>(Node->getDecl())) {
116 Indent() << localVar->getType().getAsString();
117 OS << " " << localVar->getName() << ";\n";
118 } else if (TypedefDecl *localType = dyn_cast<TypedefDecl>(Node->getDecl())) {
119 Indent() << "typedef " << localType->getUnderlyingType().getAsString();
120 OS << " " << localType->getName() << ";\n";
Steve Naroff9992bba2007-05-30 16:27:15 +0000121 } else
122 assert(0 && "Unexpected decl (expecting BlockVarDecl or TypedefDecl");
Steve Naroff2a8ad182007-05-29 22:59:26 +0000123}
124
Chris Lattner073926e2007-05-20 23:04:55 +0000125void StmtPrinter::VisitCompoundStmt(CompoundStmt *Node) {
126 Indent();
127 PrintRawCompoundStmt(Node);
Chris Lattnerdf3cafb2007-05-31 05:08:56 +0000128 OS << "\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000129}
130
Chris Lattner6c0ff132006-11-05 00:19:50 +0000131void StmtPrinter::VisitCaseStmt(CaseStmt *Node) {
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000132 Indent(-1) << "case ";
Chris Lattner6c0ff132006-11-05 00:19:50 +0000133 PrintExpr(Node->getLHS());
134 if (Node->getRHS()) {
135 OS << " ... ";
136 PrintExpr(Node->getRHS());
137 }
138 OS << ":\n";
139
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000140 PrintStmt(Node->getSubStmt(), 0);
Chris Lattner6c0ff132006-11-05 00:19:50 +0000141}
142
143void StmtPrinter::VisitDefaultStmt(DefaultStmt *Node) {
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000144 Indent(-1) << "default:\n";
145 PrintStmt(Node->getSubStmt(), 0);
Chris Lattner6c0ff132006-11-05 00:19:50 +0000146}
147
148void StmtPrinter::VisitLabelStmt(LabelStmt *Node) {
Chris Lattnereefa10e2007-05-28 06:56:27 +0000149 Indent(-1) << Node->getName() << ":\n";
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000150 PrintStmt(Node->getSubStmt(), 0);
Chris Lattner6c0ff132006-11-05 00:19:50 +0000151}
152
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000153void StmtPrinter::VisitIfStmt(IfStmt *If) {
Chris Lattner85ed8732006-11-04 20:40:44 +0000154 Indent() << "if (";
Chris Lattner882f7882006-11-04 18:52:07 +0000155 PrintExpr(If->getCond());
Chris Lattner073926e2007-05-20 23:04:55 +0000156 OS << ')';
157
158 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(If->getThen())) {
159 OS << ' ';
160 PrintRawCompoundStmt(CS);
161 OS << (If->getElse() ? ' ' : '\n');
162 } else {
163 OS << '\n';
164 PrintStmt(If->getThen());
165 if (If->getElse()) Indent();
166 }
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000167
Chris Lattner073926e2007-05-20 23:04:55 +0000168 if (Stmt *Else = If->getElse()) {
169 OS << "else";
170
171 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Else)) {
172 OS << ' ';
173 PrintRawCompoundStmt(CS);
174 OS << '\n';
175 } else {
176 OS << '\n';
177 PrintStmt(If->getElse());
178 }
Chris Lattner882f7882006-11-04 18:52:07 +0000179 }
Chris Lattner85ed8732006-11-04 20:40:44 +0000180}
181
Chris Lattnerf2174b62006-11-04 20:59:27 +0000182void StmtPrinter::VisitSwitchStmt(SwitchStmt *Node) {
183 Indent() << "switch (";
184 PrintExpr(Node->getCond());
Chris Lattner073926e2007-05-20 23:04:55 +0000185 OS << ")";
186
187 // Pretty print compoundstmt bodies (very common).
188 if (CompoundStmt *CS = dyn_cast<CompoundStmt>(Node->getBody())) {
189 OS << " ";
190 PrintRawCompoundStmt(CS);
191 OS << "\n";
192 } else {
193 OS << "\n";
194 PrintStmt(Node->getBody());
195 }
Chris Lattnerf2174b62006-11-04 20:59:27 +0000196}
197
Chris Lattner85ed8732006-11-04 20:40:44 +0000198void StmtPrinter::VisitWhileStmt(WhileStmt *Node) {
199 Indent() << "while (";
200 PrintExpr(Node->getCond());
201 OS << ")\n";
202 PrintStmt(Node->getBody());
203}
204
205void StmtPrinter::VisitDoStmt(DoStmt *Node) {
206 Indent() << "do\n";
207 PrintStmt(Node->getBody());
208 Indent() << "while (";
209 PrintExpr(Node->getCond());
Chris Lattnerb4619482007-05-31 06:00:14 +0000210 OS << ");\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000211}
212
Chris Lattner71e23ce2006-11-04 20:18:38 +0000213void StmtPrinter::VisitForStmt(ForStmt *Node) {
214 Indent() << "for (";
215 if (Node->getFirst())
216 PrintExpr((Expr*)Node->getFirst());
217 OS << "; ";
218 if (Node->getSecond())
219 PrintExpr(Node->getSecond());
220 OS << "; ";
221 if (Node->getThird())
222 PrintExpr(Node->getThird());
223 OS << ")\n";
Chris Lattner85ed8732006-11-04 20:40:44 +0000224 PrintStmt(Node->getBody());
Chris Lattner71e23ce2006-11-04 20:18:38 +0000225}
226
Chris Lattner16976d32006-11-05 01:46:01 +0000227void StmtPrinter::VisitGotoStmt(GotoStmt *Node) {
Chris Lattnereefa10e2007-05-28 06:56:27 +0000228 Indent() << "goto " << Node->getLabel()->getName() << ";\n";
Chris Lattner16976d32006-11-05 01:46:01 +0000229}
230
231void StmtPrinter::VisitIndirectGotoStmt(IndirectGotoStmt *Node) {
Chris Lattner36ad1232006-11-05 01:51:06 +0000232 Indent() << "goto *";
Chris Lattner16976d32006-11-05 01:46:01 +0000233 PrintExpr(Node->getTarget());
Chris Lattnerb4619482007-05-31 06:00:14 +0000234 OS << ";\n";
Chris Lattner16976d32006-11-05 01:46:01 +0000235}
236
237void StmtPrinter::VisitContinueStmt(ContinueStmt *Node) {
Chris Lattnerb4619482007-05-31 06:00:14 +0000238 Indent() << "continue;\n";
Chris Lattner16976d32006-11-05 01:46:01 +0000239}
240
241void StmtPrinter::VisitBreakStmt(BreakStmt *Node) {
Chris Lattnerb4619482007-05-31 06:00:14 +0000242 Indent() << "break;\n";
Chris Lattner16976d32006-11-05 01:46:01 +0000243}
244
245
Chris Lattner882f7882006-11-04 18:52:07 +0000246void StmtPrinter::VisitReturnStmt(ReturnStmt *Node) {
247 Indent() << "return";
248 if (Node->getRetValue()) {
249 OS << " ";
250 PrintExpr(Node->getRetValue());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000251 }
Chris Lattnerb4619482007-05-31 06:00:14 +0000252 OS << ";\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000253}
254
Chris Lattner71e23ce2006-11-04 20:18:38 +0000255//===----------------------------------------------------------------------===//
256// Expr printing methods.
257//===----------------------------------------------------------------------===//
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000258
Chris Lattner882f7882006-11-04 18:52:07 +0000259void StmtPrinter::VisitExpr(Expr *Node) {
260 OS << "<<unknown expr type>>";
261}
262
263void StmtPrinter::VisitDeclRefExpr(DeclRefExpr *Node) {
Chris Lattner5efbb332006-11-20 05:01:40 +0000264 OS << Node->getDecl()->getName();
Chris Lattner882f7882006-11-04 18:52:07 +0000265}
266
Steve Naroffae4143e2007-04-26 20:39:23 +0000267void StmtPrinter::VisitCharacterLiteral(CharacterLiteral *Node) {
268 // FIXME: print value.
269 OS << "x";
270}
271
Steve Naroffdf7855b2007-02-21 23:46:25 +0000272void StmtPrinter::VisitIntegerLiteral(IntegerLiteral *Node) {
Chris Lattner06430412007-05-21 05:45:03 +0000273 bool isSigned = Node->getType()->isSignedIntegerType();
274 OS << Node->getValue().toString(10, isSigned);
275
276 // Emit suffixes. Integer literals are always a builtin integer type.
277 switch (cast<BuiltinType>(Node->getType().getCanonicalType())->getKind()) {
278 default: assert(0 && "Unexpected type for integer literal!");
279 case BuiltinType::Int: break; // no suffix.
280 case BuiltinType::UInt: OS << 'U'; break;
281 case BuiltinType::Long: OS << 'L'; break;
282 case BuiltinType::ULong: OS << "UL"; break;
283 case BuiltinType::LongLong: OS << "LL"; break;
284 case BuiltinType::ULongLong: OS << "ULL"; break;
285 }
Chris Lattner882f7882006-11-04 18:52:07 +0000286}
Steve Naroffab624882007-02-21 22:05:47 +0000287void StmtPrinter::VisitFloatingLiteral(FloatingLiteral *Node) {
Chris Lattner882f7882006-11-04 18:52:07 +0000288 // FIXME: print value.
Chris Lattner06430412007-05-21 05:45:03 +0000289 OS << "~1.0~";
Chris Lattner882f7882006-11-04 18:52:07 +0000290}
Steve Naroffdf7855b2007-02-21 23:46:25 +0000291void StmtPrinter::VisitStringLiteral(StringLiteral *Str) {
Chris Lattner882f7882006-11-04 18:52:07 +0000292 if (Str->isWide()) OS << 'L';
Chris Lattner5d8f4942006-11-04 20:29:31 +0000293 OS << '"';
294
295 // FIXME: this doesn't print wstrings right.
296 for (unsigned i = 0, e = Str->getByteLength(); i != e; ++i) {
297 switch (Str->getStrData()[i]) {
298 default: OS << Str->getStrData()[i]; break;
299 // Handle some common ones to make dumps prettier.
300 case '\\': OS << "\\\\"; break;
301 case '"': OS << "\\\""; break;
302 case '\n': OS << "\\n"; break;
303 case '\t': OS << "\\t"; break;
304 case '\a': OS << "\\a"; break;
305 case '\b': OS << "\\b"; break;
306 }
307 }
308 OS << '"';
Chris Lattner882f7882006-11-04 18:52:07 +0000309}
310void StmtPrinter::VisitParenExpr(ParenExpr *Node) {
311 OS << "(";
312 PrintExpr(Node->getSubExpr());
313 OS << ")'";
314}
315void StmtPrinter::VisitUnaryOperator(UnaryOperator *Node) {
Chris Lattner15768702006-11-05 23:54:51 +0000316 if (!Node->isPostfix())
317 OS << UnaryOperator::getOpcodeStr(Node->getOpcode());
Chris Lattner882f7882006-11-04 18:52:07 +0000318 PrintExpr(Node->getSubExpr());
Chris Lattner15768702006-11-05 23:54:51 +0000319
320 if (Node->isPostfix())
321 OS << UnaryOperator::getOpcodeStr(Node->getOpcode());
322
Chris Lattner882f7882006-11-04 18:52:07 +0000323}
324void StmtPrinter::VisitSizeOfAlignOfTypeExpr(SizeOfAlignOfTypeExpr *Node) {
Chris Lattner33e8a552006-11-19 01:48:02 +0000325 OS << (Node->isSizeOf() ? "sizeof(" : "__alignof(");
Chris Lattner3dc3d772007-05-16 18:07:12 +0000326 OS << Node->getArgumentType().getAsString() << ")";
Chris Lattner882f7882006-11-04 18:52:07 +0000327}
328void StmtPrinter::VisitArraySubscriptExpr(ArraySubscriptExpr *Node) {
329 PrintExpr(Node->getBase());
330 OS << "[";
331 PrintExpr(Node->getIdx());
332 OS << "]";
333}
334
335void StmtPrinter::VisitCallExpr(CallExpr *Call) {
336 PrintExpr(Call->getCallee());
337 OS << "(";
338 for (unsigned i = 0, e = Call->getNumArgs(); i != e; ++i) {
339 if (i) OS << ", ";
340 PrintExpr(Call->getArg(i));
341 }
342 OS << ")";
343}
344void StmtPrinter::VisitMemberExpr(MemberExpr *Node) {
345 PrintExpr(Node->getBase());
346 OS << (Node->isArrow() ? "->" : ".");
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000347
Chris Lattner24e0d6c2007-05-24 00:47:01 +0000348 FieldDecl *Field = Node->getMemberDecl();
349 assert(Field && "MemberExpr should alway reference a field!");
350 OS << Field->getName();
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000351}
Chris Lattner882f7882006-11-04 18:52:07 +0000352void StmtPrinter::VisitCastExpr(CastExpr *Node) {
Chris Lattner3dc3d772007-05-16 18:07:12 +0000353 OS << "(" << Node->getDestType().getAsString() << ")";
Chris Lattner882f7882006-11-04 18:52:07 +0000354 PrintExpr(Node->getSubExpr());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000355}
Chris Lattner882f7882006-11-04 18:52:07 +0000356void StmtPrinter::VisitBinaryOperator(BinaryOperator *Node) {
357 PrintExpr(Node->getLHS());
358 OS << " " << BinaryOperator::getOpcodeStr(Node->getOpcode()) << " ";
359 PrintExpr(Node->getRHS());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000360}
Chris Lattner882f7882006-11-04 18:52:07 +0000361void StmtPrinter::VisitConditionalOperator(ConditionalOperator *Node) {
362 PrintExpr(Node->getCond());
363 OS << " ? ";
364 PrintExpr(Node->getLHS());
Bill Wendling48fbdd02007-05-23 08:04:21 +0000365 OS << " : ";
Chris Lattner882f7882006-11-04 18:52:07 +0000366 PrintExpr(Node->getRHS());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000367}
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000368
Chris Lattnereefa10e2007-05-28 06:56:27 +0000369// GNU extensions.
370
371void StmtPrinter::VisitAddrLabel(AddrLabel *Node) {
372 OS << "&&" << Node->getLabel()->getName();
373
374}
375
376// C++
377
378void StmtPrinter::VisitCXXCastExpr(CXXCastExpr *Node) {
379 switch (Node->getOpcode()) {
380 default:
381 assert(0 && "Not a C++ cast expression");
382 abort();
383 case CXXCastExpr::ConstCast: OS << "const_cast<"; break;
384 case CXXCastExpr::DynamicCast: OS << "dynamic_cast<"; break;
385 case CXXCastExpr::ReinterpretCast: OS << "reinterpret_cast<"; break;
386 case CXXCastExpr::StaticCast: OS << "static_cast<"; break;
387 }
388
389 OS << Node->getDestType().getAsString() << ">(";
390 PrintExpr(Node->getSubExpr());
391 OS << ")";
392}
393
394void StmtPrinter::VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *Node) {
395 OS << (Node->getValue() ? "true" : "false");
396}
397
398
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000399//===----------------------------------------------------------------------===//
400// Stmt method implementations
401//===----------------------------------------------------------------------===//
402
403void Stmt::dump() const {
404 print(std::cerr);
405}
406
407void Stmt::print(std::ostream &OS) const {
Chris Lattner882f7882006-11-04 18:52:07 +0000408 if (this == 0) {
409 OS << "<NULL>";
410 return;
411 }
412
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000413 StmtPrinter P(OS);
414 const_cast<Stmt*>(this)->visit(P);
415}