blob: 46547fb7a19acddd7b7124cba96099bc68ad6734 [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, 2004, 2005, 2006 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 KJS_OBJECT_H
#define KJS_OBJECT_H
#include "JSType.h"
#include "CommonIdentifiers.h"
#include "interpreter.h"
#include "property_map.h"
#include "property_slot.h"
#include "scope_chain.h"
#include <wtf/AlwaysInline.h>
namespace KJS {
class InternalFunctionImp;
class PropertyNameArray;
struct HashEntry;
struct HashTable;
// ECMA 262-3 8.6.1
// Property attributes
enum Attribute { None = 0,
ReadOnly = 1 << 1, // property can be only read, not written
DontEnum = 1 << 2, // property doesn't appear in (for .. in ..)
DontDelete = 1 << 3, // property can't be deleted
Internal = 1 << 4, // an internal property, set to bypass checks
Function = 1 << 5, // property is a function - only used by static hashtables
GetterSetter = 1 << 6 }; // property is a getter or setter
/**
* Class Information
*/
struct ClassInfo {
/**
* A string denoting the class name. Example: "Window".
*/
const char* className;
/**
* Pointer to the class information of the base class.
* 0L if there is none.
*/
const ClassInfo *parentClass;
/**
* Static hash-table of properties.
*/
const HashTable *propHashTable;
/**
* Reserved for future extension.
*/
void *dummy;
};
// This is an internal value object which stores getter and setter functions
// for a property.
class GetterSetterImp : public JSCell {
public:
JSType type() const { return GetterSetterType; }
GetterSetterImp() : getter(0), setter(0) { }
virtual JSValue *toPrimitive(ExecState *exec, JSType preferred = UnspecifiedType) const;
virtual bool toBoolean(ExecState *exec) const;
virtual double toNumber(ExecState *exec) const;
virtual UString toString(ExecState *exec) const;
virtual JSObject *toObject(ExecState *exec) const;
virtual void mark();
JSObject *getGetter() { return getter; }
void setGetter(JSObject *g) { getter = g; }
JSObject *getSetter() { return setter; }
void setSetter(JSObject *s) { setter = s; }
private:
JSObject *getter;
JSObject *setter;
};
class JSObject : public JSCell {
public:
/**
* Creates a new JSObject with the specified prototype
*
* @param proto The prototype
*/
JSObject(JSValue* proto);
/**
* Creates a new JSObject with a prototype of jsNull()
* (that is, the ECMAScript "null" value, not a null object pointer).
*/
JSObject();
virtual void mark();
virtual JSType type() const;
/**
* A pointer to a ClassInfo struct for this class. This provides a basic
* facility for run-time type information, and can be used to check an
* object's class an inheritance (see inherits()). This should
* always return a statically declared pointer, or 0 to indicate that
* there is no class information.
*
* This is primarily useful if you have application-defined classes that you
* wish to check against for casting purposes.
*
* For example, to specify the class info for classes FooImp and BarImp,
* where FooImp inherits from BarImp, you would add the following in your
* class declarations:
*
* \code
* class BarImp : public JSObject {
* virtual const ClassInfo *classInfo() const { return &info; }
* static const ClassInfo info;
* // ...
* };
*
* class FooImp : public JSObject {
* virtual const ClassInfo *classInfo() const { return &info; }
* static const ClassInfo info;
* // ...
* };
* \endcode
*
* And in your source file:
*
* \code
* const ClassInfo BarImp::info = {"Bar", 0, 0, 0}; // no parent class
* const ClassInfo FooImp::info = {"Foo", &BarImp::info, 0, 0};
* \endcode
*
* @see inherits()
*/
virtual const ClassInfo *classInfo() const;
/**
* Checks whether this object inherits from the class with the specified
* classInfo() pointer. This requires that both this class and the other
* class return a non-NULL pointer for their classInfo() methods (otherwise
* it will return false).
*
* For example, for two JSObject pointers obj1 and obj2, you can check
* if obj1's class inherits from obj2's class using the following:
*
* if (obj1->inherits(obj2->classInfo())) {
* // ...
* }
*
* If you have a handle to a statically declared ClassInfo, such as in the
* classInfo() example, you can check for inheritance without needing
* an instance of the other class:
*
* if (obj1->inherits(FooImp::info)) {
* // ...
* }
*
* @param cinfo The ClassInfo pointer for the class you want to check
* inheritance against.
* @return true if this object's class inherits from class with the
* ClassInfo pointer specified in cinfo
*/
bool inherits(const ClassInfo *cinfo) const;
// internal properties (ECMA 262-3 8.6.2)
/**
* Returns the prototype of this object. Note that this is not the same as
* the "prototype" property.
*
* See ECMA 8.6.2
*
* @return The object's prototype
*/
JSValue *prototype() const;
void setPrototype(JSValue *proto);
/**
* Returns the class name of the object
*
* See ECMA 8.6.2
*
* @return The object's class name
*/
/**
* Implementation of the [[Class]] internal property (implemented by all
* Objects)
*
* The default implementation uses classInfo().
* You should either implement classInfo(), or
* if you simply need a classname, you can reimplement className()
* instead.
*/
virtual UString className() const;
/**
* Retrieves the specified property from the object. If neither the object
* or any other object in it's prototype chain have the property, this
* function will return Undefined.
*
* See ECMA 8.6.2.1
*
* @param exec The current execution state
* @param propertyName The name of the property to retrieve
*
* @return The specified property, or Undefined
*/
JSValue *get(ExecState *exec, const Identifier &propertyName) const;
JSValue *get(ExecState *exec, unsigned propertyName) const;
bool getPropertySlot(ExecState *, const Identifier&, PropertySlot&);
bool getPropertySlot(ExecState *, unsigned, PropertySlot&);
virtual bool getOwnPropertySlot(ExecState *, const Identifier&, PropertySlot&);
virtual bool getOwnPropertySlot(ExecState *, unsigned index, PropertySlot&);
/**
* Sets the specified property.
*
* See ECMA 8.6.2.2
*
* @param exec The current execution state
* @param propertyName The name of the property to set
* @param propertyValue The value to set
*/
virtual void put(ExecState* exec, const Identifier &propertyName, JSValue* value, int attr = None);
virtual void put(ExecState* exec, unsigned propertyName, JSValue* value, int attr = None);
/**
* Used to check whether or not a particular property is allowed to be set
* on an object
*
* See ECMA 8.6.2.3
*
* @param exec The current execution state
* @param propertyName The name of the property
* @return true if the property can be set, otherwise false
*/
/**
* Implementation of the [[CanPut]] internal property (implemented by all
* Objects)
*/
virtual bool canPut(ExecState *exec, const Identifier &propertyName) const;
/**
* Checks if a property is enumerable, that is if it doesn't have the DontEnum
* flag set
*
* See ECMA 15.2.4
* @param exec The current execution state
* @param propertyName The name of the property
* @return true if the property is enumerable, otherwise false
*/
bool propertyIsEnumerable(ExecState *exec, const Identifier &propertyName) const;
/**
* Checks to see whether the object (or any object in it's prototype chain)
* has a property with the specified name.
*
* See ECMA 8.6.2.4
*
* @param exec The current execution state
* @param propertyName The name of the property to check for
* @return true if the object has the property, otherwise false
*/
bool hasProperty(ExecState *exec, const Identifier &propertyName) const;
bool hasProperty(ExecState *exec, unsigned propertyName) const;
/**
* Removes the specified property from the object.
*
* See ECMA 8.6.2.5
*
* @param exec The current execution state
* @param propertyName The name of the property to delete
* @return true if the property was successfully deleted or did not
* exist on the object. false if deleting the specified property is not
* allowed.
*/
virtual bool deleteProperty(ExecState *exec, const Identifier &propertyName);
virtual bool deleteProperty(ExecState *exec, unsigned propertyName);
/**
* Converts the object into a primitive value. The value return may differ
* depending on the supplied hint
*
* See ECMA 8.6.2.6
*
* @param exec The current execution state
* @param hint The desired primitive type to convert to
* @return A primitive value converted from the objetc. Note that the
* type of primitive value returned may not be the same as the requested
* hint.
*/
/**
* Implementation of the [[DefaultValue]] internal property (implemented by
* all Objects)
*/
virtual JSValue *defaultValue(ExecState *exec, JSType hint) const;
/**
* Whether or not the object implements the construct() method. If this
* returns false you should not call the construct() method on this
* object (typically, an assertion will fail to indicate this).
*
* @return true if this object implements the construct() method, otherwise
* false
*/
virtual bool implementsConstruct() const;
/**
* Creates a new object based on this object. Typically this means the
* following:
* 1. A new object is created
* 2. The prototype of the new object is set to the value of this object's
* "prototype" property
* 3. The call() method of this object is called, with the new object
* passed as the this value
* 4. The new object is returned
*
* In some cases, Host objects may differ from these semantics, although
* this is discouraged.
*
* If an error occurs during construction, the execution state's exception
* will be set. This can be tested for with ExecState::hadException().
* Under some circumstances, the exception object may also be returned.
*
* Note: This function should not be called if implementsConstruct() returns
* false, in which case it will result in an assertion failure.
*
* @param exec The current execution state
* @param args The arguments to be passed to call() once the new object has
* been created
* @return The newly created &amp; initialized object
*/
/**
* Implementation of the [[Construct]] internal property
*/
virtual JSObject* construct(ExecState* exec, const List& args);
virtual JSObject* construct(ExecState* exec, const List& args, const Identifier& functionName, const UString& sourceURL, int lineNumber);
/**
* Whether or not the object implements the call() method. If this returns
* false you should not call the call() method on this object (typically,
* an assertion will fail to indicate this).
*
* @return true if this object implements the call() method, otherwise
* false
*/
virtual bool implementsCall() const;
/**
* Calls this object as if it is a function.
*
* Note: This function should not be called if implementsCall() returns
* false, in which case it will result in an assertion failure.
*
* See ECMA 8.6.2.3
*
* @param exec The current execution state
* @param thisObj The obj to be used as "this" within function execution.
* Note that in most cases this will be different from the C++ "this"
* object. For example, if the ECMAScript code "window.location->toString()"
* is executed, call() will be invoked on the C++ object which implements
* the toString method, with the thisObj being window.location
* @param args List of arguments to be passed to the function
* @return The return value from the function
*/
JSValue *call(ExecState *exec, JSObject *thisObj, const List &args);
virtual JSValue *callAsFunction(ExecState *exec, JSObject *thisObj, const List &args);
/**
* Whether or not the object implements the hasInstance() method. If this
* returns false you should not call the hasInstance() method on this
* object (typically, an assertion will fail to indicate this).
*
* @return true if this object implements the hasInstance() method,
* otherwise false
*/
virtual bool implementsHasInstance() const;
/**
* Checks whether value delegates behavior to this object. Used by the
* instanceof operator.
*
* @param exec The current execution state
* @param value The value to check
* @return true if value delegates behavior to this object, otherwise
* false
*/
virtual bool hasInstance(ExecState *exec, JSValue *value);
virtual void getPropertyNames(ExecState*, PropertyNameArray&);
virtual JSValue *toPrimitive(ExecState *exec, JSType preferredType = UnspecifiedType) const;
virtual bool toBoolean(ExecState *exec) const;
virtual double toNumber(ExecState *exec) const;
virtual UString toString(ExecState *exec) const;
virtual JSObject *toObject(ExecState *exec) const;
bool getPropertyAttributes(const Identifier& propertyName, unsigned& attributes) const;
// WebCore uses this to make document.all and style.filter undetectable
virtual bool masqueradeAsUndefined() const { return false; }
// This get function only looks at the property map.
// This is used e.g. by lookupOrCreateFunction (to cache a function, we don't want
// to look up in the prototype, it might already exist there)
JSValue *getDirect(const Identifier& propertyName) const
{ return _prop.get(propertyName); }
JSValue **getDirectLocation(const Identifier& propertyName)
{ return _prop.getLocation(propertyName); }
void putDirect(const Identifier &propertyName, JSValue *value, int attr = 0);
void putDirect(const Identifier &propertyName, int value, int attr = 0);
// convenience to add a function property under the function's own built-in name
void putDirectFunction(InternalFunctionImp*, int attr = 0);
void fillGetterPropertySlot(PropertySlot& slot, JSValue **location);
void defineGetter(ExecState *exec, const Identifier& propertyName, JSObject *getterFunc);
void defineSetter(ExecState *exec, const Identifier& propertyName, JSObject *setterFunc);
/**
* Remove all properties from this object.
* This doesn't take DontDelete into account, and isn't in the ECMA spec.
* It's simply a quick way to remove everything stored in the property map.
*/
void clearProperties() { _prop.clear(); }
void saveProperties(SavedProperties &p) const { _prop.save(p); }
void restoreProperties(const SavedProperties &p) { _prop.restore(p); }
virtual bool isActivation() { return false; }
protected:
PropertyMap _prop;
private:
const HashEntry* findPropertyHashEntry( const Identifier& propertyName ) const;
JSValue *_proto;
};
/**
* Types of Native Errors available. For custom errors, GeneralError
* should be used.
*/
enum ErrorType { GeneralError = 0,
EvalError = 1,
RangeError = 2,
ReferenceError = 3,
SyntaxError = 4,
TypeError = 5,
URIError = 6};
/**
* @short Factory methods for error objects.
*/
class Error {
public:
/**
* Factory method for error objects.
*
* @param exec The current execution state
* @param errtype Type of error.
* @param message Optional error message.
* @param lineNumber Optional line number.
* @param sourceId Optional source id.
* @param sourceURL Optional source URL.
*/
static JSObject *create(ExecState *, ErrorType, const UString &message, int lineNumber, int sourceId, const UString &sourceURL);
static JSObject *create(ExecState *, ErrorType, const char *message);
/**
* Array of error names corresponding to ErrorType
*/
static const char * const * const errorNames;
};
JSObject *throwError(ExecState *, ErrorType, const UString &message, int lineNumber, int sourceId, const UString &sourceURL);
JSObject *throwError(ExecState *, ErrorType, const UString &message);
JSObject *throwError(ExecState *, ErrorType, const char *message);
JSObject *throwError(ExecState *, ErrorType);
inline JSObject::JSObject(JSValue* proto)
: _proto(proto)
{
assert(proto);
}
inline JSObject::JSObject()
: _proto(jsNull())
{
}
inline JSValue *JSObject::prototype() const
{
return _proto;
}
inline void JSObject::setPrototype(JSValue *proto)
{
assert(proto);
_proto = proto;
}
inline bool JSObject::inherits(const ClassInfo *info) const
{
for (const ClassInfo *ci = classInfo(); ci; ci = ci->parentClass)
if (ci == info)
return true;
return false;
}
// this method is here to be after the inline declaration of JSObject::inherits
inline bool JSCell::isObject(const ClassInfo *info) const
{
return isObject() && static_cast<const JSObject *>(this)->inherits(info);
}
// this method is here to be after the inline declaration of JSCell::isObject
inline bool JSValue::isObject(const ClassInfo *c) const
{
return !JSImmediate::isImmediate(this) && asCell()->isObject(c);
}
// It may seem crazy to inline a function this large but it makes a big difference
// since this is function very hot in variable lookup
inline bool JSObject::getPropertySlot(ExecState *exec, const Identifier& propertyName, PropertySlot& slot)
{
JSObject *object = this;
while (true) {
if (object->getOwnPropertySlot(exec, propertyName, slot))
return true;
JSValue *proto = object->_proto;
if (!proto->isObject())
return false;
object = static_cast<JSObject *>(proto);
}
}
// It may seem crazy to inline a function this large, especially a virtual function,
// but it makes a big difference to property lookup that derived classes can inline their
// base class call to this.
ALWAYS_INLINE bool JSObject::getOwnPropertySlot(ExecState* exec, const Identifier& propertyName, PropertySlot& slot)
{
if (JSValue **location = getDirectLocation(propertyName)) {
if (_prop.hasGetterSetterProperties() && location[0]->type() == GetterSetterType)
fillGetterPropertySlot(slot, location);
else
slot.setValueSlot(this, location);
return true;
}
// non-standard Netscape extension
if (propertyName == exec->propertyNames().underscoreProto) {
slot.setValueSlot(this, &_proto);
return true;
}
return false;
}
// FIXME: Put this function in a separate file named something like scope_chain_mark.h -- can't put it in scope_chain.h since it depends on JSObject.
inline void ScopeChain::mark()
{
for (ScopeChainNode *n = _node; n; n = n->next) {
JSObject *o = n->object;
if (!o->marked())
o->mark();
}
}
inline void ScopeChain::release()
{
// This function is only called by deref(),
// Deref ensures these conditions are true.
assert(_node && _node->refCount == 0);
ScopeChainNode *n = _node;
do {
ScopeChainNode *next = n->next;
delete n;
n = next;
} while (n && --n->refCount == 0);
}
inline JSValue* JSObject::toPrimitive(ExecState* exec, JSType preferredType) const
{
return defaultValue(exec, preferredType);
}
} // namespace
#endif // KJS_OBJECT_H