mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
1462 lines
46 KiB
C++
1462 lines
46 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
* vim: set ts=8 sw=4 et tw=78:
|
|
*
|
|
* ***** 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 Communicator client code, released
|
|
* March 31, 1998.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 1998
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* 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 ***** */
|
|
|
|
/*
|
|
* JavaScript iterators.
|
|
*/
|
|
#include <string.h> /* for memcpy */
|
|
#include "jstypes.h"
|
|
#include "jsstdint.h"
|
|
#include "jsutil.h"
|
|
#include "jsarena.h"
|
|
#include "jsapi.h"
|
|
#include "jsarray.h"
|
|
#include "jsatom.h"
|
|
#include "jsbool.h"
|
|
#include "jsbuiltins.h"
|
|
#include "jscntxt.h"
|
|
#include "jsversion.h"
|
|
#include "jsexn.h"
|
|
#include "jsfun.h"
|
|
#include "jsgc.h"
|
|
#include "jshashtable.h"
|
|
#include "jsinterp.h"
|
|
#include "jsiter.h"
|
|
#include "jslock.h"
|
|
#include "jsnum.h"
|
|
#include "jsobj.h"
|
|
#include "jsopcode.h"
|
|
#include "jsproxy.h"
|
|
#include "jsscan.h"
|
|
#include "jsscope.h"
|
|
#include "jsscript.h"
|
|
#include "jsstaticcheck.h"
|
|
#include "jstracer.h"
|
|
#include "jsvector.h"
|
|
|
|
#if JS_HAS_XML_SUPPORT
|
|
#include "jsxml.h"
|
|
#endif
|
|
|
|
#include "jscntxtinlines.h"
|
|
#include "jsobjinlines.h"
|
|
#include "jsstrinlines.h"
|
|
|
|
using namespace js;
|
|
|
|
static void iterator_finalize(JSContext *cx, JSObject *obj);
|
|
static void iterator_trace(JSTracer *trc, JSObject *obj);
|
|
static JSObject *iterator_iterator(JSContext *cx, JSObject *obj, JSBool keysonly);
|
|
|
|
Class js_IteratorClass = {
|
|
"Iterator",
|
|
JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_Iterator) | JSCLASS_MARK_IS_TRACE,
|
|
PropertyStub, /* addProperty */
|
|
PropertyStub, /* delProperty */
|
|
PropertyStub, /* getProperty */
|
|
PropertyStub, /* setProperty */
|
|
EnumerateStub,
|
|
ResolveStub,
|
|
ConvertStub,
|
|
iterator_finalize,
|
|
NULL, /* reserved */
|
|
NULL, /* checkAccess */
|
|
NULL, /* call */
|
|
NULL, /* construct */
|
|
NULL, /* xdrObject */
|
|
NULL, /* hasInstance */
|
|
JS_CLASS_TRACE(iterator_trace),
|
|
{
|
|
NULL, /* equality */
|
|
NULL, /* outerObject */
|
|
NULL, /* innerObject */
|
|
iterator_iterator,
|
|
NULL /* wrappedObject */
|
|
}
|
|
};
|
|
|
|
void
|
|
NativeIterator::mark(JSTracer *trc)
|
|
{
|
|
if (isKeyIter())
|
|
MarkIdRange(trc, beginKey(), endKey(), "props");
|
|
else
|
|
MarkValueRange(trc, beginValue(), endValue(), "props");
|
|
if (obj)
|
|
MarkObject(trc, obj, "obj");
|
|
}
|
|
|
|
static void
|
|
iterator_finalize(JSContext *cx, JSObject *obj)
|
|
{
|
|
JS_ASSERT(obj->getClass() == &js_IteratorClass);
|
|
|
|
NativeIterator *ni = obj->getNativeIterator();
|
|
if (ni) {
|
|
cx->free(ni);
|
|
obj->setNativeIterator(NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
iterator_trace(JSTracer *trc, JSObject *obj)
|
|
{
|
|
NativeIterator *ni = obj->getNativeIterator();
|
|
|
|
if (ni)
|
|
ni->mark(trc);
|
|
}
|
|
|
|
struct IdHashPolicy {
|
|
typedef jsid Lookup;
|
|
static HashNumber hash(jsid id) {
|
|
return JSID_BITS(id);
|
|
}
|
|
static bool match(jsid id1, jsid id2) {
|
|
return id1 == id2;
|
|
}
|
|
};
|
|
|
|
typedef HashSet<jsid, IdHashPolicy, ContextAllocPolicy> IdSet;
|
|
|
|
static inline bool
|
|
NewKeyValuePair(JSContext *cx, jsid id, const Value &val, Value *rval)
|
|
{
|
|
Value vec[2] = { IdToValue(id), val };
|
|
AutoArrayRooter tvr(cx, JS_ARRAY_LENGTH(vec), vec);
|
|
|
|
JSObject *aobj = js_NewArrayObject(cx, 2, vec);
|
|
if (!aobj)
|
|
return false;
|
|
rval->setObject(*aobj);
|
|
return true;
|
|
}
|
|
|
|
struct KeyEnumeration
|
|
{
|
|
typedef AutoIdVector ResultVector;
|
|
|
|
static JS_ALWAYS_INLINE bool
|
|
append(JSContext *, AutoIdVector &keys, JSObject *, jsid id, uintN flags)
|
|
{
|
|
JS_ASSERT((flags & JSITER_FOREACH) == 0);
|
|
return keys.append(id);
|
|
}
|
|
};
|
|
|
|
struct ValueEnumeration
|
|
{
|
|
typedef AutoValueVector ResultVector;
|
|
|
|
static JS_ALWAYS_INLINE bool
|
|
append(JSContext *cx, AutoValueVector &vals, JSObject *obj, jsid id, uintN flags)
|
|
{
|
|
JS_ASSERT(flags & JSITER_FOREACH);
|
|
|
|
if (!vals.growBy(1))
|
|
return false;
|
|
|
|
/* Do the lookup on the original object instead of the prototype. */
|
|
Value *vp = vals.end() - 1;
|
|
if (!obj->getProperty(cx, id, vp))
|
|
return false;
|
|
if ((flags & JSITER_KEYVALUE) && !NewKeyValuePair(cx, id, *vp, vp))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
};
|
|
|
|
template <class EnumPolicy>
|
|
static inline bool
|
|
Enumerate(JSContext *cx, JSObject *obj, JSObject *pobj, jsid id,
|
|
bool enumerable, bool sharedPermanent, uintN flags, IdSet& ht,
|
|
typename EnumPolicy::ResultVector &props)
|
|
{
|
|
IdSet::AddPtr p = ht.lookupForAdd(id);
|
|
JS_ASSERT_IF(obj == pobj && !obj->isProxy(), !p);
|
|
|
|
/* If we've already seen this, we definitely won't add it. */
|
|
if (JS_UNLIKELY(!!p))
|
|
return true;
|
|
|
|
/*
|
|
* It's not necessary to add properties to the hash table at the end of the
|
|
* prototype chain -- but a proxy might return duplicated properties, so
|
|
* always add for them.
|
|
*/
|
|
if ((pobj->getProto() || pobj->isProxy()) && !ht.add(p, id))
|
|
return false;
|
|
|
|
if (JS_UNLIKELY(flags & JSITER_OWNONLY)) {
|
|
/*
|
|
* Shared-permanent hack: If this property is shared permanent
|
|
* and pobj and obj have the same class, then treat it as an own
|
|
* property of obj, even if pobj != obj. (But see bug 575997.)
|
|
*
|
|
* Omit the magic __proto__ property so that JS code can use
|
|
* Object.getOwnPropertyNames without worrying about it.
|
|
*/
|
|
if (!pobj->getProto() && id == ATOM_TO_JSID(cx->runtime->atomState.protoAtom))
|
|
return true;
|
|
if (pobj != obj && !(sharedPermanent && pobj->getClass() == obj->getClass()))
|
|
return true;
|
|
}
|
|
|
|
if (enumerable || (flags & JSITER_HIDDEN))
|
|
return EnumPolicy::append(cx, props, obj, id, flags);
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class EnumPolicy>
|
|
static bool
|
|
EnumerateNativeProperties(JSContext *cx, JSObject *obj, JSObject *pobj, uintN flags, IdSet &ht,
|
|
typename EnumPolicy::ResultVector &props)
|
|
{
|
|
JS_LOCK_OBJ(cx, pobj);
|
|
|
|
size_t initialLength = props.length();
|
|
|
|
/* Collect all unique properties from this object's scope. */
|
|
JSScope *scope = pobj->scope();
|
|
for (JSScopeProperty *sprop = scope->lastProperty(); sprop; sprop = sprop->parent) {
|
|
if (!JSID_IS_DEFAULT_XML_NAMESPACE(sprop->id) &&
|
|
!sprop->isAlias() &&
|
|
!Enumerate<EnumPolicy>(cx, obj, pobj, sprop->id, sprop->enumerable(), sprop->isSharedPermanent(),
|
|
flags, ht, props))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Reverse(props.begin() + initialLength, props.end());
|
|
|
|
JS_UNLOCK_SCOPE(cx, scope);
|
|
return true;
|
|
}
|
|
|
|
template <class EnumPolicy>
|
|
static bool
|
|
EnumerateDenseArrayProperties(JSContext *cx, JSObject *obj, JSObject *pobj, uintN flags,
|
|
IdSet &ht, typename EnumPolicy::ResultVector &props)
|
|
{
|
|
if (!Enumerate<EnumPolicy>(cx, obj, pobj, ATOM_TO_JSID(cx->runtime->atomState.lengthAtom), false, true,
|
|
flags, ht, props)) {
|
|
return false;
|
|
}
|
|
|
|
if (pobj->getArrayLength() > 0) {
|
|
size_t capacity = pobj->getDenseArrayCapacity();
|
|
Value *vp = pobj->dslots;
|
|
for (size_t i = 0; i < capacity; ++i, ++vp) {
|
|
if (!vp->isMagic(JS_ARRAY_HOLE)) {
|
|
/* Dense arrays never get so large that i would not fit into an integer id. */
|
|
if (!Enumerate<EnumPolicy>(cx, obj, pobj, INT_TO_JSID(i), true, false, flags, ht, props))
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class EnumPolicy>
|
|
static bool
|
|
Snapshot(JSContext *cx, JSObject *obj, uintN flags, typename EnumPolicy::ResultVector &props)
|
|
{
|
|
/*
|
|
* FIXME: Bug 575997 - We won't need to initialize this hash table if
|
|
* (flags & JSITER_OWNONLY) when we eliminate inheritance of
|
|
* shared-permanent properties as own properties.
|
|
*/
|
|
IdSet ht(cx);
|
|
if (!ht.init(32))
|
|
return NULL;
|
|
|
|
JSObject *pobj = obj;
|
|
do {
|
|
Class *clasp = pobj->getClass();
|
|
if (pobj->isNative() &&
|
|
!pobj->getOps()->enumerate &&
|
|
!(clasp->flags & JSCLASS_NEW_ENUMERATE)) {
|
|
if (!clasp->enumerate(cx, pobj))
|
|
return false;
|
|
if (!EnumerateNativeProperties<EnumPolicy>(cx, obj, pobj, flags, ht, props))
|
|
return false;
|
|
} else if (pobj->isDenseArray()) {
|
|
if (!EnumerateDenseArrayProperties<EnumPolicy>(cx, obj, pobj, flags, ht, props))
|
|
return false;
|
|
} else {
|
|
if (pobj->isProxy()) {
|
|
AutoIdVector proxyProps(cx);
|
|
if (flags & JSITER_OWNONLY) {
|
|
if (!JSProxy::enumerateOwn(cx, pobj, proxyProps))
|
|
return false;
|
|
} else {
|
|
if (!JSProxy::enumerate(cx, pobj, proxyProps))
|
|
return false;
|
|
}
|
|
for (size_t n = 0, len = proxyProps.length(); n < len; n++) {
|
|
if (!Enumerate<EnumPolicy>(cx, obj, pobj, proxyProps[n], true, false, flags, ht, props))
|
|
return false;
|
|
}
|
|
/* Proxy objects enumerate the prototype on their own, so we are done here. */
|
|
break;
|
|
}
|
|
Value state;
|
|
JSIterateOp op = (flags & JSITER_HIDDEN) ? JSENUMERATE_INIT_ALL : JSENUMERATE_INIT;
|
|
if (!pobj->enumerate(cx, op, &state, NULL))
|
|
return false;
|
|
if (state.isMagic(JS_NATIVE_ENUMERATE)) {
|
|
if (!EnumerateNativeProperties<EnumPolicy>(cx, obj, pobj, flags, ht, props))
|
|
return false;
|
|
} else {
|
|
while (true) {
|
|
jsid id;
|
|
if (!pobj->enumerate(cx, JSENUMERATE_NEXT, &state, &id))
|
|
return false;
|
|
if (state.isNull())
|
|
break;
|
|
if (!Enumerate<EnumPolicy>(cx, obj, pobj, id, true, false, flags, ht, props))
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (JS_UNLIKELY(pobj->isXML()))
|
|
break;
|
|
} while ((pobj = pobj->getProto()) != NULL);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
VectorToIdArray(JSContext *cx, AutoIdVector &props, JSIdArray **idap)
|
|
{
|
|
JS_STATIC_ASSERT(sizeof(JSIdArray) > sizeof(jsid));
|
|
size_t len = props.length();
|
|
size_t idsz = len * sizeof(jsid);
|
|
size_t sz = (sizeof(JSIdArray) - sizeof(jsid)) + idsz;
|
|
JSIdArray *ida = static_cast<JSIdArray *>(cx->malloc(sz));
|
|
if (!ida)
|
|
return false;
|
|
|
|
ida->length = static_cast<jsint>(len);
|
|
memcpy(ida->vector, props.begin(), idsz);
|
|
*idap = ida;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
GetPropertyNames(JSContext *cx, JSObject *obj, uintN flags, AutoIdVector &props)
|
|
{
|
|
return Snapshot<KeyEnumeration>(cx, obj, flags & (JSITER_OWNONLY | JSITER_HIDDEN), props);
|
|
}
|
|
|
|
static inline bool
|
|
GetCustomIterator(JSContext *cx, JSObject *obj, uintN flags, Value *vp)
|
|
{
|
|
/* Check whether we have a valid __iterator__ method. */
|
|
JSAtom *atom = cx->runtime->atomState.iteratorAtom;
|
|
if (!js_GetMethod(cx, obj, ATOM_TO_JSID(atom), JSGET_NO_METHOD_BARRIER, vp))
|
|
return false;
|
|
|
|
/* If there is no custom __iterator__ method, we are done here. */
|
|
if (vp->isUndefined())
|
|
return true;
|
|
|
|
/* Otherwise call it and return that object. */
|
|
LeaveTrace(cx);
|
|
Value arg = BooleanValue((flags & JSITER_FOREACH) == 0);
|
|
if (!InternalCall(cx, obj, *vp, 1, &arg, vp))
|
|
return false;
|
|
if (vp->isPrimitive()) {
|
|
/*
|
|
* We are always coming from js_ValueToIterator, and we are no longer on
|
|
* trace, so the object we are iterating over is on top of the stack (-1).
|
|
*/
|
|
js_ReportValueError2(cx, JSMSG_BAD_TRAP_RETURN_VALUE,
|
|
-1, ObjectValue(*obj), NULL,
|
|
js_AtomToPrintableString(cx, atom));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <typename T>
|
|
static inline bool
|
|
Compare(T *a, T *b, size_t c)
|
|
{
|
|
size_t n = (c + size_t(7)) / size_t(8);
|
|
switch (c % 8) {
|
|
case 0: do { if (*a++ != *b++) return false;
|
|
case 7: if (*a++ != *b++) return false;
|
|
case 6: if (*a++ != *b++) return false;
|
|
case 5: if (*a++ != *b++) return false;
|
|
case 4: if (*a++ != *b++) return false;
|
|
case 3: if (*a++ != *b++) return false;
|
|
case 2: if (*a++ != *b++) return false;
|
|
case 1: if (*a++ != *b++) return false;
|
|
} while (--n > 0);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static inline JSObject *
|
|
NewIteratorObject(JSContext *cx, uintN flags)
|
|
{
|
|
if (flags & JSITER_ENUMERATE) {
|
|
/*
|
|
* Non-escaping native enumerator objects do not need map, proto, or
|
|
* parent. However, code in jstracer.cpp and elsewhere may find such a
|
|
* native enumerator object via the stack and (as for all objects that
|
|
* are not stillborn, with the exception of "NoSuchMethod" internal
|
|
* helper objects) expect it to have a non-null map pointer, so we
|
|
* share an empty Enumerator scope in the runtime.
|
|
*/
|
|
JSObject *obj = js_NewGCObject(cx);
|
|
if (!obj)
|
|
return false;
|
|
obj->map = cx->runtime->emptyEnumeratorScope->hold();
|
|
obj->init(&js_IteratorClass, NULL, NULL, NullValue());
|
|
return obj;
|
|
}
|
|
|
|
return NewBuiltinClassInstance(cx, &js_IteratorClass);
|
|
}
|
|
|
|
NativeIterator *
|
|
NativeIterator::allocateKeyIterator(JSContext *cx, uint32 slength, const AutoIdVector &props)
|
|
{
|
|
size_t plength = props.length();
|
|
NativeIterator *ni = (NativeIterator *)
|
|
cx->malloc(sizeof(NativeIterator) + plength * sizeof(jsid) + slength * sizeof(uint32));
|
|
if (!ni)
|
|
return NULL;
|
|
ni->props_array = ni->props_cursor = (jsid *) (ni + 1);
|
|
ni->props_end = (jsid *)ni->props_array + plength;
|
|
if (plength)
|
|
memcpy(ni->props_array, props.begin(), plength * sizeof(jsid));
|
|
return ni;
|
|
}
|
|
|
|
NativeIterator *
|
|
NativeIterator::allocateValueIterator(JSContext *cx, uint32 slength, const AutoValueVector &props)
|
|
{
|
|
size_t plength = props.length();
|
|
NativeIterator *ni = (NativeIterator *)
|
|
cx->malloc(sizeof(NativeIterator) + plength * sizeof(Value) + slength * sizeof(uint32));
|
|
if (!ni)
|
|
return NULL;
|
|
ni->props_array = ni->props_cursor = (Value *) (ni + 1);
|
|
ni->props_end = (Value *)ni->props_array + plength;
|
|
if (plength)
|
|
memcpy(ni->props_array, props.begin(), plength * sizeof(Value));
|
|
return ni;
|
|
}
|
|
|
|
inline void
|
|
NativeIterator::init(JSObject *obj, uintN flags, const uint32 *sarray, uint32 slength, uint32 key)
|
|
{
|
|
this->obj = obj;
|
|
this->flags = flags;
|
|
this->shapes_array = (uint32 *) this->props_end;
|
|
this->shapes_length = slength;
|
|
this->shapes_key = key;
|
|
if (slength)
|
|
memcpy(this->shapes_array, sarray, slength * sizeof(uint32));
|
|
}
|
|
|
|
static inline void
|
|
RegisterEnumerator(JSContext *cx, JSObject *iterobj, NativeIterator *ni)
|
|
{
|
|
/* Register non-escaping native enumerators (for-in) with the current context. */
|
|
if (ni->flags & JSITER_ENUMERATE) {
|
|
ni->next = cx->enumerators;
|
|
cx->enumerators = iterobj;
|
|
}
|
|
}
|
|
|
|
static inline bool
|
|
VectorToKeyIterator(JSContext *cx, JSObject *obj, uintN flags, AutoIdVector &keys,
|
|
const uint32 *sarray, uint32 slength, uint32 key, Value *vp)
|
|
{
|
|
JS_ASSERT(!(flags & JSITER_FOREACH));
|
|
|
|
JSObject *iterobj = NewIteratorObject(cx, flags);
|
|
if (!iterobj)
|
|
return false;
|
|
|
|
|
|
NativeIterator *ni = NativeIterator::allocateKeyIterator(cx, slength, keys);
|
|
if (!ni)
|
|
return NULL;
|
|
ni->init(obj, flags, sarray, slength, key);
|
|
|
|
iterobj->setNativeIterator(ni);
|
|
vp->setObject(*iterobj);
|
|
|
|
RegisterEnumerator(cx, iterobj, ni);
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
VectorToKeyIterator(JSContext *cx, JSObject *obj, uintN flags, AutoIdVector &props, Value *vp)
|
|
{
|
|
return VectorToKeyIterator(cx, obj, flags, props, NULL, 0, 0, vp);
|
|
}
|
|
|
|
static inline bool
|
|
VectorToValueIterator(JSContext *cx, JSObject *obj, uintN flags, AutoValueVector &vals,
|
|
const uint32 *sarray, uint32 slength, uint32 key, Value *vp)
|
|
{
|
|
JS_ASSERT(flags & JSITER_FOREACH);
|
|
|
|
JSObject *iterobj = NewIteratorObject(cx, flags);
|
|
if (!iterobj)
|
|
return false;
|
|
|
|
NativeIterator *ni = NativeIterator::allocateValueIterator(cx, slength, vals);
|
|
if (!ni)
|
|
return NULL;
|
|
ni->init(obj, flags, sarray, slength, key);
|
|
|
|
iterobj->setNativeIterator(ni);
|
|
vp->setObject(*iterobj);
|
|
|
|
RegisterEnumerator(cx, iterobj, ni);
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
VectorToValueIterator(JSContext *cx, JSObject *obj, uintN flags, AutoValueVector &props, Value *vp)
|
|
{
|
|
return VectorToValueIterator(cx, obj, flags, props, NULL, 0, 0, vp);
|
|
}
|
|
|
|
bool
|
|
EnumeratedIdVectorToIterator(JSContext *cx, JSObject *obj, uintN flags, AutoIdVector &props, Value *vp)
|
|
{
|
|
if (!(flags & JSITER_FOREACH))
|
|
return VectorToKeyIterator(cx, obj, flags, props, vp);
|
|
|
|
/* For for-each iteration, we need to look up the value of each id. */
|
|
|
|
size_t plength = props.length();
|
|
|
|
AutoValueVector vals(cx);
|
|
if (!vals.reserve(plength))
|
|
return NULL;
|
|
|
|
for (size_t i = 0; i < plength; ++i) {
|
|
if (!ValueEnumeration::append(cx, vals, obj, props[i], flags))
|
|
return false;
|
|
}
|
|
|
|
return VectorToValueIterator(cx, obj, flags, vals, vp);
|
|
}
|
|
|
|
typedef Vector<uint32, 8> ShapeVector;
|
|
|
|
bool
|
|
GetIterator(JSContext *cx, JSObject *obj, uintN flags, Value *vp)
|
|
{
|
|
uint32 hash;
|
|
JSObject **hp;
|
|
Vector<uint32, 8> shapes(cx);
|
|
uint32 key = 0;
|
|
|
|
bool keysOnly = (flags == JSITER_ENUMERATE);
|
|
|
|
if (obj) {
|
|
if (keysOnly) {
|
|
/*
|
|
* The iterator object for JSITER_ENUMERATE never escapes, so we
|
|
* don't care for the proper parent/proto to be set. This also
|
|
* allows us to re-use a previous iterator object that was freed
|
|
* by JSOP_ENDITER.
|
|
*/
|
|
JSObject *pobj = obj;
|
|
do {
|
|
if (!pobj->isNative() ||
|
|
obj->getOps()->enumerate ||
|
|
pobj->getClass()->enumerate != JS_EnumerateStub) {
|
|
shapes.clear();
|
|
goto miss;
|
|
}
|
|
uint32 shape = pobj->shape();
|
|
key = (key + (key << 16)) ^ shape;
|
|
if (!shapes.append(shape))
|
|
return false;
|
|
pobj = pobj->getProto();
|
|
} while (pobj);
|
|
|
|
hash = key % JS_ARRAY_LENGTH(JS_THREAD_DATA(cx)->cachedNativeIterators);
|
|
hp = &JS_THREAD_DATA(cx)->cachedNativeIterators[hash];
|
|
JSObject *iterobj = *hp;
|
|
if (iterobj) {
|
|
NativeIterator *ni = iterobj->getNativeIterator();
|
|
if (ni->shapes_key == key &&
|
|
ni->shapes_length == shapes.length() &&
|
|
Compare(ni->shapes_array, shapes.begin(), ni->shapes_length)) {
|
|
vp->setObject(*iterobj);
|
|
*hp = ni->next;
|
|
|
|
RegisterEnumerator(cx, iterobj, ni);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
miss:
|
|
if (obj->isProxy())
|
|
return JSProxy::iterate(cx, obj, flags, vp);
|
|
if (!GetCustomIterator(cx, obj, flags, vp))
|
|
return false;
|
|
if (!vp->isUndefined())
|
|
return true;
|
|
}
|
|
|
|
/* NB: for (var p in null) succeeds by iterating over no properties. */
|
|
|
|
if (flags & JSITER_FOREACH) {
|
|
AutoValueVector vals(cx);
|
|
if (JS_LIKELY(obj != NULL) && !Snapshot<ValueEnumeration>(cx, obj, flags, vals))
|
|
return false;
|
|
return VectorToValueIterator(cx, obj, flags, vals, shapes.begin(), shapes.length(), key, vp);
|
|
}
|
|
|
|
AutoIdVector keys(cx);
|
|
if (JS_LIKELY(obj != NULL) && !Snapshot<KeyEnumeration>(cx, obj, flags, keys))
|
|
return false;
|
|
return VectorToKeyIterator(cx, obj, flags, keys, shapes.begin(), shapes.length(), key, vp);
|
|
}
|
|
|
|
static JSObject *
|
|
iterator_iterator(JSContext *cx, JSObject *obj, JSBool keysonly)
|
|
{
|
|
return obj;
|
|
}
|
|
|
|
static JSBool
|
|
Iterator(JSContext *cx, JSObject *iterobj, uintN argc, Value *argv, Value *rval)
|
|
{
|
|
JSBool keyonly;
|
|
uintN flags;
|
|
|
|
keyonly = js_ValueToBoolean(argv[1]);
|
|
flags = JSITER_OWNONLY | (keyonly ? 0 : (JSITER_FOREACH | JSITER_KEYVALUE));
|
|
*rval = argv[0];
|
|
return js_ValueToIterator(cx, flags, rval);
|
|
}
|
|
|
|
JSBool
|
|
js_ThrowStopIteration(JSContext *cx)
|
|
{
|
|
Value v;
|
|
|
|
JS_ASSERT(!JS_IsExceptionPending(cx));
|
|
if (js_FindClassObject(cx, NULL, JSProto_StopIteration, &v))
|
|
SetPendingException(cx, v);
|
|
return JS_FALSE;
|
|
}
|
|
|
|
static JSBool
|
|
iterator_next(JSContext *cx, uintN argc, Value *vp)
|
|
{
|
|
JSObject *obj;
|
|
|
|
obj = ComputeThisFromVp(cx, vp);
|
|
if (!InstanceOf(cx, obj, &js_IteratorClass, vp + 2))
|
|
return false;
|
|
|
|
if (!js_IteratorMore(cx, obj, vp))
|
|
return false;
|
|
if (!vp->toBoolean()) {
|
|
js_ThrowStopIteration(cx);
|
|
return false;
|
|
}
|
|
return js_IteratorNext(cx, obj, vp);
|
|
}
|
|
|
|
#define JSPROP_ROPERM (JSPROP_READONLY | JSPROP_PERMANENT)
|
|
|
|
static JSFunctionSpec iterator_methods[] = {
|
|
JS_FN(js_next_str, iterator_next, 0,JSPROP_ROPERM),
|
|
JS_FS_END
|
|
};
|
|
|
|
/*
|
|
* Call ToObject(v).__iterator__(keyonly) if ToObject(v).__iterator__ exists.
|
|
* Otherwise construct the default iterator.
|
|
*/
|
|
JS_FRIEND_API(JSBool)
|
|
js_ValueToIterator(JSContext *cx, uintN flags, Value *vp)
|
|
{
|
|
/* JSITER_KEYVALUE must always come with JSITER_FOREACH */
|
|
JS_ASSERT_IF(flags & JSITER_KEYVALUE, flags & JSITER_FOREACH);
|
|
|
|
/*
|
|
* Make sure the more/next state machine doesn't get stuck. A value might be
|
|
* left in iterValue when a trace is left due to an operation time-out after
|
|
* JSOP_MOREITER but before the value is picked up by FOR*.
|
|
*/
|
|
cx->iterValue.setMagic(JS_NO_ITER_VALUE);
|
|
|
|
JSObject *obj;
|
|
if (vp->isObject()) {
|
|
/* Common case. */
|
|
obj = &vp->toObject();
|
|
} else {
|
|
/*
|
|
* Enumerating over null and undefined gives an empty enumerator.
|
|
* This is contrary to ECMA-262 9.9 ToObject, invoked from step 3 of
|
|
* the first production in 12.6.4 and step 4 of the second production,
|
|
* but it's "web JS" compatible. ES5 fixed for-in to match this de-facto
|
|
* standard.
|
|
*/
|
|
if ((flags & JSITER_ENUMERATE)) {
|
|
if (!js_ValueToObjectOrNull(cx, *vp, &obj))
|
|
return false;
|
|
if (!obj)
|
|
return GetIterator(cx, NULL, flags, vp);
|
|
} else {
|
|
obj = js_ValueToNonNullObject(cx, *vp);
|
|
if (!obj)
|
|
return false;
|
|
}
|
|
}
|
|
|
|
AutoObjectRooter tvr(cx, obj);
|
|
|
|
/* Enumerate Iterator.prototype directly. */
|
|
JSIteratorOp op = obj->getClass()->ext.iteratorObject;
|
|
if (op && (obj->getClass() != &js_IteratorClass || obj->getNativeIterator())) {
|
|
JSObject *iterobj = op(cx, obj, !(flags & JSITER_FOREACH));
|
|
if (!iterobj)
|
|
return false;
|
|
vp->setObject(*iterobj);
|
|
return true;
|
|
}
|
|
|
|
return GetIterator(cx, obj, flags, vp);
|
|
}
|
|
|
|
#if JS_HAS_GENERATORS
|
|
static JS_REQUIRES_STACK JSBool
|
|
CloseGenerator(JSContext *cx, JSObject *genobj);
|
|
#endif
|
|
|
|
JS_FRIEND_API(JSBool)
|
|
js_CloseIterator(JSContext *cx, JSObject *obj)
|
|
{
|
|
cx->iterValue.setMagic(JS_NO_ITER_VALUE);
|
|
|
|
Class *clasp = obj->getClass();
|
|
if (clasp == &js_IteratorClass) {
|
|
/* Remove enumerators from the active list, which is a stack. */
|
|
NativeIterator *ni = obj->getNativeIterator();
|
|
if (ni->flags & JSITER_ENUMERATE) {
|
|
JS_ASSERT(cx->enumerators == obj);
|
|
cx->enumerators = ni->next;
|
|
}
|
|
|
|
/* Cache the iterator object if possible. */
|
|
if (ni->shapes_length) {
|
|
uint32 hash = ni->shapes_key % NATIVE_ITER_CACHE_SIZE;
|
|
JSObject **hp = &JS_THREAD_DATA(cx)->cachedNativeIterators[hash];
|
|
ni->props_cursor = ni->props_array;
|
|
ni->next = *hp;
|
|
*hp = obj;
|
|
}
|
|
}
|
|
#if JS_HAS_GENERATORS
|
|
else if (clasp == &js_GeneratorClass) {
|
|
return CloseGenerator(cx, obj);
|
|
}
|
|
#endif
|
|
return JS_TRUE;
|
|
}
|
|
|
|
/*
|
|
* Suppress enumeration of deleted properties. We maintain a list of all active
|
|
* non-escaping for-in enumerators. Whenever a property is deleted, we check
|
|
* whether any active enumerator contains the (obj, id) pair and has not
|
|
* enumerated id yet. If so, we delete the id from the list (or advance the
|
|
* cursor if it is the next id to be enumerated).
|
|
*
|
|
* We do not suppress enumeration of a property deleted along an object's
|
|
* prototype chain. Only direct deletions on the object are handled.
|
|
*/
|
|
bool
|
|
js_SuppressDeletedProperty(JSContext *cx, JSObject *obj, jsid id)
|
|
{
|
|
JSObject *iterobj = cx->enumerators;
|
|
while (iterobj) {
|
|
again:
|
|
NativeIterator *ni = iterobj->getNativeIterator();
|
|
/* This only works for identified surpressed keys, not values. */
|
|
if (ni->isKeyIter() && ni->obj == obj && ni->props_cursor < ni->props_end) {
|
|
/* Check whether id is still to come. */
|
|
jsid *props_cursor = ni->currentKey();
|
|
jsid *props_end = ni->endKey();
|
|
for (jsid *idp = props_cursor; idp < props_end; ++idp) {
|
|
if (*idp == id) {
|
|
/*
|
|
* Check whether another property along the prototype chain
|
|
* became visible as a result of this deletion.
|
|
*/
|
|
if (obj->getProto()) {
|
|
AutoObjectRooter proto(cx, obj->getProto());
|
|
AutoObjectRooter obj2(cx);
|
|
JSProperty *prop;
|
|
if (!proto.object()->lookupProperty(cx, id, obj2.addr(), &prop))
|
|
return false;
|
|
if (prop) {
|
|
uintN attrs;
|
|
if (obj2.object()->isNative()) {
|
|
attrs = ((JSScopeProperty *) prop)->attributes();
|
|
JS_UNLOCK_OBJ(cx, obj2.object());
|
|
} else if (!obj2.object()->getAttributes(cx, id, &attrs)) {
|
|
return false;
|
|
}
|
|
if (attrs & JSPROP_ENUMERATE)
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If lookupProperty or getAttributes above removed a property from
|
|
* ni, start over.
|
|
*/
|
|
if (props_end != ni->props_end || props_cursor != ni->props_cursor)
|
|
goto again;
|
|
|
|
/*
|
|
* No property along the prototype chain steppeded in to take the
|
|
* property's place, so go ahead and delete id from the list.
|
|
* If it is the next property to be enumerated, just skip it.
|
|
*/
|
|
if (idp == props_cursor) {
|
|
ni->incKeyCursor();
|
|
} else {
|
|
memmove(idp, idp + 1, (props_end - (idp + 1)) * sizeof(jsid));
|
|
ni->props_end = ni->endKey() - 1;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
iterobj = ni->next;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
JSBool
|
|
js_IteratorMore(JSContext *cx, JSObject *iterobj, Value *rval)
|
|
{
|
|
/* Fast path for native iterators */
|
|
if (iterobj->getClass() == &js_IteratorClass) {
|
|
/*
|
|
* Implement next directly as all the methods of native iterator are
|
|
* read-only and permanent.
|
|
*/
|
|
NativeIterator *ni = iterobj->getNativeIterator();
|
|
rval->setBoolean(ni->props_cursor < ni->props_end);
|
|
return true;
|
|
}
|
|
|
|
/* We might still have a pending value. */
|
|
if (!cx->iterValue.isMagic(JS_NO_ITER_VALUE)) {
|
|
rval->setBoolean(true);
|
|
return true;
|
|
}
|
|
|
|
/* Fetch and cache the next value from the iterator. */
|
|
jsid id = ATOM_TO_JSID(cx->runtime->atomState.nextAtom);
|
|
if (!js_GetMethod(cx, iterobj, id, JSGET_METHOD_BARRIER, rval))
|
|
return false;
|
|
if (!InternalCall(cx, iterobj, *rval, 0, NULL, rval)) {
|
|
/* Check for StopIteration. */
|
|
if (!cx->throwing || !js_ValueIsStopIteration(cx->exception))
|
|
return false;
|
|
|
|
/* Inline JS_ClearPendingException(cx). */
|
|
cx->throwing = JS_FALSE;
|
|
cx->exception.setUndefined();
|
|
cx->iterValue.setMagic(JS_NO_ITER_VALUE);
|
|
rval->setBoolean(false);
|
|
return true;
|
|
}
|
|
|
|
/* Cache the value returned by iterobj.next() so js_IteratorNext() can find it. */
|
|
JS_ASSERT(!rval->isMagic(JS_NO_ITER_VALUE));
|
|
cx->iterValue = *rval;
|
|
rval->setBoolean(true);
|
|
return true;
|
|
}
|
|
|
|
JSBool
|
|
js_IteratorNext(JSContext *cx, JSObject *iterobj, Value *rval)
|
|
{
|
|
/* Fast path for native iterators */
|
|
if (iterobj->getClass() == &js_IteratorClass) {
|
|
/*
|
|
* Implement next directly as all the methods of the native iterator are
|
|
* read-only and permanent.
|
|
*/
|
|
NativeIterator *ni = iterobj->getNativeIterator();
|
|
JS_ASSERT(ni->props_cursor < ni->props_end);
|
|
if (ni->isKeyIter()) {
|
|
*rval = IdToValue(*ni->currentKey());
|
|
ni->incKeyCursor();
|
|
} else {
|
|
*rval = *ni->currentValue();
|
|
ni->incValueCursor();
|
|
}
|
|
|
|
if (rval->isString() || !ni->isKeyIter())
|
|
return true;
|
|
|
|
JSString *str;
|
|
jsint i;
|
|
if (rval->isInt32() && (jsuint(i = rval->toInt32()) < INT_STRING_LIMIT)) {
|
|
str = JSString::intString(i);
|
|
} else {
|
|
str = js_ValueToString(cx, *rval);
|
|
if (!str)
|
|
return false;
|
|
}
|
|
|
|
rval->setString(str);
|
|
return true;
|
|
}
|
|
|
|
JS_ASSERT(!cx->iterValue.isMagic(JS_NO_ITER_VALUE));
|
|
*rval = cx->iterValue;
|
|
cx->iterValue.setMagic(JS_NO_ITER_VALUE);
|
|
|
|
return true;
|
|
}
|
|
|
|
static JSBool
|
|
stopiter_hasInstance(JSContext *cx, JSObject *obj, const Value *v, JSBool *bp)
|
|
{
|
|
*bp = js_ValueIsStopIteration(*v);
|
|
return JS_TRUE;
|
|
}
|
|
|
|
Class js_StopIterationClass = {
|
|
js_StopIteration_str,
|
|
JSCLASS_HAS_CACHED_PROTO(JSProto_StopIteration),
|
|
PropertyStub, /* addProperty */
|
|
PropertyStub, /* delProperty */
|
|
PropertyStub, /* getProperty */
|
|
PropertyStub, /* setProperty */
|
|
EnumerateStub,
|
|
ResolveStub,
|
|
ConvertStub,
|
|
NULL, /* finalize */
|
|
NULL, /* reserved0 */
|
|
NULL, /* checkAccess */
|
|
NULL, /* call */
|
|
NULL, /* construct */
|
|
NULL, /* xdrObject */
|
|
stopiter_hasInstance
|
|
};
|
|
|
|
#if JS_HAS_GENERATORS
|
|
|
|
static void
|
|
generator_finalize(JSContext *cx, JSObject *obj)
|
|
{
|
|
JSGenerator *gen = (JSGenerator *) obj->getPrivate();
|
|
if (!gen)
|
|
return;
|
|
|
|
/*
|
|
* gen is open when a script has not called its close method while
|
|
* explicitly manipulating it.
|
|
*/
|
|
JS_ASSERT(gen->state == JSGEN_NEWBORN ||
|
|
gen->state == JSGEN_CLOSED ||
|
|
gen->state == JSGEN_OPEN);
|
|
cx->free(gen);
|
|
}
|
|
|
|
static void
|
|
generator_trace(JSTracer *trc, JSObject *obj)
|
|
{
|
|
JSGenerator *gen = (JSGenerator *) obj->getPrivate();
|
|
if (!gen)
|
|
return;
|
|
|
|
/*
|
|
* Do not mark if the generator is running; the contents may be trash and
|
|
* will be replaced when the generator stops.
|
|
*/
|
|
if (gen->state == JSGEN_RUNNING || gen->state == JSGEN_CLOSING)
|
|
return;
|
|
|
|
JSStackFrame *fp = gen->getFloatingFrame();
|
|
JS_ASSERT(gen->getLiveFrame() == fp);
|
|
MarkValueRange(trc, gen->floatingStack, fp->argEnd(), "generator slots");
|
|
js_TraceStackFrame(trc, fp);
|
|
MarkValueRange(trc, fp->slots(), gen->savedRegs.sp, "generator slots");
|
|
}
|
|
|
|
Class js_GeneratorClass = {
|
|
js_Generator_str,
|
|
JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_Generator) |
|
|
JSCLASS_IS_ANONYMOUS | JSCLASS_MARK_IS_TRACE,
|
|
PropertyStub, /* addProperty */
|
|
PropertyStub, /* delProperty */
|
|
PropertyStub, /* getProperty */
|
|
PropertyStub, /* setProperty */
|
|
EnumerateStub,
|
|
ResolveStub,
|
|
ConvertStub,
|
|
generator_finalize,
|
|
NULL, /* reserved */
|
|
NULL, /* checkAccess */
|
|
NULL, /* call */
|
|
NULL, /* construct */
|
|
NULL, /* xdrObject */
|
|
NULL, /* hasInstance */
|
|
JS_CLASS_TRACE(generator_trace),
|
|
{
|
|
NULL, /* equality */
|
|
NULL, /* outerObject */
|
|
NULL, /* innerObject */
|
|
iterator_iterator,
|
|
NULL, /* wrappedObject */
|
|
}
|
|
};
|
|
|
|
/*
|
|
* Called from the JSOP_GENERATOR case in the interpreter, with fp referring
|
|
* to the frame by which the generator function was activated. Create a new
|
|
* JSGenerator object, which contains its own JSStackFrame that we populate
|
|
* from *fp. We know that upon return, the JSOP_GENERATOR opcode will return
|
|
* from the activation in fp, so we can steal away fp->callobj and fp->argsobj
|
|
* if they are non-null.
|
|
*/
|
|
JS_REQUIRES_STACK JSObject *
|
|
js_NewGenerator(JSContext *cx)
|
|
{
|
|
JSObject *obj = NewBuiltinClassInstance(cx, &js_GeneratorClass);
|
|
if (!obj)
|
|
return NULL;
|
|
|
|
/* Load and compute stack slot counts. */
|
|
JSStackFrame *fp = cx->fp();
|
|
uintN argc = fp->numActualArgs();
|
|
uintN nargs = JS_MAX(argc, fp->numFormalArgs());
|
|
uintN vplen = 2 + nargs;
|
|
|
|
/* Compute JSGenerator size. */
|
|
uintN nbytes = sizeof(JSGenerator) +
|
|
(-1 + /* one Value included in JSGenerator */
|
|
vplen +
|
|
VALUES_PER_STACK_FRAME +
|
|
fp->getSlotCount()) * sizeof(Value);
|
|
|
|
JSGenerator *gen = (JSGenerator *) cx->malloc(nbytes);
|
|
if (!gen)
|
|
return NULL;
|
|
|
|
/* Cut up floatingStack space. */
|
|
Value *vp = gen->floatingStack;
|
|
JSStackFrame *newfp = reinterpret_cast<JSStackFrame *>(vp + vplen);
|
|
Value *slots = newfp->slots();
|
|
|
|
/* Initialize JSGenerator. */
|
|
gen->obj = obj;
|
|
gen->state = JSGEN_NEWBORN;
|
|
gen->savedRegs.pc = cx->regs->pc;
|
|
JS_ASSERT(cx->regs->sp == fp->slots() + fp->getFixedCount());
|
|
gen->savedRegs.sp = slots + fp->getFixedCount();
|
|
gen->vplen = vplen;
|
|
gen->enumerators = NULL;
|
|
gen->liveFrame = newfp;
|
|
|
|
/* Copy generator's stack frame copy in from |cx->fp|. */
|
|
newfp->setCallObj(fp->maybeCallObj());
|
|
if (fp->hasCallObj()) { /* Steal call object. */
|
|
fp->getCallObj()->setPrivate(newfp);
|
|
fp->setCallObj(NULL);
|
|
}
|
|
newfp->setArgsObj(fp->maybeArgsObj());
|
|
if (fp->hasArgsObj()) { /* Steal args object. */
|
|
fp->getArgsObj()->setPrivate(newfp);
|
|
fp->setArgsObj(NULL);
|
|
}
|
|
newfp->setScript(fp->getScript());
|
|
newfp->setFunction(fp->getFunction());
|
|
newfp->setThisValue(fp->getThisValue());
|
|
newfp->setNumActualArgs(fp->numActualArgs());
|
|
newfp->argv = vp + 2;
|
|
newfp->setReturnValue(fp->getReturnValue());
|
|
newfp->setAnnotation(NULL);
|
|
newfp->setScopeChain(fp->maybeScopeChain());
|
|
JS_ASSERT(!fp->hasBlockChain());
|
|
newfp->setBlockChain(NULL);
|
|
newfp->flags = fp->flags | JSFRAME_GENERATOR | JSFRAME_FLOATING_GENERATOR;
|
|
JS_ASSERT(!newfp->hasIMacroPC());
|
|
|
|
/* Copy in arguments and slots. */
|
|
memcpy(vp, fp->argv - 2, vplen * sizeof(Value));
|
|
memcpy(slots, fp->slots(), fp->getFixedCount() * sizeof(Value));
|
|
|
|
obj->setPrivate(gen);
|
|
return obj;
|
|
}
|
|
|
|
JSGenerator *
|
|
js_FloatingFrameToGenerator(JSStackFrame *fp)
|
|
{
|
|
JS_ASSERT(fp->isGenerator() && fp->isFloatingGenerator());
|
|
char *floatingStackp = (char *)(fp->argv - 2);
|
|
char *p = floatingStackp - offsetof(JSGenerator, floatingStack);
|
|
return reinterpret_cast<JSGenerator *>(p);
|
|
}
|
|
|
|
typedef enum JSGeneratorOp {
|
|
JSGENOP_NEXT,
|
|
JSGENOP_SEND,
|
|
JSGENOP_THROW,
|
|
JSGENOP_CLOSE
|
|
} JSGeneratorOp;
|
|
|
|
/*
|
|
* Start newborn or restart yielding generator and perform the requested
|
|
* operation inside its frame.
|
|
*/
|
|
static JS_REQUIRES_STACK JSBool
|
|
SendToGenerator(JSContext *cx, JSGeneratorOp op, JSObject *obj,
|
|
JSGenerator *gen, const Value &arg)
|
|
{
|
|
if (gen->state == JSGEN_RUNNING || gen->state == JSGEN_CLOSING) {
|
|
js_ReportValueError(cx, JSMSG_NESTING_GENERATOR,
|
|
JSDVG_SEARCH_STACK, ObjectOrNullValue(obj),
|
|
JS_GetFunctionId(gen->getFloatingFrame()->getFunction()));
|
|
return JS_FALSE;
|
|
}
|
|
|
|
/* Check for OOM errors here, where we can fail easily. */
|
|
if (!cx->ensureGeneratorStackSpace())
|
|
return JS_FALSE;
|
|
|
|
JS_ASSERT(gen->state == JSGEN_NEWBORN || gen->state == JSGEN_OPEN);
|
|
switch (op) {
|
|
case JSGENOP_NEXT:
|
|
case JSGENOP_SEND:
|
|
if (gen->state == JSGEN_OPEN) {
|
|
/*
|
|
* Store the argument to send as the result of the yield
|
|
* expression.
|
|
*/
|
|
gen->savedRegs.sp[-1] = arg;
|
|
}
|
|
gen->state = JSGEN_RUNNING;
|
|
break;
|
|
|
|
case JSGENOP_THROW:
|
|
SetPendingException(cx, arg);
|
|
gen->state = JSGEN_RUNNING;
|
|
break;
|
|
|
|
default:
|
|
JS_ASSERT(op == JSGENOP_CLOSE);
|
|
SetPendingException(cx, MagicValue(JS_GENERATOR_CLOSING));
|
|
gen->state = JSGEN_CLOSING;
|
|
break;
|
|
}
|
|
|
|
JSStackFrame *genfp = gen->getFloatingFrame();
|
|
JSBool ok;
|
|
{
|
|
Value *genVp = gen->floatingStack;
|
|
uintN vplen = gen->vplen;
|
|
uintN nfixed = genfp->getSlotCount();
|
|
|
|
/*
|
|
* Get a pointer to new frame/slots. This memory is not "claimed", so
|
|
* the code before pushExecuteFrame must not reenter the interpreter.
|
|
*/
|
|
FrameGuard frame;
|
|
if (!cx->stack().getExecuteFrame(cx, cx->maybefp(), vplen, nfixed, frame)) {
|
|
gen->state = JSGEN_CLOSED;
|
|
return JS_FALSE;
|
|
}
|
|
|
|
Value *vp = frame.getvp();
|
|
JSStackFrame *fp = frame.getFrame();
|
|
|
|
/*
|
|
* Copy and rebase stack frame/args/slots. The "floating" flag must
|
|
* only be set on the generator's frame. See args_or_call_trace.
|
|
*/
|
|
uintN usedBefore = gen->savedRegs.sp - genVp;
|
|
memcpy(vp, genVp, usedBefore * sizeof(Value));
|
|
fp->flags &= ~JSFRAME_FLOATING_GENERATOR;
|
|
fp->argv = vp + 2;
|
|
gen->savedRegs.sp = fp->slots() + (gen->savedRegs.sp - genfp->slots());
|
|
JS_ASSERT(uintN(gen->savedRegs.sp - fp->slots()) <= fp->getSlotCount());
|
|
|
|
#ifdef DEBUG
|
|
JSObject *callobjBefore = fp->maybeCallObj();
|
|
JSObject *argsobjBefore = fp->maybeArgsObj();
|
|
#endif
|
|
|
|
/*
|
|
* Repoint Call, Arguments, Block and With objects to the new live
|
|
* frame. Call and Arguments are done directly because we have
|
|
* pointers to them. Block and With objects are done indirectly through
|
|
* 'liveFrame'. See js_LiveFrameToFloating comment in jsiter.h.
|
|
*/
|
|
if (genfp->hasCallObj())
|
|
fp->getCallObj()->setPrivate(fp);
|
|
if (genfp->hasArgsObj())
|
|
fp->getArgsObj()->setPrivate(fp);
|
|
gen->liveFrame = fp;
|
|
(void)cx->enterGenerator(gen); /* OOM check above. */
|
|
|
|
/* Officially push |fp|. |frame|'s destructor pops. */
|
|
cx->stack().pushExecuteFrame(cx, frame, gen->savedRegs, NULL);
|
|
|
|
/* Swap the enumerators stack for the generator's stack. */
|
|
JSObject *enumerators = cx->enumerators;
|
|
cx->enumerators = gen->enumerators;
|
|
|
|
ok = Interpret(cx);
|
|
|
|
/* Restore the original enumerators stack. */
|
|
gen->enumerators = cx->enumerators;
|
|
cx->enumerators = enumerators;
|
|
|
|
/* Restore call/args/block objects. */
|
|
cx->leaveGenerator(gen);
|
|
gen->liveFrame = genfp;
|
|
if (fp->hasArgsObj())
|
|
fp->getArgsObj()->setPrivate(genfp);
|
|
if (fp->hasCallObj())
|
|
fp->getCallObj()->setPrivate(genfp);
|
|
|
|
JS_ASSERT_IF(argsobjBefore, argsobjBefore == fp->maybeArgsObj());
|
|
JS_ASSERT_IF(callobjBefore, callobjBefore == fp->maybeCallObj());
|
|
|
|
/* Copy and rebase stack frame/args/slots. Restore "floating" flag. */
|
|
JS_ASSERT(uintN(gen->savedRegs.sp - fp->slots()) <= fp->getSlotCount());
|
|
uintN usedAfter = gen->savedRegs.sp - vp;
|
|
memcpy(genVp, vp, usedAfter * sizeof(Value));
|
|
genfp->flags |= JSFRAME_FLOATING_GENERATOR;
|
|
genfp->argv = genVp + 2;
|
|
gen->savedRegs.sp = genfp->slots() + (gen->savedRegs.sp - fp->slots());
|
|
JS_ASSERT(uintN(gen->savedRegs.sp - genfp->slots()) <= genfp->getSlotCount());
|
|
}
|
|
|
|
if (gen->getFloatingFrame()->flags & JSFRAME_YIELDING) {
|
|
/* Yield cannot fail, throw or be called on closing. */
|
|
JS_ASSERT(ok);
|
|
JS_ASSERT(!cx->throwing);
|
|
JS_ASSERT(gen->state == JSGEN_RUNNING);
|
|
JS_ASSERT(op != JSGENOP_CLOSE);
|
|
genfp->flags &= ~JSFRAME_YIELDING;
|
|
gen->state = JSGEN_OPEN;
|
|
return JS_TRUE;
|
|
}
|
|
|
|
genfp->clearReturnValue();
|
|
gen->state = JSGEN_CLOSED;
|
|
if (ok) {
|
|
/* Returned, explicitly or by falling off the end. */
|
|
if (op == JSGENOP_CLOSE)
|
|
return JS_TRUE;
|
|
return js_ThrowStopIteration(cx);
|
|
}
|
|
|
|
/*
|
|
* An error, silent termination by operation callback or an exception.
|
|
* Propagate the condition to the caller.
|
|
*/
|
|
return JS_FALSE;
|
|
}
|
|
|
|
static JS_REQUIRES_STACK JSBool
|
|
CloseGenerator(JSContext *cx, JSObject *obj)
|
|
{
|
|
JS_ASSERT(obj->getClass() == &js_GeneratorClass);
|
|
|
|
JSGenerator *gen = (JSGenerator *) obj->getPrivate();
|
|
if (!gen) {
|
|
/* Generator prototype object. */
|
|
return JS_TRUE;
|
|
}
|
|
|
|
if (gen->state == JSGEN_CLOSED)
|
|
return JS_TRUE;
|
|
|
|
return SendToGenerator(cx, JSGENOP_CLOSE, obj, gen, UndefinedValue());
|
|
}
|
|
|
|
/*
|
|
* Common subroutine of generator_(next|send|throw|close) methods.
|
|
*/
|
|
static JSBool
|
|
generator_op(JSContext *cx, JSGeneratorOp op, Value *vp, uintN argc)
|
|
{
|
|
JSObject *obj;
|
|
LeaveTrace(cx);
|
|
|
|
obj = ComputeThisFromVp(cx, vp);
|
|
if (!InstanceOf(cx, obj, &js_GeneratorClass, vp + 2))
|
|
return JS_FALSE;
|
|
|
|
JSGenerator *gen = (JSGenerator *) obj->getPrivate();
|
|
if (!gen) {
|
|
/* This happens when obj is the generator prototype. See bug 352885. */
|
|
goto closed_generator;
|
|
}
|
|
|
|
if (gen->state == JSGEN_NEWBORN) {
|
|
switch (op) {
|
|
case JSGENOP_NEXT:
|
|
case JSGENOP_THROW:
|
|
break;
|
|
|
|
case JSGENOP_SEND:
|
|
if (argc >= 1 && !vp[2].isUndefined()) {
|
|
js_ReportValueError(cx, JSMSG_BAD_GENERATOR_SEND,
|
|
JSDVG_SEARCH_STACK, vp[2], NULL);
|
|
return JS_FALSE;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
JS_ASSERT(op == JSGENOP_CLOSE);
|
|
gen->state = JSGEN_CLOSED;
|
|
return JS_TRUE;
|
|
}
|
|
} else if (gen->state == JSGEN_CLOSED) {
|
|
closed_generator:
|
|
switch (op) {
|
|
case JSGENOP_NEXT:
|
|
case JSGENOP_SEND:
|
|
return js_ThrowStopIteration(cx);
|
|
case JSGENOP_THROW:
|
|
SetPendingException(cx, argc >= 1 ? vp[2] : UndefinedValue());
|
|
return JS_FALSE;
|
|
default:
|
|
JS_ASSERT(op == JSGENOP_CLOSE);
|
|
return JS_TRUE;
|
|
}
|
|
}
|
|
|
|
bool undef = ((op == JSGENOP_SEND || op == JSGENOP_THROW) && argc != 0);
|
|
if (!SendToGenerator(cx, op, obj, gen, undef ? vp[2] : UndefinedValue()))
|
|
return JS_FALSE;
|
|
*vp = gen->getFloatingFrame()->getReturnValue();
|
|
return JS_TRUE;
|
|
}
|
|
|
|
static JSBool
|
|
generator_send(JSContext *cx, uintN argc, Value *vp)
|
|
{
|
|
return generator_op(cx, JSGENOP_SEND, vp, argc);
|
|
}
|
|
|
|
static JSBool
|
|
generator_next(JSContext *cx, uintN argc, Value *vp)
|
|
{
|
|
return generator_op(cx, JSGENOP_NEXT, vp, argc);
|
|
}
|
|
|
|
static JSBool
|
|
generator_throw(JSContext *cx, uintN argc, Value *vp)
|
|
{
|
|
return generator_op(cx, JSGENOP_THROW, vp, argc);
|
|
}
|
|
|
|
static JSBool
|
|
generator_close(JSContext *cx, uintN argc, Value *vp)
|
|
{
|
|
return generator_op(cx, JSGENOP_CLOSE, vp, argc);
|
|
}
|
|
|
|
static JSFunctionSpec generator_methods[] = {
|
|
JS_FN(js_next_str, generator_next, 0,JSPROP_ROPERM),
|
|
JS_FN(js_send_str, generator_send, 1,JSPROP_ROPERM),
|
|
JS_FN(js_throw_str, generator_throw, 1,JSPROP_ROPERM),
|
|
JS_FN(js_close_str, generator_close, 0,JSPROP_ROPERM),
|
|
JS_FS_END
|
|
};
|
|
|
|
#endif /* JS_HAS_GENERATORS */
|
|
|
|
JSObject *
|
|
js_InitIteratorClasses(JSContext *cx, JSObject *obj)
|
|
{
|
|
JSObject *proto, *stop;
|
|
|
|
/* Idempotency required: we initialize several things, possibly lazily. */
|
|
if (!js_GetClassObject(cx, obj, JSProto_StopIteration, &stop))
|
|
return NULL;
|
|
if (stop)
|
|
return stop;
|
|
|
|
proto = js_InitClass(cx, obj, NULL, &js_IteratorClass, Iterator, 2,
|
|
NULL, iterator_methods, NULL, NULL);
|
|
if (!proto)
|
|
return NULL;
|
|
|
|
#if JS_HAS_GENERATORS
|
|
/* Initialize the generator internals if configured. */
|
|
if (!js_InitClass(cx, obj, NULL, &js_GeneratorClass, NULL, 0,
|
|
NULL, generator_methods, NULL, NULL)) {
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
return js_InitClass(cx, obj, NULL, &js_StopIterationClass, NULL, 0,
|
|
NULL, NULL, NULL, NULL);
|
|
}
|