gecko/js/src/jsproxy.cpp

1323 lines
40 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim: set ts=4 sw=4 et tw=99:
*
* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is Mozilla SpiderMonkey JavaScript 1.9 code, released
* May 28, 2008.
*
* The Initial Developer of the Original Code is
* Mozilla Foundation
* Portions created by the Initial Developer are Copyright (C) 2009
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Andreas Gal <gal@mozilla.com>
*
* Alternatively, the contents of this file may be used under the terms of
* either of the GNU General Public License Version 2 or later (the "GPL"),
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
#include <string.h>
#include "jsapi.h"
#include "jscntxt.h"
#include "jsprvtd.h"
#include "jsnum.h"
#include "jsobj.h"
#include "jsproxy.h"
#include "jsscope.h"
#include "jsobjinlines.h"
using namespace js;
namespace js {
static jsval
GetCall(JSObject *proxy) {
JS_ASSERT(proxy->isFunctionProxy());
return proxy->getSlot(JSSLOT_PROXY_CALL);
}
static jsval
GetConstruct(JSObject *proxy) {
if (proxy->numSlots() <= JSSLOT_PROXY_CONSTRUCT)
return JSVAL_VOID;
return proxy->getSlot(JSSLOT_PROXY_CONSTRUCT);
}
static bool
OperationInProgress(JSContext *cx, JSObject *proxy)
{
JSPendingProxyOperation *op = JS_THREAD_DATA(cx)->pendingProxyOperation;
while (op) {
if (op->object == proxy)
return true;
op = op->next;
}
return false;
}
JSProxyHandler::JSProxyHandler(void *family) : mFamily(family)
{
}
JSProxyHandler::~JSProxyHandler()
{
}
bool
JSProxyHandler::has(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
JS_ASSERT(OperationInProgress(cx, proxy));
AutoDescriptor desc(cx);
if (!getPropertyDescriptor(cx, proxy, id, &desc))
return false;
*bp = !!desc.obj;
return true;
}
bool
JSProxyHandler::hasOwn(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
JS_ASSERT(OperationInProgress(cx, proxy));
AutoDescriptor desc(cx);
if (!getOwnPropertyDescriptor(cx, proxy, id, &desc))
return false;
*bp = !!desc.obj;
return true;
}
bool
JSProxyHandler::get(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp)
{
JS_ASSERT(OperationInProgress(cx, proxy));
AutoDescriptor desc(cx);
if (!getPropertyDescriptor(cx, proxy, id, &desc))
return false;
if (!desc.obj) {
*vp = JSVAL_VOID;
return true;
}
if (!desc.getter) {
*vp = desc.value;
return true;
}
if (desc.attrs & JSPROP_GETTER) {
return js_InternalGetOrSet(cx, proxy, id, CastAsObjectJSVal(desc.getter),
JSACC_READ, 0, 0, vp);
}
if (desc.attrs & JSPROP_SHORTID)
id = INT_TO_JSID(desc.shortid);
return callJSPropertyOp(cx, desc.getter, proxy, id, vp);
}
bool
JSProxyHandler::set(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp)
{
JS_ASSERT(OperationInProgress(cx, proxy));
AutoDescriptor desc(cx);
if (!getOwnPropertyDescriptor(cx, proxy, id, &desc))
return false;
/* The control-flow here differs from ::get() because of the fall-through case below. */
if (desc.obj) {
if (desc.setter) {
if (desc.attrs & JSPROP_SETTER) {
return js_InternalGetOrSet(cx, proxy, id, CastAsObjectJSVal(desc.setter),
JSACC_READ, 0, 0, vp);
}
if (desc.attrs & JSPROP_SHORTID)
id = INT_TO_JSID(desc.shortid);
return callJSPropertyOpSetter(cx, desc.setter, proxy, id, vp);
}
if (desc.attrs & JSPROP_READONLY)
return true;
desc.value = *vp;
return defineProperty(cx, proxy, id, &desc);
}
if (!getPropertyDescriptor(cx, proxy, id, &desc))
return false;
if (desc.obj) {
if (desc.setter) {
if (desc.attrs & JSPROP_SETTER) {
return js_InternalGetOrSet(cx, proxy, id, CastAsObjectJSVal(desc.setter),
JSACC_READ, 0, 0, vp);
}
if (desc.attrs & JSPROP_SHORTID)
id = INT_TO_JSID(desc.shortid);
return callJSPropertyOpSetter(cx, desc.setter, proxy, id, vp);
}
if (desc.attrs & JSPROP_READONLY)
return true;
/* fall through */
}
desc.obj = proxy;
desc.value = *vp;
desc.attrs = 0;
desc.getter = JSVAL_NULL;
desc.setter = JSVAL_NULL;
desc.shortid = 0;
return defineProperty(cx, proxy, id, &desc);
}
bool
JSProxyHandler::enumerateOwn(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
JS_ASSERT(OperationInProgress(cx, proxy));
JS_ASSERT(props.length() == 0);
if (!getOwnPropertyNames(cx, proxy, props))
return false;
/* Select only the enumerable properties through in-place iteration. */
AutoDescriptor desc(cx);
size_t i = 0;
for (size_t j = 0, len = props.length(); j < len; j++) {
JS_ASSERT(i <= j);
jsid id = props[j];
if (!getOwnPropertyDescriptor(cx, proxy, id, &desc))
return false;
if (desc.obj && (desc.attrs & JSPROP_ENUMERATE))
props[i++] = id;
}
JS_ASSERT(i <= props.length());
props.resize(i);
return true;
}
bool
JSProxyHandler::iterate(JSContext *cx, JSObject *proxy, uintN flags, jsval *vp)
{
JS_ASSERT(OperationInProgress(cx, proxy));
AutoValueVector props(cx);
if (!enumerate(cx, proxy, props))
return false;
return IdVectorToIterator(cx, proxy, flags, props, vp);
}
JSString *
JSProxyHandler::obj_toString(JSContext *cx, JSObject *proxy)
{
JS_ASSERT(proxy->isProxy());
return JS_NewStringCopyZ(cx, proxy->isFunctionProxy()
? "[object Function]"
: "[object Object]");
}
JSString *
JSProxyHandler::fun_toString(JSContext *cx, JSObject *proxy, uintN indent)
{
JS_ASSERT(proxy->isProxy());
jsval fval = GetCall(proxy);
if (proxy->isFunctionProxy() &&
(JSVAL_IS_PRIMITIVE(fval) ||
!JSVAL_TO_OBJECT(fval)->isFunction())) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
JSMSG_INCOMPATIBLE_PROTO,
js_Function_str, js_toString_str,
"object");
return NULL;
}
return fun_toStringHelper(cx, JSVAL_TO_OBJECT(fval), indent);
}
bool
JSProxyHandler::call(JSContext *cx, JSObject *proxy, uintN argc, jsval *vp)
{
JS_ASSERT(OperationInProgress(cx, proxy));
AutoValueRooter rval(cx);
JSBool ok = js_InternalInvoke(cx, vp[1], GetCall(proxy), 0, argc, JS_ARGV(cx, vp),
rval.addr());
if (ok)
JS_SET_RVAL(cx, vp, rval.value());
return ok;
}
bool
JSProxyHandler::construct(JSContext *cx, JSObject *proxy, JSObject *receiver,
uintN argc, jsval *argv, jsval *rval)
{
JS_ASSERT(OperationInProgress(cx, proxy));
jsval fval = GetConstruct(proxy);
if (fval == JSVAL_VOID) {
/*
* proxy is the constructor, so get proxy.prototype as the proto
* of the new object.
*/
if (!JSProxy::get(cx, proxy, proxy, ATOM_TO_JSID(ATOM(classPrototype)), rval))
return false;
JSObject *proto;
if (!JSVAL_IS_PRIMITIVE(*rval)) {
proto = JSVAL_TO_OBJECT(*rval);
} else {
if (!js_GetClassPrototype(cx, NULL, JSProto_Object, &proto))
return false;
}
JSObject *newobj = NewNativeClassInstance(cx, &js_ObjectClass, proto, proto->getParent());
*rval = OBJECT_TO_JSVAL(newobj);
/* If the call returns an object, return that, otherwise the original newobj. */
if (!js_InternalCall(cx, newobj, GetCall(proxy), argc, argv, rval))
return false;
if (JSVAL_IS_PRIMITIVE(*rval))
*rval = OBJECT_TO_JSVAL(newobj);
return true;
}
return js_InternalCall(cx, receiver, fval, argc, argv, rval);
}
void
JSProxyHandler::finalize(JSContext *cx, JSObject *proxy)
{
}
void
JSProxyHandler::trace(JSTracer *trc, JSObject *proxy)
{
}
static bool
GetTrap(JSContext *cx, JSObject *handler, JSAtom *atom, jsval *fvalp)
{
return handler->getProperty(cx, ATOM_TO_JSID(atom), fvalp);
}
static bool
FundamentalTrap(JSContext *cx, JSObject *handler, JSAtom *atom, jsval *fvalp)
{
if (!GetTrap(cx, handler, atom, fvalp))
return false;
if (!js_IsCallable(*fvalp)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_NOT_FUNCTION,
js_AtomToPrintableString(cx, atom));
return false;
}
return true;
}
static bool
DerivedTrap(JSContext *cx, JSObject *handler, JSAtom *atom, jsval *fvalp)
{
JS_ASSERT(atom == ATOM(has) ||
atom == ATOM(hasOwn) ||
atom == ATOM(get) ||
atom == ATOM(set) ||
atom == ATOM(enumerateOwn) ||
atom == ATOM(iterate));
return GetTrap(cx, handler, atom, fvalp);
}
static bool
Trap(JSContext *cx, JSObject *handler, jsval fval, uintN argc, jsval* argv, jsval *rval)
{
JS_CHECK_RECURSION(cx, return false);
return js_InternalCall(cx, handler, fval, argc, argv, rval);
}
static bool
Trap1(JSContext *cx, JSObject *handler, jsval fval, jsid id, jsval *rval)
{
JSString *str = js_ValueToString(cx, ID_TO_VALUE(id));
if (!str)
return false;
*rval = STRING_TO_JSVAL(str);
return Trap(cx, handler, fval, 1, rval, rval);
}
static bool
Trap2(JSContext *cx, JSObject *handler, jsval fval, jsid id, jsval v, jsval *rval)
{
JSString *str = js_ValueToString(cx, ID_TO_VALUE(id));
if (!str)
return false;
*rval = STRING_TO_JSVAL(str);
jsval argv[2] = { *rval, v };
return Trap(cx, handler, fval, 2, argv, rval);
}
static bool
ParsePropertyDescriptorObject(JSContext *cx, JSObject *obj, jsid id, jsval v,
JSPropertyDescriptor *desc)
{
AutoDescriptorArray descs(cx);
PropertyDescriptor *d = descs.append();
if (!d || !d->initialize(cx, id, v))
return false;
desc->obj = obj;
desc->value = d->value;
JS_ASSERT(!(d->attrs & JSPROP_SHORTID));
desc->attrs = d->attrs;
desc->getter = d->getter();
desc->setter = d->setter();
desc->shortid = 0;
return true;
}
static bool
MakePropertyDescriptorObject(JSContext *cx, jsid id, JSPropertyDescriptor *desc, jsval *vp)
{
if (!desc->obj) {
*vp = JSVAL_VOID;
return true;
}
uintN attrs = desc->attrs;
jsval getter = (attrs & JSPROP_GETTER) ? CastAsObjectJSVal(desc->getter) : JSVAL_VOID;
jsval setter = (attrs & JSPROP_SETTER) ? CastAsObjectJSVal(desc->setter) : JSVAL_VOID;
return js_NewPropertyDescriptorObject(cx, id, attrs, getter, setter, desc->value, vp);
}
static bool
ValueToBool(JSContext *cx, jsval v, bool *bp)
{
JSBool b;
if (!JS_ValueToBoolean(cx, v, &b))
return false;
*bp = !!b;
return true;
}
bool
ArrayToIdVector(JSContext *cx, jsval array, AutoValueVector &props)
{
JS_ASSERT(props.length() == 0);
if (JSVAL_IS_PRIMITIVE(array))
return true;
JSObject *obj = JSVAL_TO_OBJECT(array);
jsuint length;
if (!js_GetLengthProperty(cx, obj, &length)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_ARRAY_LENGTH);
return false;
}
AutoIdRooter idr(cx);
AutoValueRooter tvr(cx);
for (jsuint n = 0; n < length; ++n) {
if (!js_IndexToId(cx, n, idr.addr()))
return false;
if (!obj->getProperty(cx, idr.id(), tvr.addr()))
return false;
if (!JS_ValueToId(cx, tvr.value(), idr.addr()))
return false;
if (!props.append(js_CheckForStringIndex(idr.id())))
return false;
}
return true;
}
/* Derived class for all scripted proxy handlers. */
class JSScriptedProxyHandler : public JSProxyHandler {
public:
JSScriptedProxyHandler();
virtual ~JSScriptedProxyHandler();
/* ES5 Harmony fundamental proxy traps. */
virtual bool getPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc);
virtual bool getOwnPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc);
virtual bool defineProperty(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc);
virtual bool getOwnPropertyNames(JSContext *cx, JSObject *proxy, AutoValueVector &props);
virtual bool delete_(JSContext *cx, JSObject *proxy, jsid id, bool *bp);
virtual bool enumerate(JSContext *cx, JSObject *proxy, AutoValueVector &props);
virtual bool fix(JSContext *cx, JSObject *proxy, jsval *vp);
/* ES5 Harmony derived proxy traps. */
virtual bool has(JSContext *cx, JSObject *proxy, jsid id, bool *bp);
virtual bool hasOwn(JSContext *cx, JSObject *proxy, jsid id, bool *bp);
virtual bool get(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp);
virtual bool set(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp);
virtual bool enumerateOwn(JSContext *cx, JSObject *proxy, AutoValueVector &props);
virtual bool iterate(JSContext *cx, JSObject *proxy, uintN flags, jsval *vp);
static JSScriptedProxyHandler singleton;
};
static int sScriptedProxyHandlerFamily = 0;
JSScriptedProxyHandler::JSScriptedProxyHandler() : JSProxyHandler(&sScriptedProxyHandlerFamily)
{
}
JSScriptedProxyHandler::~JSScriptedProxyHandler()
{
}
static bool
ReturnedValueMustNotBePrimitive(JSContext *cx, JSObject *proxy, JSAtom *atom, jsval v)
{
if (JSVAL_IS_PRIMITIVE(v)) {
js_ReportValueError2(cx, JSMSG_BAD_TRAP_RETURN_VALUE,
JSDVG_SEARCH_STACK, OBJECT_TO_JSVAL(proxy), NULL,
js_AtomToPrintableString(cx, atom));
return false;
}
return true;
}
static JSObject *
GetProxyHandlerObject(JSContext *cx, JSObject *proxy)
{
JS_ASSERT(OperationInProgress(cx, proxy));
return JSVAL_TO_OBJECT(proxy->getProxyPrivate());
}
bool
JSScriptedProxyHandler::getPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
return FundamentalTrap(cx, handler, ATOM(getPropertyDescriptor), tvr.addr()) &&
Trap1(cx, handler, tvr.value(), id, tvr.addr()) &&
ReturnedValueMustNotBePrimitive(cx, proxy, ATOM(getPropertyDescriptor), tvr.value()) &&
ParsePropertyDescriptorObject(cx, proxy, id, tvr.value(), desc);
}
bool
JSScriptedProxyHandler::getOwnPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
return FundamentalTrap(cx, handler, ATOM(getOwnPropertyDescriptor), tvr.addr()) &&
Trap1(cx, handler, tvr.value(), id, tvr.addr()) &&
ReturnedValueMustNotBePrimitive(cx, proxy, ATOM(getPropertyDescriptor), tvr.value()) &&
ParsePropertyDescriptorObject(cx, proxy, id, tvr.value(), desc);
}
bool
JSScriptedProxyHandler::defineProperty(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
AutoValueRooter fval(cx);
return FundamentalTrap(cx, handler, ATOM(defineProperty), fval.addr()) &&
MakePropertyDescriptorObject(cx, id, desc, tvr.addr()) &&
Trap2(cx, handler, fval.value(), id, tvr.value(), tvr.addr());
}
bool
JSScriptedProxyHandler::getOwnPropertyNames(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
return FundamentalTrap(cx, handler, ATOM(getOwnPropertyNames), tvr.addr()) &&
Trap(cx, handler, tvr.value(), 0, NULL, tvr.addr()) &&
ArrayToIdVector(cx, tvr.value(), props);
}
bool
JSScriptedProxyHandler::delete_(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
return FundamentalTrap(cx, handler, ATOM(delete), tvr.addr()) &&
Trap1(cx, handler, tvr.value(), id, tvr.addr()) &&
ValueToBool(cx, tvr.value(), bp);
}
bool
JSScriptedProxyHandler::enumerate(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
return FundamentalTrap(cx, handler, ATOM(enumerate), tvr.addr()) &&
Trap(cx, handler, tvr.value(), 0, NULL, tvr.addr()) &&
ArrayToIdVector(cx, tvr.value(), props);
}
bool
JSScriptedProxyHandler::fix(JSContext *cx, JSObject *proxy, jsval *vp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
return FundamentalTrap(cx, handler, ATOM(fix), vp) &&
Trap(cx, handler, *vp, 0, NULL, vp);
}
bool
JSScriptedProxyHandler::has(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
if (!DerivedTrap(cx, handler, ATOM(has), tvr.addr()))
return false;
if (!js_IsCallable(tvr.value()))
return JSProxyHandler::has(cx, proxy, id, bp);
return Trap1(cx, handler, tvr.value(), id, tvr.addr()) &&
ValueToBool(cx, tvr.value(), bp);
}
bool
JSScriptedProxyHandler::hasOwn(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
if (!DerivedTrap(cx, handler, ATOM(hasOwn), tvr.addr()))
return false;
if (!js_IsCallable(tvr.value()))
return JSProxyHandler::hasOwn(cx, proxy, id, bp);
return Trap1(cx, handler, tvr.value(), id, tvr.addr()) &&
ValueToBool(cx, tvr.value(), bp);
}
bool
JSScriptedProxyHandler::get(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
JSString *str = js_ValueToString(cx, ID_TO_VALUE(id));
if (!str)
return false;
AutoValueRooter tvr(cx, STRING_TO_JSVAL(str));
jsval argv[] = { OBJECT_TO_JSVAL(receiver), tvr.value() };
AutoValueRooter fval(cx);
if (!DerivedTrap(cx, handler, ATOM(get), fval.addr()))
return false;
if (!js_IsCallable(fval.value()))
return JSProxyHandler::get(cx, proxy, receiver, id, vp);
return Trap(cx, handler, fval.value(), 2, argv, vp);
}
bool
JSScriptedProxyHandler::set(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
JSString *str = js_ValueToString(cx, ID_TO_VALUE(id));
if (!str)
return false;
AutoValueRooter tvr(cx, STRING_TO_JSVAL(str));
jsval argv[] = { OBJECT_TO_JSVAL(receiver), tvr.value(), *vp };
AutoValueRooter fval(cx);
if (!DerivedTrap(cx, handler, ATOM(set), fval.addr()))
return false;
if (!js_IsCallable(fval.value()))
return JSProxyHandler::set(cx, proxy, receiver, id, vp);
return Trap(cx, handler, fval.value(), 3, argv, tvr.addr());
}
bool
JSScriptedProxyHandler::enumerateOwn(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
if (!DerivedTrap(cx, handler, ATOM(enumerateOwn), tvr.addr()))
return false;
if (!js_IsCallable(tvr.value()))
return JSProxyHandler::enumerateOwn(cx, proxy, props);
return Trap(cx, handler, tvr.value(), 0, NULL, tvr.addr()) &&
ArrayToIdVector(cx, tvr.value(), props);
}
bool
JSScriptedProxyHandler::iterate(JSContext *cx, JSObject *proxy, uintN flags, jsval *vp)
{
JSObject *handler = GetProxyHandlerObject(cx, proxy);
AutoValueRooter tvr(cx);
if (!DerivedTrap(cx, handler, ATOM(iterate), tvr.addr()))
return false;
if (!js_IsCallable(tvr.value()))
return JSProxyHandler::iterate(cx, proxy, flags, vp);
return Trap(cx, handler, tvr.value(), 0, NULL, vp) &&
ReturnedValueMustNotBePrimitive(cx, proxy, ATOM(iterate), *vp);
}
JSScriptedProxyHandler JSScriptedProxyHandler::singleton;
class AutoPendingProxyOperation {
JSThreadData *data;
JSPendingProxyOperation op;
public:
AutoPendingProxyOperation(JSContext *cx, JSObject *proxy) : data(JS_THREAD_DATA(cx)) {
op.next = data->pendingProxyOperation;
op.object = proxy;
data->pendingProxyOperation = &op;
}
~AutoPendingProxyOperation() {
JS_ASSERT(data->pendingProxyOperation == &op);
data->pendingProxyOperation = op.next;
}
};
bool
JSProxy::getPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id, JSPropertyDescriptor *desc)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->getPropertyDescriptor(cx, proxy, id, desc);
}
bool
JSProxy::getPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
AutoDescriptor desc(cx);
return JSProxy::getPropertyDescriptor(cx, proxy, id, &desc) &&
MakePropertyDescriptorObject(cx, id, &desc, vp);
}
bool
JSProxy::getOwnPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id,
JSPropertyDescriptor *desc)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->getOwnPropertyDescriptor(cx, proxy, id, desc);
}
bool
JSProxy::getOwnPropertyDescriptor(JSContext *cx, JSObject *proxy, jsid id, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
AutoDescriptor desc(cx);
return JSProxy::getOwnPropertyDescriptor(cx, proxy, id, &desc) &&
MakePropertyDescriptorObject(cx, id, &desc, vp);
}
bool
JSProxy::defineProperty(JSContext *cx, JSObject *proxy, jsid id, JSPropertyDescriptor *desc)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->defineProperty(cx, proxy, id, desc);
}
bool
JSProxy::defineProperty(JSContext *cx, JSObject *proxy, jsid id, jsval v)
{
AutoPendingProxyOperation pending(cx, proxy);
AutoDescriptor desc(cx);
return ParsePropertyDescriptorObject(cx, proxy, id, v, &desc) &&
JSProxy::defineProperty(cx, proxy, id, &desc);
}
bool
JSProxy::getOwnPropertyNames(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->getOwnPropertyNames(cx, proxy, props);
}
bool
JSProxy::delete_(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->delete_(cx, proxy, id, bp);
}
bool
JSProxy::enumerate(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->enumerate(cx, proxy, props);
}
bool
JSProxy::fix(JSContext *cx, JSObject *proxy, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->fix(cx, proxy, vp);
}
bool
JSProxy::has(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->has(cx, proxy, id, bp);
}
bool
JSProxy::hasOwn(JSContext *cx, JSObject *proxy, jsid id, bool *bp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->hasOwn(cx, proxy, id, bp);
}
bool
JSProxy::get(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->get(cx, proxy, receiver, id, vp);
}
bool
JSProxy::set(JSContext *cx, JSObject *proxy, JSObject *receiver, jsid id, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->set(cx, proxy, receiver, id, vp);
}
bool
JSProxy::enumerateOwn(JSContext *cx, JSObject *proxy, AutoValueVector &props)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->enumerateOwn(cx, proxy, props);
}
bool
JSProxy::iterate(JSContext *cx, JSObject *proxy, uintN flags, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->iterate(cx, proxy, flags, vp);
}
bool
JSProxy::call(JSContext *cx, JSObject *proxy, uintN argc, jsval *vp)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->call(cx, proxy, argc, vp);
}
bool
JSProxy::construct(JSContext *cx, JSObject *proxy, JSObject *receiver,
uintN argc, jsval *argv, jsval *rval)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->construct(cx, proxy, receiver, argc, argv, rval);
}
JSString *
JSProxy::obj_toString(JSContext *cx, JSObject *proxy)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->obj_toString(cx, proxy);
}
JSString *
JSProxy::fun_toString(JSContext *cx, JSObject *proxy, uintN indent)
{
AutoPendingProxyOperation pending(cx, proxy);
return proxy->getProxyHandler()->fun_toString(cx, proxy, indent);
}
static JSBool
proxy_LookupProperty(JSContext *cx, JSObject *obj, jsid id, JSObject **objp,
JSProperty **propp)
{
bool found;
if (!JSProxy::has(cx, obj, id, &found))
return false;
if (found) {
*propp = (JSProperty *)id;
*objp = obj;
} else {
*objp = NULL;
*propp = NULL;
}
return true;
}
static JSBool
proxy_DefineProperty(JSContext *cx, JSObject *obj, jsid id, jsval value,
JSPropertyOp getter, JSPropertyOp setter, uintN attrs)
{
AutoDescriptor desc(cx);
desc.obj = obj;
desc.value = value;
desc.attrs = (attrs & (~JSPROP_SHORTID));
desc.getter = getter;
desc.setter = setter;
desc.shortid = 0;
return JSProxy::defineProperty(cx, obj, id, &desc);
}
static JSBool
proxy_GetProperty(JSContext *cx, JSObject *obj, jsid id, jsval *vp)
{
return JSProxy::get(cx, obj, obj, id, vp);
}
static JSBool
proxy_SetProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
{
return JSProxy::set(cx, obj, obj, id, vp);
}
static JSBool
proxy_GetAttributes(JSContext *cx, JSObject *obj, jsid id, uintN *attrsp)
{
AutoDescriptor desc(cx);
if (!JSProxy::getOwnPropertyDescriptor(cx, obj, id, &desc))
return false;
*attrsp = desc.attrs;
return true;
}
static JSBool
proxy_SetAttributes(JSContext *cx, JSObject *obj, jsid id, uintN *attrsp)
{
/* Lookup the current property descriptor so we have setter/getter/value. */
AutoDescriptor desc(cx);
if (!JSProxy::getOwnPropertyDescriptor(cx, obj, id, &desc))
return false;
desc.attrs = (*attrsp & (~JSPROP_SHORTID));
return JSProxy::defineProperty(cx, obj, id, &desc);
}
static JSBool
proxy_DeleteProperty(JSContext *cx, JSObject *obj, jsval id, jsval *rval)
{
bool deleted;
if (!JSProxy::delete_(cx, obj, id, &deleted))
return false;
*rval = BOOLEAN_TO_JSVAL(deleted);
return true;
}
static void
proxy_TraceObject(JSTracer *trc, JSObject *obj)
{
JSContext *cx = trc->context;
if (!JS_CLIST_IS_EMPTY(&cx->runtime->watchPointList))
js_TraceWatchPoints(trc, obj);
JSClass *clasp = obj->getClass();
if (clasp->mark) {
if (clasp->flags & JSCLASS_MARK_IS_TRACE)
((JSTraceOp) clasp->mark)(trc, obj);
else if (IS_GC_MARKING_TRACER(trc))
(void) clasp->mark(cx, obj, trc);
}
obj->traceProtoAndParent(trc);
obj->getProxyHandler()->trace(trc, obj);
JS_CALL_VALUE_TRACER(trc, obj->getProxyPrivate(), "private");
if (obj->isFunctionProxy()) {
JS_CALL_VALUE_TRACER(trc, GetCall(obj), "call");
JS_CALL_VALUE_TRACER(trc, GetConstruct(obj), "construct");
}
}
static JSType
proxy_TypeOf_obj(JSContext *cx, JSObject *obj)
{
return JSTYPE_OBJECT;
}
void
proxy_Finalize(JSContext *cx, JSObject *obj)
{
JS_ASSERT(obj->isProxy());
if (obj->getSlot(JSSLOT_PROXY_HANDLER) != JSVAL_VOID)
obj->getProxyHandler()->finalize(cx, obj);
}
extern JSObjectOps js_ObjectProxyObjectOps;
static const JSObjectMap SharedObjectProxyMap(&js_ObjectProxyObjectOps, JSObjectMap::SHAPELESS);
JSObjectOps js_ObjectProxyObjectOps = {
&SharedObjectProxyMap,
proxy_LookupProperty,
proxy_DefineProperty,
proxy_GetProperty,
proxy_SetProperty,
proxy_GetAttributes,
proxy_SetAttributes,
proxy_DeleteProperty,
js_DefaultValue,
js_Enumerate,
proxy_TypeOf_obj,
proxy_TraceObject,
NULL, /* thisObject */
NULL, /* call */
NULL, /* construct */
js_HasInstance,
proxy_Finalize
};
static JSObjectOps *
obj_proxy_getObjectOps(JSContext *cx, JSClass *clasp)
{
return &js_ObjectProxyObjectOps;
}
JS_FRIEND_API(JSClass) ObjectProxyClass = {
"Proxy",
JSCLASS_HAS_RESERVED_SLOTS(2),
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, NULL,
obj_proxy_getObjectOps, NULL, NULL, NULL,
NULL, NULL, NULL, NULL
};
JSBool
proxy_Call(JSContext *cx, uintN argc, jsval *vp)
{
JSObject *proxy = JSVAL_TO_OBJECT(JS_CALLEE(cx, vp));
JS_ASSERT(proxy->isProxy());
return JSProxy::call(cx, proxy, argc, vp);
}
JSBool
proxy_Construct(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
JSObject *proxy = JSVAL_TO_OBJECT(argv[-2]);
JS_ASSERT(proxy->isProxy());
return JSProxy::construct(cx, proxy, obj, argc, argv, rval);
}
static JSType
proxy_TypeOf_fun(JSContext *cx, JSObject *obj)
{
return JSTYPE_FUNCTION;
}
extern JSObjectOps js_FunctionProxyObjectOps;
static const JSObjectMap SharedFunctionProxyMap(&js_FunctionProxyObjectOps, JSObjectMap::SHAPELESS);
#define proxy_HasInstance js_FunctionClass.hasInstance
JSObjectOps js_FunctionProxyObjectOps = {
&SharedFunctionProxyMap,
proxy_LookupProperty,
proxy_DefineProperty,
proxy_GetProperty,
proxy_SetProperty,
proxy_GetAttributes,
proxy_SetAttributes,
proxy_DeleteProperty,
js_DefaultValue,
js_Enumerate,
proxy_TypeOf_fun,
proxy_TraceObject,
NULL, /* thisObject */
proxy_Call,
proxy_Construct,
proxy_HasInstance,
NULL
};
static JSObjectOps *
fun_proxy_getObjectOps(JSContext *cx, JSClass *clasp)
{
return &js_FunctionProxyObjectOps;
}
JS_FRIEND_API(JSClass) FunctionProxyClass = {
"Proxy",
JSCLASS_HAS_RESERVED_SLOTS(4),
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, NULL,
fun_proxy_getObjectOps, NULL, NULL, NULL,
NULL, NULL, NULL, NULL
};
JS_FRIEND_API(JSObject *)
NewProxyObject(JSContext *cx, JSProxyHandler *handler, jsval priv, JSObject *proto, JSObject *parent,
JSObject *call, JSObject *construct)
{
bool fun = call || construct;
JSClass *clasp = fun ? &FunctionProxyClass : &ObjectProxyClass;
JSObject *obj = NewObjectWithGivenProto(cx, clasp, proto, parent);
if (!obj || (construct && !js_EnsureReservedSlots(cx, obj, 0)))
return NULL;
obj->setSlot(JSSLOT_PROXY_HANDLER, PRIVATE_TO_JSVAL(handler));
obj->setSlot(JSSLOT_PROXY_PRIVATE, priv);
if (fun) {
obj->setSlot(JSSLOT_PROXY_CALL, call ? OBJECT_TO_JSVAL(call) : JSVAL_VOID);
if (construct)
obj->setSlot(JSSLOT_PROXY_CONSTRUCT, construct ? OBJECT_TO_JSVAL(construct) : JSVAL_VOID);
}
return obj;
}
static JSObject *
NonNullObject(JSContext *cx, jsval v)
{
if (JSVAL_IS_PRIMITIVE(v)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_NOT_NONNULL_OBJECT);
return NULL;
}
return JSVAL_TO_OBJECT(v);
}
static JSBool
proxy_create(JSContext *cx, uintN argc, jsval *vp)
{
if (argc < 1) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_MORE_ARGS_NEEDED,
"create", "0", "s");
return false;
}
JSObject *handler;
if (!(handler = NonNullObject(cx, vp[2])))
return false;
JSObject *proto, *parent;
if (argc > 1 && !JSVAL_IS_PRIMITIVE(vp[3])) {
proto = JSVAL_TO_OBJECT(vp[3]);
parent = proto->getParent();
} else {
JS_ASSERT(VALUE_IS_FUNCTION(cx, vp[0]));
proto = NULL;
parent = JSVAL_TO_OBJECT(vp[0])->getParent();
}
JSObject *proxy = NewProxyObject(cx, &JSScriptedProxyHandler::singleton, OBJECT_TO_JSVAL(handler),
proto, parent);
if (!proxy)
return false;
*vp = OBJECT_TO_JSVAL(proxy);
return true;
}
static JSBool
proxy_createFunction(JSContext *cx, uintN argc, jsval *vp)
{
if (argc < 2) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_MORE_ARGS_NEEDED,
"createFunction", "1", "");
return false;
}
JSObject *handler;
if (!(handler = NonNullObject(cx, vp[2])))
return false;
JSObject *proto, *parent;
parent = JSVAL_TO_OBJECT(vp[0])->getParent();
if (!js_GetClassPrototype(cx, parent, JSProto_Function, &proto))
return false;
parent = proto->getParent();
JSObject *call = js_ValueToCallableObject(cx, &vp[3], JSV2F_SEARCH_STACK);
if (!call)
return false;
JSObject *construct = NULL;
if (argc > 2) {
construct = js_ValueToCallableObject(cx, &vp[4], JSV2F_SEARCH_STACK);
if (!construct)
return false;
}
JSObject *proxy = NewProxyObject(cx, &JSScriptedProxyHandler::singleton, OBJECT_TO_JSVAL(handler),
proto, parent, call, construct);
if (!proxy)
return false;
*vp = OBJECT_TO_JSVAL(proxy);
return true;
}
#ifdef DEBUG
static JSBool
proxy_isTrapping(JSContext *cx, uintN argc, jsval *vp)
{
if (argc < 1) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_MORE_ARGS_NEEDED,
"isTrapping", "0", "s");
return false;
}
JSObject *obj;
if (!(obj = NonNullObject(cx, vp[2])))
return false;
*vp = BOOLEAN_TO_JSVAL(obj->isProxy());
return true;
}
static JSBool
proxy_fix(JSContext *cx, uintN argc, jsval *vp)
{
if (argc < 1) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_MORE_ARGS_NEEDED,
"fix", "0", "s");
return false;
}
JSObject *obj;
if (!(obj = NonNullObject(cx, vp[2])))
return false;
if (obj->isProxy()) {
JSBool flag;
if (!FixProxy(cx, obj, &flag))
return false;
*vp = BOOLEAN_TO_JSVAL(flag);
} else {
*vp = JSVAL_TRUE;
}
return true;
}
#endif
static JSFunctionSpec static_methods[] = {
JS_FN("create", proxy_create, 2, 0),
JS_FN("createFunction", proxy_createFunction, 3, 0),
#ifdef DEBUG
JS_FN("isTrapping", proxy_isTrapping, 1, 0),
JS_FN("fix", proxy_fix, 1, 0),
#endif
JS_FS_END
};
extern JSClass CallableObjectClass;
static const uint32 JSSLOT_CALLABLE_CALL = JSSLOT_PRIVATE;
static const uint32 JSSLOT_CALLABLE_CONSTRUCT = JSSLOT_PRIVATE + 1;
static JSBool
callable_Call(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
JSObject *callable = JSVAL_TO_OBJECT(argv[-2]);
JS_ASSERT(callable->getClass() == &CallableObjectClass);
jsval fval = callable->fslots[JSSLOT_CALLABLE_CALL];
return js_InternalCall(cx, obj, fval, argc, argv, rval);
}
static JSBool
callable_Construct(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
{
JSObject *callable = JSVAL_TO_OBJECT(argv[-2]);
JS_ASSERT(callable->getClass() == &CallableObjectClass);
jsval fval = callable->fslots[JSSLOT_CALLABLE_CONSTRUCT];
if (fval == JSVAL_VOID) {
/* We don't have an explicit constructor so allocate a new object and use the call. */
fval = callable->fslots[JSSLOT_CALLABLE_CALL];
JS_ASSERT(JSVAL_IS_OBJECT(fval));
/* callable is the constructor, so get callable.prototype is the proto of the new object. */
if (!callable->getProperty(cx, ATOM_TO_JSID(ATOM(classPrototype)), rval))
return false;
JSObject *proto;
if (!JSVAL_IS_PRIMITIVE(*rval)) {
proto = JSVAL_TO_OBJECT(*rval);
} else {
if (!js_GetClassPrototype(cx, NULL, JSProto_Object, &proto))
return false;
}
JSObject *newobj = NewNativeClassInstance(cx, &js_ObjectClass, proto, proto->getParent());
*rval = OBJECT_TO_JSVAL(newobj);
/* If the call returns an object, return that, otherwise the original newobj. */
if (!js_InternalCall(cx, newobj, callable->fslots[JSSLOT_CALLABLE_CALL],
argc, argv, rval)) {
return false;
}
if (JSVAL_IS_PRIMITIVE(*rval))
*rval = OBJECT_TO_JSVAL(newobj);
return true;
}
return js_InternalCall(cx, obj, fval, argc, argv, rval);
}
JSClass CallableObjectClass = {
"Function",
JSCLASS_HAS_RESERVED_SLOTS(2),
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, NULL,
NULL, NULL, callable_Call, callable_Construct,
NULL, NULL, NULL, NULL
};
JS_FRIEND_API(JSBool)
FixProxy(JSContext *cx, JSObject *proxy, JSBool *bp)
{
AutoValueRooter tvr(cx);
if (!JSProxy::fix(cx, proxy, tvr.addr()))
return false;
if (tvr.value() == JSVAL_VOID) {
*bp = false;
return true;
}
if (OperationInProgress(cx, proxy)) {
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_PROXY_FIX);
return false;
}
JSObject *props;
if (!(props = NonNullObject(cx, tvr.value())))
return false;
JSObject *proto = proxy->getProto();
JSObject *parent = proxy->getParent();
JSClass *clasp = proxy->isFunctionProxy() ? &CallableObjectClass : &js_ObjectClass;
/* Make a blank object from the recipe fix provided to us. */
JSObject *newborn = NewObjectWithGivenProto(cx, clasp, proto, parent);
if (!newborn)
return NULL;
AutoValueRooter tvr2(cx, newborn);
if (clasp == &CallableObjectClass) {
newborn->fslots[JSSLOT_CALLABLE_CALL] = GetCall(proxy);
newborn->fslots[JSSLOT_CALLABLE_CONSTRUCT] = GetConstruct(proxy);
}
{
AutoPendingProxyOperation pending(cx, proxy);
if (!js_PopulateObject(cx, newborn, props))
return false;
}
/* Trade spaces between the newborn object and the proxy. */
proxy->swap(newborn);
/* The GC will dispose of the proxy object. */
*bp = true;
return true;
}
}
JSClass js_ProxyClass = {
"Proxy",
JSCLASS_HAS_CACHED_PROTO(JSProto_Proxy),
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, NULL,
JSCLASS_NO_OPTIONAL_MEMBERS
};
JS_FRIEND_API(JSObject *)
js_InitProxyClass(JSContext *cx, JSObject *obj)
{
JSObject *module = NewObject(cx, &js_ProxyClass, NULL, obj);
if (!module)
return NULL;
if (!JS_DefineProperty(cx, obj, "Proxy", OBJECT_TO_JSVAL(module),
JS_PropertyStub, JS_PropertyStub, 0)) {
return NULL;
}
if (!JS_DefineFunctions(cx, module, static_methods))
return NULL;
return module;
}