a563800b4c
implementation is not yet complete (it prints only the node name for some AST constructs) and does not serialize everything needed to reconstitute the AST. It is motivated by a desire to prototype source-to-source transformations in JavaScript itself (or anything else that can grok JSON), but it should have other uses too. Feedback is welcome. Review URL: http://codereview.chromium.org/131101 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@3051 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
220 lines
7.4 KiB
C++
220 lines
7.4 KiB
C++
// Copyright 2006-2008 the V8 project authors. All rights reserved.
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following
|
|
// disclaimer in the documentation and/or other materials provided
|
|
// with the distribution.
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
// contributors may be used to endorse or promote products derived
|
|
// from this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
#ifndef V8_PRETTYPRINTER_H_
|
|
#define V8_PRETTYPRINTER_H_
|
|
|
|
#include "ast.h"
|
|
|
|
namespace v8 {
|
|
namespace internal {
|
|
|
|
#ifdef DEBUG
|
|
|
|
class PrettyPrinter: public AstVisitor {
|
|
public:
|
|
PrettyPrinter();
|
|
virtual ~PrettyPrinter();
|
|
|
|
// The following routines print a node into a string.
|
|
// The result string is alive as long as the PrettyPrinter is alive.
|
|
const char* Print(AstNode* node);
|
|
const char* PrintExpression(FunctionLiteral* program);
|
|
const char* PrintProgram(FunctionLiteral* program);
|
|
|
|
void Print(const char* format, ...);
|
|
|
|
// Print a node to stdout.
|
|
static void PrintOut(AstNode* node);
|
|
|
|
// Individual nodes
|
|
#define DECLARE_VISIT(type) virtual void Visit##type(type* node);
|
|
AST_NODE_LIST(DECLARE_VISIT)
|
|
#undef DECLARE_VISIT
|
|
|
|
private:
|
|
char* output_; // output string buffer
|
|
int size_; // output_ size
|
|
int pos_; // current printing position
|
|
|
|
protected:
|
|
void Init();
|
|
const char* Output() const { return output_; }
|
|
|
|
virtual void PrintStatements(ZoneList<Statement*>* statements);
|
|
void PrintLabels(ZoneStringList* labels);
|
|
virtual void PrintArguments(ZoneList<Expression*>* arguments);
|
|
void PrintLiteral(Handle<Object> value, bool quote);
|
|
void PrintParameters(Scope* scope);
|
|
void PrintDeclarations(ZoneList<Declaration*>* declarations);
|
|
void PrintFunctionLiteral(FunctionLiteral* function);
|
|
void PrintCaseClause(CaseClause* clause);
|
|
};
|
|
|
|
|
|
// Prints the AST structure
|
|
class AstPrinter: public PrettyPrinter {
|
|
public:
|
|
AstPrinter();
|
|
virtual ~AstPrinter();
|
|
|
|
const char* PrintProgram(FunctionLiteral* program);
|
|
|
|
// Individual nodes
|
|
#define DECLARE_VISIT(type) virtual void Visit##type(type* node);
|
|
AST_NODE_LIST(DECLARE_VISIT)
|
|
#undef DECLARE_VISIT
|
|
private:
|
|
friend class IndentedScope;
|
|
void PrintIndented(const char* txt);
|
|
void PrintIndentedVisit(const char* s, AstNode* node);
|
|
|
|
void PrintStatements(ZoneList<Statement*>* statements);
|
|
void PrintDeclarations(ZoneList<Declaration*>* declarations);
|
|
void PrintParameters(Scope* scope);
|
|
void PrintArguments(ZoneList<Expression*>* arguments);
|
|
void PrintCaseClause(CaseClause* clause);
|
|
void PrintLiteralIndented(const char* info, Handle<Object> value, bool quote);
|
|
void PrintLiteralWithModeIndented(const char* info,
|
|
Variable* var,
|
|
Handle<Object> value,
|
|
SmiAnalysis* type);
|
|
void PrintLabelsIndented(const char* info, ZoneStringList* labels);
|
|
|
|
void inc_indent() { indent_++; }
|
|
void dec_indent() { indent_--; }
|
|
|
|
static int indent_;
|
|
};
|
|
|
|
|
|
// Forward declaration of helper classes.
|
|
class TagScope;
|
|
class AttributesScope;
|
|
|
|
// Build a C string containing a JSON representation of a function's
|
|
// AST. The representation is based on JsonML (www.jsonml.org).
|
|
class JsonAstBuilder: public PrettyPrinter {
|
|
public:
|
|
JsonAstBuilder()
|
|
: indent_(0), top_tag_scope_(NULL), attributes_scope_(NULL) {
|
|
}
|
|
virtual ~JsonAstBuilder() {}
|
|
|
|
// Controls the indentation of subsequent lines of a tag body after
|
|
// the first line.
|
|
static const int kTagIndentSize = 2;
|
|
|
|
// Controls the indentation of subsequent lines of an attributes
|
|
// blocks's body after the first line.
|
|
static const int kAttributesIndentSize = 1;
|
|
|
|
// Construct a JSON representation of a function literal.
|
|
const char* BuildProgram(FunctionLiteral* program);
|
|
|
|
// Print text indented by the current indentation level.
|
|
void PrintIndented(const char* text) { Print("%*s%s", indent_, "", text); }
|
|
|
|
// Change the indentation level.
|
|
void increase_indent(int amount) { indent_ += amount; }
|
|
void decrease_indent(int amount) { indent_ -= amount; }
|
|
|
|
// The builder maintains a stack of opened AST node constructors.
|
|
// Each node constructor corresponds to a JsonML tag.
|
|
TagScope* tag() { return top_tag_scope_; }
|
|
void set_tag(TagScope* scope) { top_tag_scope_ = scope; }
|
|
|
|
// The builder maintains a pointer to the currently opened attributes
|
|
// of current AST node or NULL if the attributes are not opened.
|
|
AttributesScope* attributes() { return attributes_scope_; }
|
|
void set_attributes(AttributesScope* scope) { attributes_scope_ = scope; }
|
|
|
|
// Add an attribute to the currently opened attributes.
|
|
void AddAttribute(const char* name, Handle<String> value);
|
|
void AddAttribute(const char* name, const char* value);
|
|
void AddAttribute(const char* name, int value);
|
|
void AddAttribute(const char* name, bool value);
|
|
|
|
// AST node visit functions.
|
|
#define DECLARE_VISIT(type) virtual void Visit##type(type* node);
|
|
AST_NODE_LIST(DECLARE_VISIT)
|
|
#undef DECLARE_VISIT
|
|
|
|
private:
|
|
int indent_;
|
|
TagScope* top_tag_scope_;
|
|
AttributesScope* attributes_scope_;
|
|
|
|
// Utility function used by AddAttribute implementations.
|
|
void AddAttributePrefix(const char* name);
|
|
};
|
|
|
|
|
|
// The JSON AST builder keeps a stack of open element tags (AST node
|
|
// constructors from the current iteration point to the root of the
|
|
// AST). TagScope is a helper class to manage the opening and closing
|
|
// of tags, the indentation of their bodies, and comma separating their
|
|
// contents.
|
|
class TagScope BASE_EMBEDDED {
|
|
public:
|
|
TagScope(JsonAstBuilder* builder, const char* name);
|
|
~TagScope();
|
|
|
|
void use() { has_body_ = true; }
|
|
|
|
private:
|
|
JsonAstBuilder* builder_;
|
|
TagScope* next_;
|
|
bool has_body_;
|
|
};
|
|
|
|
|
|
// AttributesScope is a helper class to manage the opening and closing
|
|
// of attribute blocks, the indentation of their bodies, and comma
|
|
// separating their contents. JsonAstBuilder::AddAttribute adds an
|
|
// attribute to the currently open AttributesScope. They cannot be
|
|
// nested so the builder keeps an optional single scope rather than a
|
|
// stack.
|
|
class AttributesScope BASE_EMBEDDED {
|
|
public:
|
|
explicit AttributesScope(JsonAstBuilder* builder);
|
|
~AttributesScope();
|
|
|
|
bool is_used() { return attribute_count_ > 0; }
|
|
void use() { ++attribute_count_; }
|
|
|
|
private:
|
|
JsonAstBuilder* builder_;
|
|
int attribute_count_;
|
|
};
|
|
|
|
#endif // DEBUG
|
|
|
|
} } // namespace v8::internal
|
|
|
|
#endif // V8_PRETTYPRINTER_H_
|