| //===--- Stmt.cpp - Statement AST Node Implementation ---------------------===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file was developed by Chris Lattner and is distributed under |
| // the University of Illinois Open Source License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file implements the Stmt class and statement subclasses. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "clang/AST/Stmt.h" |
| #include "clang/AST/ExprCXX.h" |
| #include "clang/AST/StmtVisitor.h" |
| #include "clang/Lex/IdentifierTable.h" |
| using namespace clang; |
| |
| static struct StmtClassNameTable { |
| int enumValue; |
| const char *className; |
| unsigned counter; |
| unsigned size; |
| } sNames[] = { |
| #define STMT(N, CLASS, PARENT) { N, #CLASS, 0, sizeof(CLASS) }, |
| #include "clang/AST/StmtNodes.def" |
| { 0, 0, 0, 0 } |
| }; |
| |
| const char *Stmt::getStmtClassName() const { |
| for (int i = 0; sNames[i].className; i++) { |
| if (sClass == sNames[i].enumValue) |
| return sNames[i].className; |
| } |
| return 0; // should never happen.... |
| } |
| |
| void Stmt::PrintStats() { |
| unsigned sum = 0; |
| fprintf(stderr, "*** Stmt/Expr Stats:\n"); |
| for (int i = 0; sNames[i].className; i++) { |
| sum += sNames[i].counter; |
| } |
| fprintf(stderr, " %d stmts/exprs total.\n", sum); |
| sum = 0; |
| for (int i = 0; sNames[i].className; i++) { |
| fprintf(stderr, " %d %s, %d each (%d bytes)\n", |
| sNames[i].counter, sNames[i].className, sNames[i].size, sNames[i].counter*sNames[i].size); |
| sum += sNames[i].counter*sNames[i].size; |
| } |
| fprintf(stderr, "Total bytes = %d\n", sum); |
| } |
| |
| void Stmt::addStmtClass(StmtClass s) { |
| for (int i = 0; sNames[i].className; i++) { |
| if (s == sNames[i].enumValue) |
| sNames[i].counter++; |
| } |
| } |
| |
| static bool StatSwitch = false; |
| |
| bool Stmt::CollectingStats(bool enable) { |
| if (enable) StatSwitch = true; |
| return StatSwitch; |
| } |
| |
| |
| |
| const char *LabelStmt::getName() const { |
| return getID()->getName(); |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // Child Iterators for iterating over subexpressions/substatements |
| //===----------------------------------------------------------------------===// |
| |
| // DeclStmt |
| Stmt::child_iterator DeclStmt::child_begin() { return NULL; } |
| Stmt::child_iterator DeclStmt::child_end() { return NULL; } |
| |
| // NullStmt |
| Stmt::child_iterator NullStmt::child_begin() { return NULL; } |
| Stmt::child_iterator NullStmt::child_end() { return NULL; } |
| |
| // CompoundStmt |
| Stmt::child_iterator CompoundStmt::child_begin() { return &Body[0]; } |
| Stmt::child_iterator CompoundStmt::child_end() { return &Body[0]+Body.size(); } |
| |
| // SwitchCase |
| Stmt::child_iterator SwitchCase::child_begin() { return &SubStmt; } |
| Stmt::child_iterator SwitchCase::child_end() { return child_begin()+1; } |
| |
| // LabelStmt |
| Stmt::child_iterator LabelStmt::child_begin() { return &SubStmt; } |
| Stmt::child_iterator LabelStmt::child_end() { return child_begin()+1; } |
| |
| // IfStmt |
| Stmt::child_iterator IfStmt::child_begin() { return &SubExprs[0]; } |
| Stmt::child_iterator IfStmt::child_end() { return &SubExprs[0]+END_EXPR; } |
| |
| // SwitchStmt |
| Stmt::child_iterator SwitchStmt::child_begin() { return &SubExprs[0]; } |
| Stmt::child_iterator SwitchStmt::child_end() { return &SubExprs[0]+END_EXPR; } |
| |
| // WhileStmt |
| Stmt::child_iterator WhileStmt::child_begin() { return &SubExprs[0]; } |
| Stmt::child_iterator WhileStmt::child_end() { return &SubExprs[0]+END_EXPR; } |
| |
| // DoStmt |
| Stmt::child_iterator DoStmt::child_begin() { return &SubExprs[0]; } |
| Stmt::child_iterator DoStmt::child_end() { return &SubExprs[0]+END_EXPR; } |
| |
| // ForStmt |
| Stmt::child_iterator ForStmt::child_begin() { return &SubExprs[0]; } |
| Stmt::child_iterator ForStmt::child_end() { return &SubExprs[0]+END_EXPR; } |
| |
| // GotoStmt |
| Stmt::child_iterator GotoStmt::child_begin() { return NULL; } |
| Stmt::child_iterator GotoStmt::child_end() { return NULL; } |
| |
| // IndirectGotoStmt |
| Stmt::child_iterator IndirectGotoStmt::child_begin() { |
| return reinterpret_cast<Stmt**>(&Target); |
| } |
| |
| Stmt::child_iterator IndirectGotoStmt::child_end() { return child_begin()+1; } |
| |
| // ContinueStmt |
| Stmt::child_iterator ContinueStmt::child_begin() { return NULL; } |
| Stmt::child_iterator ContinueStmt::child_end() { return NULL; } |
| |
| // BreakStmt |
| Stmt::child_iterator BreakStmt::child_begin() { return NULL; } |
| Stmt::child_iterator BreakStmt::child_end() { return NULL; } |
| |
| // ReturnStmt |
| Stmt::child_iterator ReturnStmt::child_begin() { |
| return reinterpret_cast<Stmt**>(&RetExpr); |
| } |
| |
| Stmt::child_iterator ReturnStmt::child_end() { return child_begin()+1; } |
| |