blob: 09e1235403ba4c1e79e177535d3c1616734fa3ac [file] [log] [blame]
// -*- 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