| // -*- c-basic-offset: 2 -*- |
| /* |
| * This file is part of the KDE libraries |
| * Copyright (C) 1999-2001 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 Steet, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| * |
| */ |
| |
| #ifndef INTERNAL_H |
| #define INTERNAL_H |
| |
| #include "ustring.h" |
| #include "object.h" |
| #include "protect.h" |
| #include "types.h" |
| #include "interpreter.h" |
| #include "scope_chain.h" |
| #include <kxmlcore/RefPtr.h> |
| |
| #define I18N_NOOP(s) s |
| |
| namespace KJS { |
| |
| class Node; |
| class ProgramNode; |
| class FunctionBodyNode; |
| class FunctionPrototype; |
| class FunctionImp; |
| class Debugger; |
| |
| // --------------------------------------------------------------------------- |
| // Primitive impls |
| // --------------------------------------------------------------------------- |
| |
| class UndefinedImp : public JSCell { |
| public: |
| Type type() const { return UndefinedType; } |
| |
| JSValue *toPrimitive(ExecState *exec, Type preferred = UnspecifiedType) const; |
| bool toBoolean(ExecState *exec) const; |
| double toNumber(ExecState *exec) const; |
| UString toString(ExecState *exec) const; |
| JSObject *toObject(ExecState *exec) const; |
| }; |
| |
| class NullImp : public JSCell { |
| public: |
| Type type() const { return NullType; } |
| |
| JSValue *toPrimitive(ExecState *exec, Type preferred = UnspecifiedType) const; |
| bool toBoolean(ExecState *exec) const; |
| double toNumber(ExecState *exec) const; |
| UString toString(ExecState *exec) const; |
| JSObject *toObject(ExecState *exec) const; |
| }; |
| |
| class BooleanImp : public JSCell { |
| public: |
| BooleanImp(bool v = false) : val(v) { } |
| bool value() const { return val; } |
| |
| Type type() const { return BooleanType; } |
| |
| JSValue *toPrimitive(ExecState *exec, Type preferred = UnspecifiedType) const; |
| bool toBoolean(ExecState *exec) const; |
| double toNumber(ExecState *exec) const; |
| UString toString(ExecState *exec) const; |
| JSObject *toObject(ExecState *exec) const; |
| |
| private: |
| bool val; |
| }; |
| |
| class StringImp : public JSCell { |
| public: |
| StringImp(const UString& v) : val(v) { } |
| UString value() const { return val; } |
| |
| Type type() const { return StringType; } |
| |
| JSValue *toPrimitive(ExecState *exec, Type preferred = UnspecifiedType) const; |
| bool toBoolean(ExecState *exec) const; |
| double toNumber(ExecState *exec) const; |
| UString toString(ExecState *exec) const; |
| JSObject *toObject(ExecState *exec) const; |
| |
| private: |
| UString val; |
| }; |
| |
| class NumberImp : public JSCell { |
| friend class ConstantValues; |
| friend class InterpreterImp; |
| friend JSValue *jsNumber(double); |
| public: |
| double value() const { return val; } |
| |
| Type type() const { return NumberType; } |
| |
| JSValue *toPrimitive(ExecState *exec, Type preferred = UnspecifiedType) const; |
| bool toBoolean(ExecState *exec) const; |
| double toNumber(ExecState *exec) const; |
| UString toString(ExecState *exec) const; |
| JSObject *toObject(ExecState *exec) const; |
| |
| private: |
| NumberImp(double v) : val(v) { } |
| |
| virtual bool getUInt32(uint32_t&) const; |
| |
| double val; |
| }; |
| |
| |
| /** |
| * @short The "label set" in Ecma-262 spec |
| */ |
| class LabelStack { |
| public: |
| LabelStack(): tos(0L), iterationDepth(0), switchDepth(0) {} |
| ~LabelStack(); |
| |
| /** |
| * If id is not empty and is not in the stack already, puts it on top of |
| * the stack and returns true, otherwise returns false |
| */ |
| bool push(const Identifier &id); |
| /** |
| * Is the id in the stack? |
| */ |
| bool contains(const Identifier &id) const; |
| /** |
| * Removes from the stack the last pushed id (what else?) |
| */ |
| void pop(); |
| |
| void pushIteration() { iterationDepth++; } |
| void popIteration() { iterationDepth--; } |
| bool inIteration() const { return (iterationDepth > 0); } |
| |
| void pushSwitch() { switchDepth++; } |
| void popSwitch() { switchDepth--; } |
| bool inSwitch() const { return (switchDepth > 0); } |
| |
| private: |
| LabelStack(const LabelStack &other); |
| LabelStack &operator=(const LabelStack &other); |
| |
| struct StackElem { |
| Identifier id; |
| StackElem *prev; |
| }; |
| |
| StackElem *tos; |
| int iterationDepth; |
| int switchDepth; |
| }; |
| |
| |
| // --------------------------------------------------------------------------- |
| // Parsing & evaluation |
| // --------------------------------------------------------------------------- |
| |
| enum CodeType { GlobalCode, |
| EvalCode, |
| FunctionCode, |
| AnonymousCode }; |
| |
| /** |
| * @internal |
| * |
| * Parses ECMAScript source code and converts into ProgramNode objects, which |
| * represent the root of a parse tree. This class provides a conveniant workaround |
| * for the problem of the bison parser working in a static context. |
| */ |
| class Parser { |
| public: |
| static RefPtr<ProgramNode> parse(const UString &sourceURL, int startingLineNumber, |
| const UChar *code, unsigned int length, int *sourceId = 0, |
| int *errLine = 0, UString *errMsg = 0); |
| static void accept(ProgramNode *prog); |
| |
| static void saveNewNode(Node *node); |
| |
| static int sid; |
| }; |
| |
| class SavedBuiltinsInternal { |
| friend class InterpreterImp; |
| private: |
| ProtectedPtr<JSObject> b_Object; |
| ProtectedPtr<JSObject> b_Function; |
| ProtectedPtr<JSObject> b_Array; |
| ProtectedPtr<JSObject> b_Boolean; |
| ProtectedPtr<JSObject> b_String; |
| ProtectedPtr<JSObject> b_Number; |
| ProtectedPtr<JSObject> b_Date; |
| ProtectedPtr<JSObject> b_RegExp; |
| ProtectedPtr<JSObject> b_Error; |
| |
| ProtectedPtr<JSObject> b_ObjectPrototype; |
| ProtectedPtr<JSObject> b_FunctionPrototype; |
| ProtectedPtr<JSObject> b_ArrayPrototype; |
| ProtectedPtr<JSObject> b_BooleanPrototype; |
| ProtectedPtr<JSObject> b_StringPrototype; |
| ProtectedPtr<JSObject> b_NumberPrototype; |
| ProtectedPtr<JSObject> b_DatePrototype; |
| ProtectedPtr<JSObject> b_RegExpPrototype; |
| ProtectedPtr<JSObject> b_ErrorPrototype; |
| |
| ProtectedPtr<JSObject> b_evalError; |
| ProtectedPtr<JSObject> b_rangeError; |
| ProtectedPtr<JSObject> b_referenceError; |
| ProtectedPtr<JSObject> b_syntaxError; |
| ProtectedPtr<JSObject> b_typeError; |
| ProtectedPtr<JSObject> b_uriError; |
| |
| ProtectedPtr<JSObject> b_evalErrorPrototype; |
| ProtectedPtr<JSObject> b_rangeErrorPrototype; |
| ProtectedPtr<JSObject> b_referenceErrorPrototype; |
| ProtectedPtr<JSObject> b_syntaxErrorPrototype; |
| ProtectedPtr<JSObject> b_typeErrorPrototype; |
| ProtectedPtr<JSObject> b_uriErrorPrototype; |
| }; |
| |
| class InterpreterImp { |
| friend class Collector; |
| public: |
| InterpreterImp(Interpreter *interp, JSObject *glob); |
| ~InterpreterImp(); |
| |
| JSObject *globalObject() { return global; } |
| Interpreter *interpreter() const { return m_interpreter; } |
| |
| void initGlobalObject(); |
| |
| void mark(); |
| |
| ExecState *globalExec() { return &globExec; } |
| bool checkSyntax(const UString &code); |
| Completion evaluate(const UChar* code, int codeLength, JSValue* thisV, const UString& sourceURL, int startingLineNumber); |
| Debugger *debugger() const { return dbg; } |
| void setDebugger(Debugger *d) { dbg = d; } |
| |
| JSObject *builtinObject() const { return b_Object; } |
| JSObject *builtinFunction() const { return b_Function; } |
| JSObject *builtinArray() const { return b_Array; } |
| JSObject *builtinBoolean() const { return b_Boolean; } |
| JSObject *builtinString() const { return b_String; } |
| JSObject *builtinNumber() const { return b_Number; } |
| JSObject *builtinDate() const { return b_Date; } |
| JSObject *builtinRegExp() const { return b_RegExp; } |
| JSObject *builtinError() const { return b_Error; } |
| |
| JSObject *builtinObjectPrototype() const { return b_ObjectPrototype; } |
| JSObject *builtinFunctionPrototype() const { return b_FunctionPrototype; } |
| JSObject *builtinArrayPrototype() const { return b_ArrayPrototype; } |
| JSObject *builtinBooleanPrototype() const { return b_BooleanPrototype; } |
| JSObject *builtinStringPrototype() const { return b_StringPrototype; } |
| JSObject *builtinNumberPrototype() const { return b_NumberPrototype; } |
| JSObject *builtinDatePrototype() const { return b_DatePrototype; } |
| JSObject *builtinRegExpPrototype() const { return b_RegExpPrototype; } |
| JSObject *builtinErrorPrototype() const { return b_ErrorPrototype; } |
| |
| JSObject *builtinEvalError() const { return b_evalError; } |
| JSObject *builtinRangeError() const { return b_rangeError; } |
| JSObject *builtinReferenceError() const { return b_referenceError; } |
| JSObject *builtinSyntaxError() const { return b_syntaxError; } |
| JSObject *builtinTypeError() const { return b_typeError; } |
| JSObject *builtinURIError() const { return b_uriError; } |
| |
| JSObject *builtinEvalErrorPrototype() const { return b_evalErrorPrototype; } |
| JSObject *builtinRangeErrorPrototype() const { return b_rangeErrorPrototype; } |
| JSObject *builtinReferenceErrorPrototype() const { return b_referenceErrorPrototype; } |
| JSObject *builtinSyntaxErrorPrototype() const { return b_syntaxErrorPrototype; } |
| JSObject *builtinTypeErrorPrototype() const { return b_typeErrorPrototype; } |
| JSObject *builtinURIErrorPrototype() const { return b_uriErrorPrototype; } |
| |
| void setCompatMode(Interpreter::CompatMode mode) { m_compatMode = mode; } |
| Interpreter::CompatMode compatMode() const { return m_compatMode; } |
| |
| // Chained list of interpreters (ring) |
| static InterpreterImp* firstInterpreter() { return s_hook; } |
| InterpreterImp *nextInterpreter() const { return next; } |
| InterpreterImp *prevInterpreter() const { return prev; } |
| |
| static InterpreterImp *interpreterWithGlobalObject(JSObject *); |
| |
| void setContext(ContextImp *c) { _context = c; } |
| ContextImp *context() const { return _context; } |
| |
| void saveBuiltins (SavedBuiltins &builtins) const; |
| void restoreBuiltins (const SavedBuiltins &builtins); |
| |
| private: |
| void clear(); |
| Interpreter *m_interpreter; |
| JSObject *global; |
| Debugger *dbg; |
| |
| // Built-in properties of the object prototype. These are accessible |
| // from here even if they are replaced by js code (e.g. assigning to |
| // Array.prototype) |
| |
| ProtectedPtr<JSObject> b_Object; |
| ProtectedPtr<JSObject> b_Function; |
| ProtectedPtr<JSObject> b_Array; |
| ProtectedPtr<JSObject> b_Boolean; |
| ProtectedPtr<JSObject> b_String; |
| ProtectedPtr<JSObject> b_Number; |
| ProtectedPtr<JSObject> b_Date; |
| ProtectedPtr<JSObject> b_RegExp; |
| ProtectedPtr<JSObject> b_Error; |
| |
| ProtectedPtr<JSObject> b_ObjectPrototype; |
| ProtectedPtr<JSObject> b_FunctionPrototype; |
| ProtectedPtr<JSObject> b_ArrayPrototype; |
| ProtectedPtr<JSObject> b_BooleanPrototype; |
| ProtectedPtr<JSObject> b_StringPrototype; |
| ProtectedPtr<JSObject> b_NumberPrototype; |
| ProtectedPtr<JSObject> b_DatePrototype; |
| ProtectedPtr<JSObject> b_RegExpPrototype; |
| ProtectedPtr<JSObject> b_ErrorPrototype; |
| |
| ProtectedPtr<JSObject> b_evalError; |
| ProtectedPtr<JSObject> b_rangeError; |
| ProtectedPtr<JSObject> b_referenceError; |
| ProtectedPtr<JSObject> b_syntaxError; |
| ProtectedPtr<JSObject> b_typeError; |
| ProtectedPtr<JSObject> b_uriError; |
| |
| ProtectedPtr<JSObject> b_evalErrorPrototype; |
| ProtectedPtr<JSObject> b_rangeErrorPrototype; |
| ProtectedPtr<JSObject> b_referenceErrorPrototype; |
| ProtectedPtr<JSObject> b_syntaxErrorPrototype; |
| ProtectedPtr<JSObject> b_typeErrorPrototype; |
| ProtectedPtr<JSObject> b_uriErrorPrototype; |
| |
| ExecState globExec; |
| Interpreter::CompatMode m_compatMode; |
| |
| // Chained list of interpreters (ring) - for collector |
| static InterpreterImp* s_hook; |
| InterpreterImp *next, *prev; |
| |
| ContextImp *_context; |
| |
| int recursion; |
| }; |
| |
| class AttachedInterpreter; |
| class DebuggerImp { |
| public: |
| |
| DebuggerImp() { |
| interps = 0; |
| isAborted = false; |
| } |
| |
| void abort() { isAborted = true; } |
| bool aborted() const { return isAborted; } |
| |
| AttachedInterpreter *interps; |
| bool isAborted; |
| }; |
| |
| |
| |
| class InternalFunctionImp : public JSObject { |
| public: |
| InternalFunctionImp(); |
| InternalFunctionImp(FunctionPrototype *funcProto); |
| bool implementsHasInstance() const; |
| bool hasInstance(ExecState *exec, JSValue *value); |
| |
| virtual const ClassInfo *classInfo() const { return &info; } |
| static const ClassInfo info; |
| }; |
| |
| // helper function for toInteger, toInt32, toUInt32 and toUInt16 |
| double roundValue(ExecState *, JSValue *); |
| |
| #ifndef NDEBUG |
| void printInfo(ExecState *exec, const char *s, JSValue *, int lineno = -1); |
| #endif |
| |
| inline LabelStack::~LabelStack() |
| { |
| StackElem *prev; |
| for (StackElem *e = tos; e; e = prev) { |
| prev = e->prev; |
| delete e; |
| } |
| } |
| |
| inline void LabelStack::pop() |
| { |
| if (StackElem *e = tos) { |
| tos = e->prev; |
| delete e; |
| } |
| } |
| |
| } // namespace |
| |
| #endif // INTERNAL_H |