blob: acec1f510443b382bcadb87445132e98db2e2ca4 [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 Lattner882f7882006-11-04 18:52:07 +000058 OS << "\n";
59 } else if (S) {
60 S->visit(*this);
61 } else {
Chris Lattner85ed8732006-11-04 20:40:44 +000062 Indent() << ";\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 }
66
67 void PrintExpr(Expr *E) {
68 if (E)
69 E->visit(*this);
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000070 else
Chris Lattner882f7882006-11-04 18:52:07 +000071 OS << "<null expr>";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000072 }
73
Chris Lattnerb9eb5a12007-05-20 22:52:15 +000074 std::ostream &Indent(int Delta = 0) const {
75 for (unsigned i = 0, e = IndentLevel+Delta; i != e; ++i)
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000076 OS << " ";
77 return OS;
78 }
79
Chris Lattner882f7882006-11-04 18:52:07 +000080 virtual void VisitStmt(Stmt *Node);
Steve Naroff7f890eb2007-02-27 02:53:10 +000081#define STMT(N, CLASS, PARENT) \
Chris Lattnerf2174b62006-11-04 20:59:27 +000082 virtual void Visit##CLASS(CLASS *Node);
83#include "clang/AST/StmtNodes.def"
Chris Lattner71e23ce2006-11-04 20:18:38 +000084 };
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000085}
86
Chris Lattner71e23ce2006-11-04 20:18:38 +000087//===----------------------------------------------------------------------===//
88// Stmt printing methods.
89//===----------------------------------------------------------------------===//
90
Chris Lattner882f7882006-11-04 18:52:07 +000091void StmtPrinter::VisitStmt(Stmt *Node) {
92 Indent() << "<<unknown stmt type>>\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000093}
94
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000095void StmtPrinter::VisitCompoundStmt(CompoundStmt *Node) {
96 Indent() << "{\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +000097
98 for (CompoundStmt::body_iterator I = Node->body_begin(), E = Node->body_end();
Chris Lattner882f7882006-11-04 18:52:07 +000099 I != E; ++I)
100 PrintStmt(*I);
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000101
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000102 Indent() << "}\n";
103}
104
Chris Lattner6c0ff132006-11-05 00:19:50 +0000105void StmtPrinter::VisitCaseStmt(CaseStmt *Node) {
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000106 Indent(-1) << "case ";
Chris Lattner6c0ff132006-11-05 00:19:50 +0000107 PrintExpr(Node->getLHS());
108 if (Node->getRHS()) {
109 OS << " ... ";
110 PrintExpr(Node->getRHS());
111 }
112 OS << ":\n";
113
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000114 PrintStmt(Node->getSubStmt(), 0);
Chris Lattner6c0ff132006-11-05 00:19:50 +0000115}
116
117void StmtPrinter::VisitDefaultStmt(DefaultStmt *Node) {
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000118 Indent(-1) << "default:\n";
119 PrintStmt(Node->getSubStmt(), 0);
Chris Lattner6c0ff132006-11-05 00:19:50 +0000120}
121
122void StmtPrinter::VisitLabelStmt(LabelStmt *Node) {
Chris Lattnerb9eb5a12007-05-20 22:52:15 +0000123 Indent(-1) << Node->getLabel()->getName() << ":\n";
124 PrintStmt(Node->getSubStmt(), 0);
Chris Lattner6c0ff132006-11-05 00:19:50 +0000125}
126
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000127void StmtPrinter::VisitIfStmt(IfStmt *If) {
Chris Lattner85ed8732006-11-04 20:40:44 +0000128 Indent() << "if (";
Chris Lattner882f7882006-11-04 18:52:07 +0000129 PrintExpr(If->getCond());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000130
Chris Lattner85ed8732006-11-04 20:40:44 +0000131 OS << ")\n";
Chris Lattner882f7882006-11-04 18:52:07 +0000132 PrintStmt(If->getThen());
133 if (If->getElse()) {
134 Indent() << "else\n";
135 PrintStmt(If->getElse());
136 }
Chris Lattner85ed8732006-11-04 20:40:44 +0000137}
138
Chris Lattnerf2174b62006-11-04 20:59:27 +0000139void StmtPrinter::VisitSwitchStmt(SwitchStmt *Node) {
140 Indent() << "switch (";
141 PrintExpr(Node->getCond());
142 OS << ")\n";
143 PrintStmt(Node->getBody());
144}
145
Chris Lattner85ed8732006-11-04 20:40:44 +0000146void StmtPrinter::VisitWhileStmt(WhileStmt *Node) {
147 Indent() << "while (";
148 PrintExpr(Node->getCond());
149 OS << ")\n";
150 PrintStmt(Node->getBody());
151}
152
153void StmtPrinter::VisitDoStmt(DoStmt *Node) {
154 Indent() << "do\n";
155 PrintStmt(Node->getBody());
156 Indent() << "while (";
157 PrintExpr(Node->getCond());
158 OS << ")\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000159}
160
Chris Lattner71e23ce2006-11-04 20:18:38 +0000161void StmtPrinter::VisitForStmt(ForStmt *Node) {
162 Indent() << "for (";
163 if (Node->getFirst())
164 PrintExpr((Expr*)Node->getFirst());
165 OS << "; ";
166 if (Node->getSecond())
167 PrintExpr(Node->getSecond());
168 OS << "; ";
169 if (Node->getThird())
170 PrintExpr(Node->getThird());
171 OS << ")\n";
Chris Lattner85ed8732006-11-04 20:40:44 +0000172 PrintStmt(Node->getBody());
Chris Lattner71e23ce2006-11-04 20:18:38 +0000173}
174
Chris Lattner16976d32006-11-05 01:46:01 +0000175void StmtPrinter::VisitGotoStmt(GotoStmt *Node) {
176 Indent() << "goto " << Node->getLabel()->getName() << "\n";
177}
178
179void StmtPrinter::VisitIndirectGotoStmt(IndirectGotoStmt *Node) {
Chris Lattner36ad1232006-11-05 01:51:06 +0000180 Indent() << "goto *";
Chris Lattner16976d32006-11-05 01:46:01 +0000181 PrintExpr(Node->getTarget());
182 OS << "\n";
183}
184
185void StmtPrinter::VisitContinueStmt(ContinueStmt *Node) {
186 Indent() << "continue\n";
187}
188
189void StmtPrinter::VisitBreakStmt(BreakStmt *Node) {
190 Indent() << "break\n";
191}
192
193
Chris Lattner882f7882006-11-04 18:52:07 +0000194void StmtPrinter::VisitReturnStmt(ReturnStmt *Node) {
195 Indent() << "return";
196 if (Node->getRetValue()) {
197 OS << " ";
198 PrintExpr(Node->getRetValue());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000199 }
Chris Lattner882f7882006-11-04 18:52:07 +0000200 OS << "\n";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000201}
202
Chris Lattner71e23ce2006-11-04 20:18:38 +0000203//===----------------------------------------------------------------------===//
204// Expr printing methods.
205//===----------------------------------------------------------------------===//
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000206
Chris Lattner882f7882006-11-04 18:52:07 +0000207void StmtPrinter::VisitExpr(Expr *Node) {
208 OS << "<<unknown expr type>>";
209}
210
211void StmtPrinter::VisitDeclRefExpr(DeclRefExpr *Node) {
Chris Lattner5efbb332006-11-20 05:01:40 +0000212 OS << Node->getDecl()->getName();
Chris Lattner882f7882006-11-04 18:52:07 +0000213}
214
Steve Naroffae4143e2007-04-26 20:39:23 +0000215void StmtPrinter::VisitCharacterLiteral(CharacterLiteral *Node) {
216 // FIXME: print value.
217 OS << "x";
218}
219
Steve Naroffdf7855b2007-02-21 23:46:25 +0000220void StmtPrinter::VisitIntegerLiteral(IntegerLiteral *Node) {
Chris Lattner882f7882006-11-04 18:52:07 +0000221 // FIXME: print value.
222 OS << "1";
223}
Steve Naroffab624882007-02-21 22:05:47 +0000224void StmtPrinter::VisitFloatingLiteral(FloatingLiteral *Node) {
Chris Lattner882f7882006-11-04 18:52:07 +0000225 // FIXME: print value.
226 OS << "1.0";
227}
Steve Naroffdf7855b2007-02-21 23:46:25 +0000228void StmtPrinter::VisitStringLiteral(StringLiteral *Str) {
Chris Lattner882f7882006-11-04 18:52:07 +0000229 if (Str->isWide()) OS << 'L';
Chris Lattner5d8f4942006-11-04 20:29:31 +0000230 OS << '"';
231
232 // FIXME: this doesn't print wstrings right.
233 for (unsigned i = 0, e = Str->getByteLength(); i != e; ++i) {
234 switch (Str->getStrData()[i]) {
235 default: OS << Str->getStrData()[i]; break;
236 // Handle some common ones to make dumps prettier.
237 case '\\': OS << "\\\\"; break;
238 case '"': OS << "\\\""; break;
239 case '\n': OS << "\\n"; break;
240 case '\t': OS << "\\t"; break;
241 case '\a': OS << "\\a"; break;
242 case '\b': OS << "\\b"; break;
243 }
244 }
245 OS << '"';
Chris Lattner882f7882006-11-04 18:52:07 +0000246}
247void StmtPrinter::VisitParenExpr(ParenExpr *Node) {
248 OS << "(";
249 PrintExpr(Node->getSubExpr());
250 OS << ")'";
251}
252void StmtPrinter::VisitUnaryOperator(UnaryOperator *Node) {
Chris Lattner15768702006-11-05 23:54:51 +0000253 if (!Node->isPostfix())
254 OS << UnaryOperator::getOpcodeStr(Node->getOpcode());
Chris Lattner882f7882006-11-04 18:52:07 +0000255 PrintExpr(Node->getSubExpr());
Chris Lattner15768702006-11-05 23:54:51 +0000256
257 if (Node->isPostfix())
258 OS << UnaryOperator::getOpcodeStr(Node->getOpcode());
259
Chris Lattner882f7882006-11-04 18:52:07 +0000260}
261void StmtPrinter::VisitSizeOfAlignOfTypeExpr(SizeOfAlignOfTypeExpr *Node) {
Chris Lattner33e8a552006-11-19 01:48:02 +0000262 OS << (Node->isSizeOf() ? "sizeof(" : "__alignof(");
Chris Lattner3dc3d772007-05-16 18:07:12 +0000263 OS << Node->getArgumentType().getAsString() << ")";
Chris Lattner882f7882006-11-04 18:52:07 +0000264}
265void StmtPrinter::VisitArraySubscriptExpr(ArraySubscriptExpr *Node) {
266 PrintExpr(Node->getBase());
267 OS << "[";
268 PrintExpr(Node->getIdx());
269 OS << "]";
270}
271
272void StmtPrinter::VisitCallExpr(CallExpr *Call) {
273 PrintExpr(Call->getCallee());
274 OS << "(";
275 for (unsigned i = 0, e = Call->getNumArgs(); i != e; ++i) {
276 if (i) OS << ", ";
277 PrintExpr(Call->getArg(i));
278 }
279 OS << ")";
280}
281void StmtPrinter::VisitMemberExpr(MemberExpr *Node) {
282 PrintExpr(Node->getBase());
283 OS << (Node->isArrow() ? "->" : ".");
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000284
Chris Lattner882f7882006-11-04 18:52:07 +0000285 if (Node->getMemberDecl())
286 assert(0 && "TODO: should print member decl!");
287 OS << "member";
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000288}
Chris Lattner882f7882006-11-04 18:52:07 +0000289void StmtPrinter::VisitCastExpr(CastExpr *Node) {
Chris Lattner3dc3d772007-05-16 18:07:12 +0000290 OS << "(" << Node->getDestType().getAsString() << ")";
Chris Lattner882f7882006-11-04 18:52:07 +0000291 PrintExpr(Node->getSubExpr());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000292}
Chris Lattner29375652006-12-04 18:06:35 +0000293void StmtPrinter::VisitCXXCastExpr(CXXCastExpr *Node) {
294 switch (Node->getOpcode()) {
295 default:
296 assert(0 && "Not a C++ cast expression");
297 abort();
298 case CXXCastExpr::ConstCast: OS << "const_cast<"; break;
299 case CXXCastExpr::DynamicCast: OS << "dynamic_cast<"; break;
300 case CXXCastExpr::ReinterpretCast: OS << "reinterpret_cast<"; break;
301 case CXXCastExpr::StaticCast: OS << "static_cast<"; break;
302 }
303
Chris Lattner3dc3d772007-05-16 18:07:12 +0000304 OS << Node->getDestType().getAsString() << ">(";
Chris Lattner29375652006-12-04 18:06:35 +0000305 PrintExpr(Node->getSubExpr());
306 OS << ")";
307}
Steve Naroff7f890eb2007-02-27 02:53:10 +0000308
309void StmtPrinter::VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *Node) {
310 assert(0 && "TODO: should print CXXBoolLiteralExpr!");
311}
312
Chris Lattner882f7882006-11-04 18:52:07 +0000313void StmtPrinter::VisitBinaryOperator(BinaryOperator *Node) {
314 PrintExpr(Node->getLHS());
315 OS << " " << BinaryOperator::getOpcodeStr(Node->getOpcode()) << " ";
316 PrintExpr(Node->getRHS());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000317}
Chris Lattner882f7882006-11-04 18:52:07 +0000318void StmtPrinter::VisitConditionalOperator(ConditionalOperator *Node) {
319 PrintExpr(Node->getCond());
320 OS << " ? ";
321 PrintExpr(Node->getLHS());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000322 std::cerr << " : ";
Chris Lattner882f7882006-11-04 18:52:07 +0000323 PrintExpr(Node->getRHS());
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000324}
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000325
326//===----------------------------------------------------------------------===//
327// Stmt method implementations
328//===----------------------------------------------------------------------===//
329
330void Stmt::dump() const {
331 print(std::cerr);
332}
333
334void Stmt::print(std::ostream &OS) const {
Chris Lattner882f7882006-11-04 18:52:07 +0000335 if (this == 0) {
336 OS << "<NULL>";
337 return;
338 }
339
Chris Lattnera3bcb7a2006-11-04 07:16:25 +0000340 StmtPrinter P(OS);
341 const_cast<Stmt*>(this)->visit(P);
342}