// -*- c-basic-offset: 2 -*- /* * This file is part of the KDE libraries * Copyright (C) 1999-2000, 2003 Harri Porten (porten@kde.org) * Copyright (C) 2001 Peter Kelly (pmk@post.com) * Copyright (C) 2003 Apple Computer, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */ #ifndef _NODES_H_ #define _NODES_H_ #include "internal.h" //#include "debugger.h" #ifndef NDEBUG #include <list> #include <assert.h> #endif namespace KJS { class RegExp; class SourceElementsNode; class ObjectLiteralNode; class PropertyNode; class SourceStream; class PropertyValueNode; class PropertyNode; enum Operator { OpEqual, OpEqEq, OpNotEq, OpStrEq, OpStrNEq, OpPlusEq, OpMinusEq, OpMultEq, OpDivEq, OpPlusPlus, OpMinusMinus, OpLess, OpLessEq, OpGreater, OpGreaterEq, OpAndEq, OpXOrEq, OpOrEq, OpModEq, OpAnd, OpOr, OpBitAnd, OpBitXOr, OpBitOr, OpLShift, OpRShift, OpURShift, OpIn, OpInstanceOf }; class Node { public: Node(); virtual ~Node(); // reusing Value Type here, declare new enum if required virtual Type type() const { return UnspecifiedType; } /** * Evaluate this node and return the result, possibly a reference. */ virtual Reference evaluateReference(ExecState *exec) const; /** * Returns the value represented by this node. Always dereferenced. */ virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual UString toString(ExecState *exec) const; UString toCode() const; virtual void streamTo(SourceStream &s) const = 0; virtual void processVarDecls(ExecState* /*exec*/) {} int lineNo() const { return line; } public: // reference counting mechanism virtual void ref() { refcount++; } #ifdef KJS_DEBUG_MEM virtual bool deref() { assert( refcount > 0 ); return (!--refcount); } #else virtual bool deref() { return (!--refcount); } #endif #ifdef KJS_DEBUG_MEM static void finalCheck(); #endif protected: Value throwError(ExecState *exec, ErrorType e, const char *msg) const; Value throwError(ExecState *exec, ErrorType e, const char *msg, const Value &v, const Node *expr) const; Value throwError(ExecState *exec, ErrorType e, const char *msg, Identifier label) const; void setExceptionDetailsIfNeeded(ExecState *exec) const; int line; unsigned int refcount; virtual int sourceId() const { return -1; } private: #ifdef KJS_DEBUG_MEM // List of all nodes, for debugging purposes. Don't remove! static std::list<Node *> *s_nodes; #endif // disallow assignment Node& operator=(const Node&); Node(const Node &other); }; class StatementNode : public Node { public: StatementNode(); virtual ~StatementNode(); void setLoc(int line0, int line1, SourceCode *src); int firstLine() const { return l0; } int lastLine() const { return l1; } int sourceId() const { return sourceCode->sid; } SourceCode *code() const { return sourceCode; } bool hitStatement(ExecState *exec); bool abortStatement(ExecState *exec); virtual Completion execute(ExecState *exec) = 0; void pushLabel(const Identifier &id) { ls.push(id); } virtual void processFuncDecl(ExecState *exec); protected: LabelStack ls; private: Reference evaluateReference(ExecState* /*exec*/) const { return Reference(0,Identifier::null()); } int l0, l1; SourceCode *sourceCode; bool breakPoint; }; class NullNode : public Node { public: NullNode() {} virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual UString toString(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; }; class BooleanNode : public Node { public: BooleanNode(bool v) : val(v) {} virtual Type type() const { return BooleanType; } virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual UString toString(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: bool val; }; class NumberNode : public Node { public: NumberNode(double v) : val(v) { } virtual Type type() const { return NumberType; } virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual UString toString(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: double val; }; class StringNode : public Node { public: StringNode(const UString *v) : val(*v) { } virtual Type type() const { return StringType; } virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual UString toString(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: UString val; }; class RegExpNode : public Node { public: RegExpNode(const UString &p, const UString &f) : pattern(p), flags(f) { } virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: UString pattern, flags; }; class ThisNode : public Node { public: ThisNode() {} virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; }; class ResolveNode : public Node { public: ResolveNode(const Identifier &s) : ident(s) { } Reference evaluateReference(ExecState *exec) const; virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Identifier ident; }; class GroupNode : public Node { public: GroupNode(Node *g) : group(g) { } virtual void ref(); virtual bool deref(); Reference evaluateReference(ExecState *exec) const; virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *group; }; class ElementNode : public Node { public: // list is circular during construction. cracked in ArrayNode ctor ElementNode(int e, Node *n) : list(this), elision(e), node(n) { } ElementNode(ElementNode *l, int e, Node *n) : list(l->list), elision(e), node(n) { l->list = this; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: friend class ArrayNode; ElementNode *list; int elision; Node *node; }; class ArrayNode : public Node { public: ArrayNode(int e) : element(0L), elision(e), opt(true) { } ArrayNode(ElementNode *ele) : element(ele->list), elision(0), opt(false) { ele->list = 0; } ArrayNode(int eli, ElementNode *ele) : element(ele->list), elision(eli), opt(true) { ele->list = 0; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: ElementNode *element; int elision; bool opt; }; class PropertyValueNode : public Node { public: // list is circular during construction, cut in ObjectLiteralNode ctor PropertyValueNode(PropertyNode *n, Node *a) : name(n), assign(a), list(this) { } PropertyValueNode(PropertyNode *n, Node *a, PropertyValueNode *l) : name(n), assign(a), list(l->list) { l->list = this; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: friend class ObjectLiteralNode; PropertyNode *name; Node *assign; PropertyValueNode *list; }; class PropertyNode : public Node { public: PropertyNode(double d) : numeric(d) { } PropertyNode(const Identifier &s) : str(s) { } virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: double numeric; Identifier str; }; class ObjectLiteralNode : public Node { public: // empty literal ObjectLiteralNode() : list(0) { } // l points to last list element, get and detach pointer to first one ObjectLiteralNode(PropertyValueNode *l) : list(l->list) { l->list = 0; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: PropertyValueNode *list; }; class AccessorNode1 : public Node { public: AccessorNode1(Node *e1, Node *e2) : expr1(e1), expr2(e2) {} virtual void ref(); virtual bool deref(); Reference evaluateReference(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr1; Node *expr2; }; class AccessorNode2 : public Node { public: AccessorNode2(Node *e, const Identifier &s) : expr(e), ident(s) { } virtual void ref(); virtual bool deref(); Reference evaluateReference(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; Identifier ident; }; class ArgumentListNode : public Node { public: // list is circular during construction. cracked in ArgumentsNode ctor ArgumentListNode(Node *e) : list(this), expr(e) {} ArgumentListNode(ArgumentListNode *l, Node *e) : list(l->list), expr(e) { l->list = this; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; List evaluateList(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: friend class ArgumentsNode; ArgumentListNode *list; Node *expr; }; class ArgumentsNode : public Node { public: ArgumentsNode() : list(0) {} ArgumentsNode(ArgumentListNode *l) : list(l->list) { l->list = 0; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; List evaluateList(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: ArgumentListNode *list; }; class NewExprNode : public Node { public: NewExprNode(Node *e) : expr(e), args(0L) {} NewExprNode(Node *e, ArgumentsNode *a) : expr(e), args(a) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; ArgumentsNode *args; }; class FunctionCallNode : public Node { public: FunctionCallNode(Node *e, ArgumentsNode *a) : expr(e), args(a) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; ArgumentsNode *args; }; class PostfixNode : public Node { public: PostfixNode(Node *e, Operator o) : expr(e), oper(o) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; Operator oper; }; class DeleteNode : public Node { public: DeleteNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class VoidNode : public Node { public: VoidNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class TypeOfNode : public Node { public: TypeOfNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class PrefixNode : public Node { public: PrefixNode(Operator o, Node *e) : oper(o), expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Operator oper; Node *expr; }; class UnaryPlusNode : public Node { public: UnaryPlusNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class NegateNode : public Node { public: NegateNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual double toNumber(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class BitwiseNotNode : public Node { public: BitwiseNotNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class LogicalNotNode : public Node { public: LogicalNotNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual bool toBoolean(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class MultNode : public Node { public: MultNode(Node *t1, Node *t2, char op) : term1(t1), term2(t2), oper(op) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *term1, *term2; char oper; }; class AddNode : public Node { public: AddNode(Node *t1, Node *t2, char op) : term1(t1), term2(t2), oper(op) {} static Node* create(Node *t1, Node *t2, char op); virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *term1, *term2; char oper; }; class AppendStringNode : public Node { public: AppendStringNode(Node *t, const UString &s) : term(t), str(s) { } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *term; UString str; }; class ShiftNode : public Node { public: ShiftNode(Node *t1, Operator o, Node *t2) : term1(t1), term2(t2), oper(o) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *term1, *term2; Operator oper; }; class RelationalNode : public Node { public: RelationalNode(Node *e1, Operator o, Node *e2) : expr1(e1), expr2(e2), oper(o) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr1, *expr2; Operator oper; }; class EqualNode : public Node { public: EqualNode(Node *e1, Operator o, Node *e2) : expr1(e1), expr2(e2), oper(o) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr1, *expr2; Operator oper; }; class BitOperNode : public Node { public: BitOperNode(Node *e1, Operator o, Node *e2) : expr1(e1), expr2(e2), oper(o) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr1, *expr2; Operator oper; }; /** * expr1 && expr2, expr1 || expr2 */ class BinaryLogicalNode : public Node { public: BinaryLogicalNode(Node *e1, Operator o, Node *e2) : expr1(e1), expr2(e2), oper(o) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr1, *expr2; Operator oper; }; /** * The ternary operator, "logical ? expr1 : expr2" */ class ConditionalNode : public Node { public: ConditionalNode(Node *l, Node *e1, Node *e2) : logical(l), expr1(e1), expr2(e2) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *logical, *expr1, *expr2; }; class AssignNode : public Node { public: AssignNode(Node *l, Operator o, Node *e) : left(l), oper(o), expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *left; Operator oper; Node *expr; }; class CommaNode : public Node { public: CommaNode(Node *e1, Node *e2) : expr1(e1), expr2(e2) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr1, *expr2; }; class StatListNode : public StatementNode { public: // list is circular during construction. cracked in CaseClauseNode ctor StatListNode(StatementNode *s); StatListNode(StatListNode *l, StatementNode *s); virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: friend class CaseClauseNode; StatementNode *statement; StatListNode *list; }; class AssignExprNode : public Node { public: AssignExprNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class VarDeclNode : public Node { public: enum Type { Variable, Constant }; VarDeclNode(const Identifier &id, AssignExprNode *in, Type t); virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Type varType; Identifier ident; AssignExprNode *init; }; class VarDeclListNode : public Node { public: // list pointer is tail of a circular list, cracked in the ForNode/VarStatementNode ctor VarDeclListNode(VarDeclNode *v) : list(this), var(v) {} VarDeclListNode(VarDeclListNode *l, VarDeclNode *v) : list(l->list), var(v) { l->list = this; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: friend class ForNode; friend class VarStatementNode; VarDeclListNode *list; VarDeclNode *var; }; class VarStatementNode : public StatementNode { public: VarStatementNode(VarDeclListNode *l) : list(l->list) { l->list = 0; } virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: VarDeclListNode *list; }; class BlockNode : public StatementNode { public: BlockNode(SourceElementsNode *s); virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; protected: SourceElementsNode *source; }; class EmptyStatementNode : public StatementNode { public: EmptyStatementNode() { } // debug virtual Completion execute(ExecState *exec); virtual void streamTo(SourceStream &s) const; }; class ExprStatementNode : public StatementNode { public: ExprStatementNode(Node *e) : expr(e) { } virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class IfNode : public StatementNode { public: IfNode(Node *e, StatementNode *s1, StatementNode *s2) : expr(e), statement1(s1), statement2(s2) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; StatementNode *statement1, *statement2; }; class DoWhileNode : public StatementNode { public: DoWhileNode(StatementNode *s, Node *e) : statement(s), expr(e) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: StatementNode *statement; Node *expr; }; class WhileNode : public StatementNode { public: WhileNode(Node *e, StatementNode *s) : expr(e), statement(s) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; StatementNode *statement; }; class ForNode : public StatementNode { public: ForNode(Node *e1, Node *e2, Node *e3, StatementNode *s) : expr1(e1), expr2(e2), expr3(e3), statement(s) {} ForNode(VarDeclListNode *e1, Node *e2, Node *e3, StatementNode *s) : expr1(e1->list), expr2(e2), expr3(e3), statement(s) { e1->list = 0; } virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr1, *expr2, *expr3; StatementNode *statement; }; class ForInNode : public StatementNode { public: ForInNode(Node *l, Node *e, StatementNode *s); ForInNode(const Identifier &i, AssignExprNode *in, Node *e, StatementNode *s); virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Identifier ident; AssignExprNode *init; Node *lexpr, *expr; VarDeclNode *varDecl; StatementNode *statement; }; class ContinueNode : public StatementNode { public: ContinueNode() { } ContinueNode(const Identifier &i) : ident(i) { } virtual Completion execute(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Identifier ident; }; class BreakNode : public StatementNode { public: BreakNode() { } BreakNode(const Identifier &i) : ident(i) { } virtual Completion execute(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Identifier ident; }; class ReturnNode : public StatementNode { public: ReturnNode(Node *v) : value(v) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *value; }; class WithNode : public StatementNode { public: WithNode(Node *e, StatementNode *s) : expr(e), statement(s) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; StatementNode *statement; }; class CaseClauseNode : public Node { public: CaseClauseNode(Node *e) : expr(e), list(0) { } CaseClauseNode(Node *e, StatListNode *l) : expr(e), list(l->list) { l->list = 0; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; Completion evalStatements(ExecState *exec) const; virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; StatListNode *list; }; class ClauseListNode : public Node { public: // list is circular during construction. cracked in CaseBlockNode ctor ClauseListNode(CaseClauseNode *c) : cl(c), nx(this) { } ClauseListNode(ClauseListNode *n, CaseClauseNode *c) : cl(c), nx(n->nx) { n->nx = this; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; CaseClauseNode *clause() const { return cl; } ClauseListNode *next() const { return nx; } virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: friend class CaseBlockNode; CaseClauseNode *cl; ClauseListNode *nx; }; class CaseBlockNode: public Node { public: CaseBlockNode(ClauseListNode *l1, CaseClauseNode *d, ClauseListNode *l2); virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; Completion evalBlock(ExecState *exec, const Value& input) const; virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: ClauseListNode *list1; CaseClauseNode *def; ClauseListNode *list2; }; class SwitchNode : public StatementNode { public: SwitchNode(Node *e, CaseBlockNode *b) : expr(e), block(b) { } virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; CaseBlockNode *block; }; class LabelNode : public StatementNode { public: LabelNode(const Identifier &l, StatementNode *s) : label(l), statement(s) { } virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Identifier label; StatementNode *statement; }; class ThrowNode : public StatementNode { public: ThrowNode(Node *e) : expr(e) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Node *expr; }; class CatchNode : public StatementNode { public: CatchNode(const Identifier &i, StatementNode *b) : ident(i), block(b) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); Completion execute(ExecState *exec, const Value &arg); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Identifier ident; StatementNode *block; }; class FinallyNode : public StatementNode { public: FinallyNode(StatementNode *b) : block(b) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: StatementNode *block; }; class TryNode : public StatementNode { public: TryNode(StatementNode *b, CatchNode *c) : block(b), _catch(c), _final(0) {} TryNode(StatementNode *b, FinallyNode *f) : block(b), _catch(0), _final(f) {} TryNode(StatementNode *b, CatchNode *c, FinallyNode *f) : block(b), _catch(c), _final(f) {} virtual void ref(); virtual bool deref(); virtual Completion execute(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: StatementNode *block; CatchNode *_catch; FinallyNode *_final; }; class ParameterNode : public Node { public: // list is circular during construction. cracked in FuncDecl/ExprNode ctor. ParameterNode(const Identifier &i) : id(i), next(this) { } ParameterNode(ParameterNode *list, const Identifier &i) : id(i), next(list->next) { list->next = this; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; Identifier ident() const { return id; } ParameterNode *nextParam() const { return next; } virtual void streamTo(SourceStream &s) const; private: friend class FuncDeclNode; friend class FuncExprNode; Identifier id; ParameterNode *next; }; // inherited by ProgramNode class FunctionBodyNode : public BlockNode { public: FunctionBodyNode(SourceElementsNode *s); virtual void processFuncDecl(ExecState *exec); }; class FuncDeclNode : public StatementNode { public: FuncDeclNode(const Identifier &i, FunctionBodyNode *b) : ident(i), param(0), body(b) { } FuncDeclNode(const Identifier &i, ParameterNode *p, FunctionBodyNode *b) : ident(i), param(p->next), body(b) { p->next = 0; } virtual void ref(); virtual bool deref(); Completion execute(ExecState* /*exec*/) { /* empty */ return Completion(); } void processFuncDecl(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: Identifier ident; ParameterNode *param; FunctionBodyNode *body; }; class FuncExprNode : public Node { public: FuncExprNode(const Identifier &i, FunctionBodyNode *b) : ident(i), param(0), body(b) { } FuncExprNode(const Identifier &i, ParameterNode *p, FunctionBodyNode *b) : ident(i), param(p->next), body(b) { p->next = 0; } virtual void ref(); virtual bool deref(); virtual Value evaluate(ExecState *exec) const; virtual void streamTo(SourceStream &s) const; private: Identifier ident; ParameterNode *param; FunctionBodyNode *body; }; // A linked list of source element nodes class SourceElementsNode : public StatementNode { public: // list is circular until cracked in BlockNode (or subclass) ctor SourceElementsNode(StatementNode *s1); SourceElementsNode(SourceElementsNode *s1, StatementNode *s2); virtual void ref(); virtual bool deref(); Completion execute(ExecState *exec); virtual void processFuncDecl(ExecState *exec); virtual void processVarDecls(ExecState *exec); virtual void streamTo(SourceStream &s) const; private: friend class BlockNode; StatementNode *element; // 'this' element SourceElementsNode *elements; // pointer to next }; } // namespace #endif