2001-09-20 00:02:59 +00:00
|
|
|
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
1998-03-28 02:44:41 +00:00
|
|
|
*
|
2003-11-15 00:11:16 +00:00
|
|
|
* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
1998-03-28 02:44:41 +00:00
|
|
|
*
|
2003-11-15 00:11:16 +00:00
|
|
|
* 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.
|
1998-03-28 02:44:41 +00:00
|
|
|
*
|
2001-09-20 00:02:59 +00:00
|
|
|
* The Original Code is Mozilla Communicator client code, released
|
|
|
|
* March 31, 1998.
|
1999-09-28 23:12:09 +00:00
|
|
|
*
|
2003-11-15 00:11:16 +00:00
|
|
|
* 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.
|
1999-09-28 23:12:09 +00:00
|
|
|
*
|
2001-01-04 10:13:18 +00:00
|
|
|
* Contributor(s):
|
1999-09-28 23:12:09 +00:00
|
|
|
*
|
2003-11-15 00:11:16 +00:00
|
|
|
* 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 ***** */
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* JS array class.
|
|
|
|
*/
|
1998-06-09 23:04:48 +00:00
|
|
|
#include "jsstddef.h"
|
1998-03-28 02:44:41 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
1998-10-14 10:22:38 +00:00
|
|
|
#include "jstypes.h"
|
|
|
|
#include "jsutil.h" /* Added by JSIFY */
|
1998-03-28 02:44:41 +00:00
|
|
|
#include "jsapi.h"
|
|
|
|
#include "jsarray.h"
|
|
|
|
#include "jsatom.h"
|
|
|
|
#include "jscntxt.h"
|
|
|
|
#include "jsconfig.h"
|
|
|
|
#include "jsfun.h"
|
|
|
|
#include "jsgc.h"
|
|
|
|
#include "jsinterp.h"
|
|
|
|
#include "jslock.h"
|
|
|
|
#include "jsnum.h"
|
|
|
|
#include "jsobj.h"
|
|
|
|
#include "jsstr.h"
|
|
|
|
|
1998-07-31 00:07:22 +00:00
|
|
|
/* 2^32 - 1 as a number and a string */
|
|
|
|
#define MAXINDEX 4294967295u
|
|
|
|
#define MAXSTR "4294967295"
|
1998-06-09 16:47:00 +00:00
|
|
|
|
1998-07-31 00:07:22 +00:00
|
|
|
/*
|
2004-10-05 10:19:07 +00:00
|
|
|
* Determine if the id represents an array index or an XML property index.
|
1998-07-31 00:07:22 +00:00
|
|
|
*
|
1998-06-09 16:47:00 +00:00
|
|
|
* An id is an array index according to ECMA by (15.4):
|
|
|
|
*
|
|
|
|
* "Array objects give special treatment to a certain class of property names.
|
|
|
|
* A property name P (in the form of a string value) is an array index if and
|
1998-07-31 00:07:22 +00:00
|
|
|
* only if ToString(ToUint32(P)) is equal to P and ToUint32(P) is not equal
|
1998-06-09 16:47:00 +00:00
|
|
|
* to 2^32-1."
|
1998-07-31 00:07:22 +00:00
|
|
|
*
|
1998-06-09 16:47:00 +00:00
|
|
|
* In our implementation, it would be sufficient to check for JSVAL_IS_INT(id)
|
1998-07-31 00:07:22 +00:00
|
|
|
* except that by using signed 32-bit integers we miss the top half of the
|
1998-06-09 16:47:00 +00:00
|
|
|
* valid range. This function checks the string representation itself; note
|
1998-07-31 00:07:22 +00:00
|
|
|
* that calling a standard conversion routine might allow strings such as
|
1998-06-09 16:47:00 +00:00
|
|
|
* "08" or "4.0" as array indices, which they are not.
|
|
|
|
*/
|
2004-10-05 10:19:07 +00:00
|
|
|
JSBool
|
|
|
|
js_IdIsIndex(jsval id, jsuint *indexp)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
JSString *str;
|
1998-07-31 00:07:22 +00:00
|
|
|
jschar *cp;
|
1998-04-24 00:31:11 +00:00
|
|
|
|
|
|
|
if (JSVAL_IS_INT(id)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
jsint i;
|
|
|
|
i = JSVAL_TO_INT(id);
|
|
|
|
if (i < 0)
|
|
|
|
return JS_FALSE;
|
|
|
|
*indexp = (jsuint)i;
|
|
|
|
return JS_TRUE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
|
2004-10-05 10:19:07 +00:00
|
|
|
/* NB: id should be a string, but jsxml.c may call us with an object id. */
|
|
|
|
if (!JSVAL_IS_STRING(id))
|
|
|
|
return JS_FALSE;
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
str = JSVAL_TO_STRING(id);
|
2001-10-25 00:26:38 +00:00
|
|
|
cp = JSSTRING_CHARS(str);
|
|
|
|
if (JS7_ISDEC(*cp) && JSSTRING_LENGTH(str) < sizeof(MAXSTR)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
jsuint index = JS7_UNDEC(*cp++);
|
|
|
|
jsuint oldIndex = 0;
|
|
|
|
jsuint c = 0;
|
|
|
|
if (index != 0) {
|
|
|
|
while (JS7_ISDEC(*cp)) {
|
|
|
|
oldIndex = index;
|
|
|
|
c = JS7_UNDEC(*cp);
|
|
|
|
index = 10*index + c;
|
|
|
|
cp++;
|
|
|
|
}
|
|
|
|
}
|
2004-10-05 10:19:07 +00:00
|
|
|
|
|
|
|
/* Ensure that all characters were consumed and we didn't overflow. */
|
2003-06-12 00:26:40 +00:00
|
|
|
if (*cp == 0 &&
|
|
|
|
(oldIndex < (MAXINDEX / 10) ||
|
|
|
|
(oldIndex == (MAXINDEX / 10) && c < (MAXINDEX % 10))))
|
|
|
|
{
|
|
|
|
*indexp = index;
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
1998-06-09 16:47:00 +00:00
|
|
|
}
|
1998-07-31 00:07:22 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
ValueIsLength(JSContext *cx, jsval v, jsuint *lengthp)
|
|
|
|
{
|
|
|
|
jsint i;
|
2001-10-26 02:35:01 +00:00
|
|
|
jsdouble d;
|
1998-07-31 00:07:22 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (JSVAL_IS_INT(v)) {
|
2001-10-26 02:35:01 +00:00
|
|
|
i = JSVAL_TO_INT(v);
|
1999-09-21 18:58:51 +00:00
|
|
|
if (i < 0) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
2001-10-26 02:35:01 +00:00
|
|
|
JSMSG_BAD_ARRAY_LENGTH);
|
1999-09-21 18:58:51 +00:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2001-10-26 02:35:01 +00:00
|
|
|
*lengthp = (jsuint) i;
|
|
|
|
return JS_TRUE;
|
1998-07-31 00:07:22 +00:00
|
|
|
}
|
2004-10-05 10:19:07 +00:00
|
|
|
|
2001-10-26 02:35:01 +00:00
|
|
|
if (!js_ValueToNumber(cx, v, &d)) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
2002-09-20 18:45:05 +00:00
|
|
|
JSMSG_BAD_ARRAY_LENGTH);
|
|
|
|
return JS_FALSE;
|
2001-10-26 02:35:01 +00:00
|
|
|
}
|
|
|
|
if (!js_DoubleToECMAUint32(cx, d, (uint32 *)lengthp)) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
2002-09-20 18:45:05 +00:00
|
|
|
JSMSG_BAD_ARRAY_LENGTH);
|
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
2001-10-26 02:35:01 +00:00
|
|
|
if (JSDOUBLE_IS_NaN(d) || d != *lengthp) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BAD_ARRAY_LENGTH);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
return JS_TRUE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
JSBool
|
|
|
|
js_GetLengthProperty(JSContext *cx, JSObject *obj, jsuint *lengthp)
|
|
|
|
{
|
|
|
|
jsid id;
|
|
|
|
jsint i;
|
|
|
|
jsval v;
|
|
|
|
|
2004-10-05 10:19:07 +00:00
|
|
|
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
|
|
|
|
/* Short-circuit, because js_ValueToECMAUint32 fails when
|
|
|
|
* called during init time.
|
|
|
|
*/
|
|
|
|
if (JSVAL_IS_INT(v)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
i = JSVAL_TO_INT(v);
|
|
|
|
/* jsuint cast does ToUint32. */
|
|
|
|
*lengthp = (jsuint)i;
|
|
|
|
return JS_TRUE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
return js_ValueToECMAUint32(cx, v, (uint32 *)lengthp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
IndexToValue(JSContext *cx, jsuint length, jsval *vp)
|
|
|
|
{
|
|
|
|
if (length <= JSVAL_INT_MAX) {
|
2003-06-12 00:26:40 +00:00
|
|
|
*vp = INT_TO_JSVAL(length);
|
|
|
|
return JS_TRUE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
return js_NewDoubleValue(cx, (jsdouble)length, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
IndexToId(JSContext *cx, jsuint length, jsid *idp)
|
|
|
|
{
|
|
|
|
JSString *str;
|
|
|
|
JSAtom *atom;
|
|
|
|
|
|
|
|
if (length <= JSVAL_INT_MAX) {
|
2004-10-05 10:19:07 +00:00
|
|
|
*idp = INT_TO_JSID(length);
|
1998-03-28 02:44:41 +00:00
|
|
|
} else {
|
2003-06-12 00:26:40 +00:00
|
|
|
str = js_NumberToString(cx, (jsdouble)length);
|
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
|
|
|
atom = js_AtomizeString(cx, str, 0);
|
|
|
|
if (!atom)
|
|
|
|
return JS_FALSE;
|
2004-10-05 10:19:07 +00:00
|
|
|
*idp = ATOM_TO_JSID(atom);
|
1998-04-24 00:31:11 +00:00
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
JSBool
|
|
|
|
js_SetLengthProperty(JSContext *cx, JSObject *obj, jsuint length)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
|
|
|
jsval v;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsid id;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!IndexToValue(cx, length, &v))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
2004-10-05 10:19:07 +00:00
|
|
|
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
1998-04-24 00:31:11 +00:00
|
|
|
return OBJ_SET_PROPERTY(cx, obj, id, &v);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
JSBool
|
|
|
|
js_HasLengthProperty(JSContext *cx, JSObject *obj, jsuint *lengthp)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
JSErrorReporter older;
|
|
|
|
jsid id;
|
|
|
|
JSBool ok;
|
1998-03-28 02:44:41 +00:00
|
|
|
jsval v;
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
older = JS_SetErrorReporter(cx, NULL);
|
2004-10-05 10:19:07 +00:00
|
|
|
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
1998-04-24 00:31:11 +00:00
|
|
|
ok = OBJ_GET_PROPERTY(cx, obj, id, &v);
|
|
|
|
JS_SetErrorReporter(cx, older);
|
|
|
|
if (!ok)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
return ValueIsLength(cx, v, lengthp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This get function is specific to Array.prototype.length and other array
|
|
|
|
* instance length properties. It calls back through the class get function
|
|
|
|
* in case some magic happens there (see call_getProperty in jsfun.c).
|
|
|
|
*/
|
|
|
|
static JSBool
|
|
|
|
array_length_getter(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
|
|
|
{
|
|
|
|
return OBJ_GET_CLASS(cx, obj)->getProperty(cx, obj, id, vp);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_length_setter(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint newlen, oldlen, slot;
|
|
|
|
jsid id2;
|
1998-03-28 02:44:41 +00:00
|
|
|
jsval junk;
|
|
|
|
|
1999-09-21 18:58:51 +00:00
|
|
|
if (!ValueIsLength(cx, *vp, &newlen))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &oldlen))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
|
|
|
slot = oldlen;
|
|
|
|
while (slot > newlen) {
|
|
|
|
--slot;
|
|
|
|
if (!IndexToId(cx, slot, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_DELETE_PROPERTY(cx, obj, id2, &junk))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
return IndexToValue(cx, newlen, vp);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_addProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint index, length;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2004-10-05 10:19:07 +00:00
|
|
|
if (!js_IdIsIndex(id, &index))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_TRUE;
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
if (index >= length) {
|
2003-06-12 00:26:40 +00:00
|
|
|
length = index + 1;
|
|
|
|
return js_SetLengthProperty(cx, obj, length);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
return JS_TRUE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_convert(JSContext *cx, JSObject *obj, JSType type, jsval *vp)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2004-12-24 00:03:59 +00:00
|
|
|
if (JS_VERSION_IS_1_2(cx)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
|
|
|
return JS_FALSE;
|
|
|
|
switch (type) {
|
|
|
|
case JSTYPE_NUMBER:
|
|
|
|
return IndexToValue(cx, length, vp);
|
|
|
|
case JSTYPE_BOOLEAN:
|
|
|
|
*vp = BOOLEAN_TO_JSVAL(length > 0);
|
|
|
|
return JS_TRUE;
|
|
|
|
default:
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-10-14 10:22:38 +00:00
|
|
|
return js_TryValueOf(cx, obj, type, vp);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
JSClass js_ArrayClass = {
|
|
|
|
"Array",
|
|
|
|
0,
|
1998-04-24 00:31:11 +00:00
|
|
|
array_addProperty, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
|
1999-11-11 21:52:35 +00:00
|
|
|
JS_EnumerateStub, JS_ResolveStub, array_convert, JS_FinalizeStub,
|
|
|
|
JSCLASS_NO_OPTIONAL_MEMBERS
|
1998-03-28 02:44:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_join_sub(JSContext *cx, JSObject *obj, JSString *sep, JSBool literalize,
|
2003-06-12 00:26:40 +00:00
|
|
|
jsval *rval, JSBool localeString)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
JSBool ok;
|
1998-03-28 02:44:41 +00:00
|
|
|
jsval v;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length, index;
|
|
|
|
jschar *chars, *ochars;
|
2001-10-25 00:26:38 +00:00
|
|
|
size_t nchars, growth, seplen, tmplen;
|
1998-03-28 02:44:41 +00:00
|
|
|
const jschar *sepstr;
|
|
|
|
JSString *str;
|
1998-10-14 10:22:38 +00:00
|
|
|
JSHashEntry *he;
|
2000-08-09 21:46:03 +00:00
|
|
|
JSObject *obj2;
|
2004-12-06 23:17:19 +00:00
|
|
|
int stackDummy;
|
|
|
|
|
|
|
|
if (!JS_CHECK_STACK_SIZE(cx, stackDummy)) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_OVER_RECURSED);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
ok = js_GetLengthProperty(cx, obj, &length);
|
|
|
|
if (!ok)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
|
2001-11-07 00:15:44 +00:00
|
|
|
he = js_EnterSharpObject(cx, obj, NULL, &chars);
|
|
|
|
if (!he)
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
if (literalize) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (IS_SHARP(he)) {
|
1998-04-24 00:31:11 +00:00
|
|
|
#if JS_HAS_SHARP_VARS
|
2003-06-12 00:26:40 +00:00
|
|
|
nchars = js_strlen(chars);
|
1998-03-28 02:44:41 +00:00
|
|
|
#else
|
2003-06-12 00:26:40 +00:00
|
|
|
chars[0] = '[';
|
|
|
|
chars[1] = ']';
|
|
|
|
chars[2] = 0;
|
|
|
|
nchars = 2;
|
1998-03-28 02:44:41 +00:00
|
|
|
#endif
|
2003-06-12 00:26:40 +00:00
|
|
|
goto make_string;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate 1 + 3 + 1 for "[", the worst-case closing ", ]", and the
|
|
|
|
* terminating 0.
|
|
|
|
*/
|
|
|
|
growth = (1 + 3 + 1) * sizeof(jschar);
|
|
|
|
if (!chars) {
|
|
|
|
nchars = 0;
|
|
|
|
chars = (jschar *) malloc(growth);
|
|
|
|
if (!chars)
|
|
|
|
goto done;
|
|
|
|
} else {
|
|
|
|
MAKE_SHARP(he);
|
|
|
|
nchars = js_strlen(chars);
|
|
|
|
chars = (jschar *)
|
2000-02-04 02:01:49 +00:00
|
|
|
realloc((ochars = chars), nchars * sizeof(jschar) + growth);
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!chars) {
|
|
|
|
free(ochars);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
chars[nchars++] = '[';
|
1998-03-28 02:44:41 +00:00
|
|
|
} else {
|
2001-11-07 00:15:44 +00:00
|
|
|
/*
|
|
|
|
* Free any sharp variable definition in chars. Normally, we would
|
|
|
|
* MAKE_SHARP(he) so that only the first sharp variable annotation is
|
|
|
|
* a definition, and all the rest are references, but in the current
|
|
|
|
* case of (!literalize), we don't need chars at all.
|
|
|
|
*/
|
|
|
|
if (chars)
|
|
|
|
JS_free(cx, chars);
|
|
|
|
chars = NULL;
|
|
|
|
nchars = 0;
|
|
|
|
|
|
|
|
/* Return the empty string on a cycle as well as on empty join. */
|
|
|
|
if (IS_BUSY(he) || length == 0) {
|
|
|
|
js_LeaveSharpObject(cx, NULL);
|
2003-06-12 00:26:40 +00:00
|
|
|
*rval = JS_GetEmptyStringValue(cx);
|
|
|
|
return ok;
|
|
|
|
}
|
2001-11-07 00:15:44 +00:00
|
|
|
|
|
|
|
/* Flag he as BUSY so we can distinguish a cycle from a join-point. */
|
|
|
|
MAKE_BUSY(he);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
sepstr = NULL;
|
2002-07-02 18:14:01 +00:00
|
|
|
seplen = JSSTRING_LENGTH(sep);
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
v = JSVAL_NULL;
|
|
|
|
for (index = 0; index < length; index++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
ok = JS_GetElement(cx, obj, index, &v);
|
|
|
|
if (!ok)
|
|
|
|
goto done;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2004-12-24 00:03:59 +00:00
|
|
|
if ((!literalize || JS_VERSION_IS_1_2(cx)) &&
|
2004-10-05 10:19:07 +00:00
|
|
|
(JSVAL_IS_VOID(v) || JSVAL_IS_NULL(v))) {
|
2003-06-12 00:26:40 +00:00
|
|
|
str = cx->runtime->emptyString;
|
2000-08-09 21:46:03 +00:00
|
|
|
} else {
|
|
|
|
if (localeString) {
|
2001-11-07 00:15:44 +00:00
|
|
|
if (!js_ValueToObject(cx, v, &obj2) ||
|
|
|
|
!js_TryMethod(cx, obj2,
|
2001-10-25 00:26:38 +00:00
|
|
|
cx->runtime->atomState.toLocaleStringAtom,
|
|
|
|
0, NULL, &v)) {
|
2001-11-07 00:15:44 +00:00
|
|
|
str = NULL;
|
|
|
|
} else {
|
|
|
|
str = js_ValueToString(cx, v);
|
2001-10-25 00:26:38 +00:00
|
|
|
}
|
2001-11-07 00:15:44 +00:00
|
|
|
} else {
|
2000-08-09 21:46:03 +00:00
|
|
|
str = (literalize ? js_ValueToSource : js_ValueToString)(cx, v);
|
2001-11-07 00:15:44 +00:00
|
|
|
}
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!str) {
|
|
|
|
ok = JS_FALSE;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate 3 + 1 at end for ", ", closing bracket, and zero. */
|
|
|
|
growth = (nchars + (sepstr ? seplen : 0) +
|
|
|
|
JSSTRING_LENGTH(str) +
|
|
|
|
3 + 1) * sizeof(jschar);
|
|
|
|
if (!chars) {
|
|
|
|
chars = (jschar *) malloc(growth);
|
|
|
|
if (!chars)
|
|
|
|
goto done;
|
|
|
|
} else {
|
|
|
|
chars = (jschar *) realloc((ochars = chars), growth);
|
|
|
|
if (!chars) {
|
|
|
|
free(ochars);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sepstr) {
|
|
|
|
js_strncpy(&chars[nchars], sepstr, seplen);
|
|
|
|
nchars += seplen;
|
|
|
|
}
|
|
|
|
sepstr = JSSTRING_CHARS(sep);
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2001-10-25 00:26:38 +00:00
|
|
|
tmplen = JSSTRING_LENGTH(str);
|
|
|
|
js_strncpy(&chars[nchars], JSSTRING_CHARS(str), tmplen);
|
|
|
|
nchars += tmplen;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
done:
|
1998-03-28 02:44:41 +00:00
|
|
|
if (literalize) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (chars) {
|
|
|
|
if (JSVAL_IS_VOID(v)) {
|
|
|
|
chars[nchars++] = ',';
|
|
|
|
chars[nchars++] = ' ';
|
|
|
|
}
|
|
|
|
chars[nchars++] = ']';
|
|
|
|
}
|
2001-11-07 00:15:44 +00:00
|
|
|
} else {
|
|
|
|
CLEAR_BUSY(he);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
2001-11-07 00:15:44 +00:00
|
|
|
js_LeaveSharpObject(cx, NULL);
|
1998-03-28 02:44:41 +00:00
|
|
|
if (!ok) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (chars)
|
|
|
|
free(chars);
|
|
|
|
return ok;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
make_string:
|
|
|
|
if (!chars) {
|
2003-06-12 00:26:40 +00:00
|
|
|
JS_ReportOutOfMemory(cx);
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
chars[nchars] = 0;
|
|
|
|
str = js_NewString(cx, chars, nchars, 0);
|
|
|
|
if (!str) {
|
2003-06-12 00:26:40 +00:00
|
|
|
free(chars);
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
*rval = STRING_TO_JSVAL(str);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jschar comma_space_ucstr[] = {',', ' ', 0};
|
|
|
|
static jschar comma_ucstr[] = {',', 0};
|
|
|
|
static JSString comma_space = {2, comma_space_ucstr};
|
|
|
|
static JSString comma = {1, comma_ucstr};
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
#if JS_HAS_TOSOURCE
|
|
|
|
static JSBool
|
|
|
|
array_toSource(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
|
2003-06-12 00:26:40 +00:00
|
|
|
jsval *rval)
|
1998-04-24 00:31:11 +00:00
|
|
|
{
|
2000-08-09 21:46:03 +00:00
|
|
|
return array_join_sub(cx, obj, &comma_space, JS_TRUE, rval, JS_FALSE);
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
static JSBool
|
|
|
|
array_toString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
|
2003-06-12 00:26:40 +00:00
|
|
|
jsval *rval)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
|
|
|
JSBool literalize;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* JS1.2 arrays convert to array literals, with a comma followed by a space
|
|
|
|
* between each element.
|
|
|
|
*/
|
2004-12-24 00:03:59 +00:00
|
|
|
literalize = JS_VERSION_IS_1_2(cx);
|
1998-04-24 00:31:11 +00:00
|
|
|
return array_join_sub(cx, obj, literalize ? &comma_space : &comma,
|
2003-06-12 00:26:40 +00:00
|
|
|
literalize, rval, JS_FALSE);
|
2000-08-09 21:46:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_toLocaleString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
|
2003-06-12 00:26:40 +00:00
|
|
|
jsval *rval)
|
2000-08-09 21:46:03 +00:00
|
|
|
{
|
|
|
|
/*
|
2001-03-10 00:20:42 +00:00
|
|
|
* Passing comma here as the separator. Need a way to get a
|
2001-01-04 10:13:18 +00:00
|
|
|
* locale-specific version.
|
2000-08-09 21:46:03 +00:00
|
|
|
*/
|
2001-03-10 00:20:42 +00:00
|
|
|
return array_join_sub(cx, obj, &comma, JS_FALSE, rval, JS_TRUE);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
2002-03-14 00:10:31 +00:00
|
|
|
static JSBool
|
|
|
|
InitArrayElements(JSContext *cx, JSObject *obj, jsuint length, jsval *vector)
|
|
|
|
{
|
|
|
|
jsuint index;
|
|
|
|
jsid id;
|
|
|
|
|
|
|
|
for (index = 0; index < length; index++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!IndexToId(cx, index, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id, &vector[index]))
|
|
|
|
return JS_FALSE;
|
2002-03-14 00:10:31 +00:00
|
|
|
}
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
static JSBool
|
1998-04-24 00:31:11 +00:00
|
|
|
InitArrayObject(JSContext *cx, JSObject *obj, jsuint length, jsval *vector)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsval v;
|
|
|
|
jsid id;
|
|
|
|
|
|
|
|
if (!IndexToValue(cx, length, &v))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
2004-10-05 10:19:07 +00:00
|
|
|
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
1999-06-23 14:18:56 +00:00
|
|
|
if (!OBJ_DEFINE_PROPERTY(cx, obj, id, v,
|
|
|
|
array_length_getter, array_length_setter,
|
|
|
|
JSPROP_PERMANENT,
|
|
|
|
NULL)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
if (!vector)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_TRUE;
|
2002-03-14 00:10:31 +00:00
|
|
|
return InitArrayElements(cx, obj, length, vector);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
2003-06-12 00:26:40 +00:00
|
|
|
#if JS_HAS_SOME_PERL_FUN
|
|
|
|
/*
|
|
|
|
* Perl-inspired join, reverse, and sort.
|
|
|
|
*/
|
|
|
|
static JSBool
|
|
|
|
array_join(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
|
|
|
JSString *str;
|
|
|
|
|
|
|
|
if (JSVAL_IS_VOID(argv[0]))
|
|
|
|
return array_join_sub(cx, obj, &comma, JS_FALSE, rval, JS_FALSE);
|
|
|
|
str = js_ValueToString(cx, argv[0]);
|
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
|
|
|
argv[0] = STRING_TO_JSVAL(str);
|
|
|
|
return array_join_sub(cx, obj, str, JS_FALSE, rval, JS_FALSE);
|
|
|
|
}
|
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
static JSBool
|
|
|
|
array_reverse(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
|
2003-06-12 00:26:40 +00:00
|
|
|
jsval *rval)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint len, half, i;
|
|
|
|
jsid id, id2;
|
|
|
|
jsval v, v2;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &len))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
half = len / 2;
|
|
|
|
for (i = 0; i < half; i++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!IndexToId(cx, i, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, len - i - 1, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id2, &v2))
|
|
|
|
return JS_FALSE;
|
1999-04-27 15:18:57 +00:00
|
|
|
|
|
|
|
#if JS_HAS_SPARSE_ARRAYS
|
|
|
|
/* This part isn't done yet. */
|
|
|
|
|
|
|
|
if (!OBJ_LOOKUP_PROPERTY(cx, obj, id, &obj2, &prop))
|
2002-02-23 03:49:27 +00:00
|
|
|
return JS_FALSE;
|
|
|
|
if (!prop) {
|
|
|
|
OBJ_DELETE_PROPERTY(cx, obj, id2, &v); /* v is junk. */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
OBJ_DROP_PROPERTY(cx, obj2, prop);
|
1999-04-27 15:18:57 +00:00
|
|
|
#endif
|
|
|
|
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id, &v2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id2, &v))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
*rval = OBJECT_TO_JSVAL(obj);
|
|
|
|
return JS_TRUE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
2002-03-13 01:50:13 +00:00
|
|
|
typedef struct HSortArgs {
|
1998-03-28 02:44:41 +00:00
|
|
|
void *vec;
|
|
|
|
size_t elsize;
|
|
|
|
void *pivot;
|
|
|
|
JSComparator cmp;
|
|
|
|
void *arg;
|
2003-02-27 01:43:44 +00:00
|
|
|
JSBool fastcopy;
|
2002-03-13 01:50:13 +00:00
|
|
|
} HSortArgs;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2001-01-04 10:13:18 +00:00
|
|
|
static int
|
|
|
|
sort_compare(const void *a, const void *b, void *arg);
|
|
|
|
|
2002-05-27 05:53:57 +00:00
|
|
|
static int
|
|
|
|
sort_compare_strings(const void *a, const void *b, void *arg);
|
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
static void
|
2003-02-27 01:43:44 +00:00
|
|
|
HeapSortHelper(JSBool building, HSortArgs *hsa, size_t lo, size_t hi)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
2002-03-13 01:50:13 +00:00
|
|
|
void *pivot, *vec, *vec2, *arg, *a, *b;
|
2001-01-04 10:13:18 +00:00
|
|
|
size_t elsize;
|
|
|
|
JSComparator cmp;
|
2003-02-27 01:43:44 +00:00
|
|
|
JSBool fastcopy;
|
|
|
|
size_t j, hiDiv2;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2003-02-27 01:43:44 +00:00
|
|
|
pivot = hsa->pivot;
|
|
|
|
vec = hsa->vec;
|
|
|
|
elsize = hsa->elsize;
|
2002-03-13 01:50:13 +00:00
|
|
|
vec2 = (char *)vec - 2 * elsize;
|
2003-02-27 01:43:44 +00:00
|
|
|
cmp = hsa->cmp;
|
|
|
|
arg = hsa->arg;
|
2001-01-04 10:13:18 +00:00
|
|
|
|
2003-02-27 01:43:44 +00:00
|
|
|
fastcopy = hsa->fastcopy;
|
|
|
|
#define MEMCPY(p,q,n) \
|
|
|
|
(fastcopy ? (void)(*(jsval*)(p) = *(jsval*)(q)) : (void)memcpy(p, q, n))
|
2001-01-04 10:13:18 +00:00
|
|
|
|
2002-03-13 01:50:13 +00:00
|
|
|
if (lo == 1) {
|
|
|
|
j = 2;
|
|
|
|
b = (char *)vec + elsize;
|
2003-02-27 01:43:44 +00:00
|
|
|
if (j < hi && cmp(vec, b, arg) < 0)
|
2002-03-13 01:50:13 +00:00
|
|
|
j++;
|
|
|
|
a = (char *)vec + (hi - 1) * elsize;
|
|
|
|
b = (char *)vec2 + j * elsize;
|
2003-02-27 01:43:44 +00:00
|
|
|
|
2004-10-05 10:19:07 +00:00
|
|
|
/*
|
2003-02-27 01:43:44 +00:00
|
|
|
* During sorting phase b points to a member of heap that cannot be
|
|
|
|
* bigger then biggest of vec[0] and vec[1], and cmp(a, b, arg) <= 0
|
|
|
|
* always holds.
|
|
|
|
*/
|
|
|
|
if ((building || hi == 2) && cmp(a, b, arg) >= 0)
|
2002-03-13 01:50:13 +00:00
|
|
|
return;
|
|
|
|
|
2003-02-27 01:43:44 +00:00
|
|
|
MEMCPY(pivot, a, elsize);
|
|
|
|
MEMCPY(a, b, elsize);
|
2002-03-13 01:50:13 +00:00
|
|
|
lo = j;
|
|
|
|
} else {
|
|
|
|
a = (char *)vec2 + lo * elsize;
|
2003-02-27 01:43:44 +00:00
|
|
|
MEMCPY(pivot, a, elsize);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
2001-01-04 10:13:18 +00:00
|
|
|
|
2002-03-13 01:50:13 +00:00
|
|
|
hiDiv2 = hi/2;
|
|
|
|
while (lo <= hiDiv2) {
|
|
|
|
j = lo + lo;
|
|
|
|
a = (char *)vec2 + j * elsize;
|
|
|
|
b = (char *)vec + (j - 1) * elsize;
|
|
|
|
if (j < hi && cmp(a, b, arg) < 0)
|
|
|
|
j++;
|
|
|
|
b = (char *)vec2 + j * elsize;
|
|
|
|
if (cmp(pivot, b, arg) >= 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
a = (char *)vec2 + lo * elsize;
|
2003-02-27 01:43:44 +00:00
|
|
|
MEMCPY(a, b, elsize);
|
2002-03-13 01:50:13 +00:00
|
|
|
lo = j;
|
|
|
|
}
|
|
|
|
|
|
|
|
a = (char *)vec2 + lo * elsize;
|
2003-02-27 01:43:44 +00:00
|
|
|
MEMCPY(a, pivot, elsize);
|
|
|
|
#undef MEMCPY
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
1998-10-14 10:22:38 +00:00
|
|
|
JSBool
|
2004-10-05 10:19:07 +00:00
|
|
|
js_HeapSort(void *vec, size_t nel, size_t elsize, JSComparator cmp, void *arg)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
|
|
|
void *pivot;
|
2003-02-27 01:43:44 +00:00
|
|
|
HSortArgs hsa;
|
|
|
|
size_t i;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
pivot = malloc(elsize);
|
|
|
|
if (!pivot)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
2003-02-27 01:43:44 +00:00
|
|
|
hsa.vec = vec;
|
|
|
|
hsa.elsize = elsize;
|
|
|
|
hsa.pivot = pivot;
|
|
|
|
hsa.cmp = cmp;
|
|
|
|
hsa.arg = arg;
|
|
|
|
hsa.fastcopy = (cmp == sort_compare || cmp == sort_compare_strings);
|
|
|
|
|
|
|
|
for (i = nel/2; i != 0; i--)
|
|
|
|
HeapSortHelper(JS_TRUE, &hsa, i, nel);
|
2002-03-13 01:50:13 +00:00
|
|
|
while (nel > 2)
|
2003-02-27 01:43:44 +00:00
|
|
|
HeapSortHelper(JS_FALSE, &hsa, 1, --nel);
|
2002-03-13 01:50:13 +00:00
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
free(pivot);
|
1998-10-14 10:22:38 +00:00
|
|
|
return JS_TRUE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct CompareArgs {
|
|
|
|
JSContext *context;
|
|
|
|
jsval fval;
|
|
|
|
JSBool status;
|
|
|
|
} CompareArgs;
|
|
|
|
|
|
|
|
static int
|
|
|
|
sort_compare(const void *a, const void *b, void *arg)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsval av = *(const jsval *)a, bv = *(const jsval *)b;
|
2000-02-04 02:01:49 +00:00
|
|
|
CompareArgs *ca = (CompareArgs *) arg;
|
1998-03-28 02:44:41 +00:00
|
|
|
JSContext *cx = ca->context;
|
|
|
|
jsdouble cmp = -1;
|
|
|
|
jsval fval, argv[2], rval;
|
|
|
|
JSBool ok;
|
|
|
|
|
|
|
|
fval = ca->fval;
|
|
|
|
if (fval == JSVAL_NULL) {
|
2003-06-12 00:26:40 +00:00
|
|
|
JSString *astr, *bstr;
|
|
|
|
|
|
|
|
if (av == bv) {
|
|
|
|
cmp = 0;
|
|
|
|
} else if (av == JSVAL_VOID || bv == JSVAL_VOID) {
|
|
|
|
/* Put undefined properties at the end. */
|
|
|
|
cmp = (av == JSVAL_VOID) ? 1 : -1;
|
|
|
|
} else if ((astr = js_ValueToString(cx, av)) != NULL &&
|
|
|
|
(bstr = js_ValueToString(cx, bv)) != NULL) {
|
|
|
|
cmp = js_CompareStrings(astr, bstr);
|
|
|
|
} else {
|
|
|
|
ca->status = JS_FALSE;
|
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
} else {
|
2003-06-12 00:26:40 +00:00
|
|
|
argv[0] = av;
|
|
|
|
argv[1] = bv;
|
|
|
|
ok = js_InternalCall(cx,
|
|
|
|
OBJ_GET_PARENT(cx, JSVAL_TO_OBJECT(fval)),
|
|
|
|
fval, 2, argv, &rval);
|
|
|
|
if (ok) {
|
|
|
|
ok = js_ValueToNumber(cx, rval, &cmp);
|
|
|
|
/* Clamp cmp to -1, 0, 1. */
|
|
|
|
if (JSDOUBLE_IS_NaN(cmp)) {
|
|
|
|
/* XXX report some kind of error here? ECMA talks about
|
|
|
|
* 'consistent compare functions' that don't return NaN, but is
|
|
|
|
* silent about what the result should be. So we currently
|
|
|
|
* ignore it.
|
|
|
|
*/
|
|
|
|
cmp = 0;
|
|
|
|
} else if (cmp != 0) {
|
|
|
|
cmp = cmp > 0 ? 1 : -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ca->status = ok;
|
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
return (int)cmp;
|
|
|
|
}
|
|
|
|
|
2002-05-27 05:53:57 +00:00
|
|
|
static int
|
|
|
|
sort_compare_strings(const void *a, const void *b, void *arg)
|
|
|
|
{
|
|
|
|
jsval av = *(const jsval *)a, bv = *(const jsval *)b;
|
|
|
|
|
|
|
|
return (int) js_CompareStrings(JSVAL_TO_STRING(av), JSVAL_TO_STRING(bv));
|
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
/* XXXmccabe do the sort helper functions need to take int? (Or can we claim
|
|
|
|
* that 2^32 * 32 is too large to worry about?) Something dumps when I change
|
|
|
|
* to unsigned int; is qsort using -1 as a fencepost?
|
|
|
|
*/
|
1998-03-28 02:44:41 +00:00
|
|
|
static JSBool
|
|
|
|
array_sort(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
|
|
|
jsval fval;
|
|
|
|
CompareArgs ca;
|
1999-04-27 15:18:57 +00:00
|
|
|
jsuint len, newlen, i;
|
1998-03-28 02:44:41 +00:00
|
|
|
jsval *vec;
|
2004-11-30 17:52:29 +00:00
|
|
|
JSStackFrame *fp;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsid id;
|
2002-07-31 21:42:12 +00:00
|
|
|
size_t nbytes;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2002-05-27 05:53:57 +00:00
|
|
|
/*
|
|
|
|
* Optimize the default compare function case if all of obj's elements
|
|
|
|
* have values of type string.
|
|
|
|
*/
|
|
|
|
JSBool all_strings;
|
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
if (argc > 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (JSVAL_IS_PRIMITIVE(argv[0])) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BAD_SORT_ARG);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
fval = argv[0];
|
2002-05-27 05:53:57 +00:00
|
|
|
all_strings = JS_FALSE; /* non-default compare function */
|
1998-03-28 02:44:41 +00:00
|
|
|
} else {
|
2003-06-12 00:26:40 +00:00
|
|
|
fval = JSVAL_NULL;
|
2002-05-27 05:53:57 +00:00
|
|
|
all_strings = JS_TRUE; /* check for all string values */
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &len))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
2002-11-07 10:51:23 +00:00
|
|
|
if (len == 0) {
|
|
|
|
*rval = OBJECT_TO_JSVAL(obj);
|
2000-06-03 19:00:28 +00:00
|
|
|
return JS_TRUE;
|
2002-11-07 10:51:23 +00:00
|
|
|
}
|
2002-07-31 21:42:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Test for size_t overflow, which could lead to indexing beyond the end
|
|
|
|
* of the malloc'd vector.
|
|
|
|
*/
|
|
|
|
nbytes = len * sizeof(jsval);
|
|
|
|
if (nbytes != (double) len * sizeof(jsval)) {
|
|
|
|
JS_ReportOutOfMemory(cx);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
vec = (jsval *) JS_malloc(cx, nbytes);
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!vec)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1999-04-27 15:18:57 +00:00
|
|
|
#if JS_HAS_SPARSE_ARRAYS
|
|
|
|
newlen = 0;
|
|
|
|
#else
|
|
|
|
newlen = len;
|
|
|
|
#endif
|
|
|
|
|
2004-11-30 17:52:29 +00:00
|
|
|
/* Root vec, clearing it first in case a GC nests while we're filling it. */
|
|
|
|
memset(vec, 0, len * sizeof(jsval));
|
|
|
|
fp = cx->fp;
|
|
|
|
fp->vars = vec;
|
|
|
|
fp->nvars = len;
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
for (i = 0; i < len; i++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
ca.status = IndexToId(cx, i, &id);
|
|
|
|
if (!ca.status)
|
|
|
|
goto out;
|
1999-04-27 15:18:57 +00:00
|
|
|
#if JS_HAS_SPARSE_ARRAYS
|
|
|
|
{
|
|
|
|
JSObject *obj2;
|
|
|
|
JSProperty *prop;
|
|
|
|
ca.status = OBJ_LOOKUP_PROPERTY(cx, obj, id, &obj2, &prop);
|
|
|
|
if (!ca.status)
|
|
|
|
goto out;
|
|
|
|
if (!prop) {
|
|
|
|
vec[i] = JSVAL_VOID;
|
|
|
|
continue;
|
|
|
|
}
|
2002-02-23 03:49:27 +00:00
|
|
|
OBJ_DROP_PROPERTY(cx, obj2, prop);
|
|
|
|
newlen++;
|
1999-04-27 15:18:57 +00:00
|
|
|
}
|
|
|
|
#endif
|
2003-06-12 00:26:40 +00:00
|
|
|
ca.status = OBJ_GET_PROPERTY(cx, obj, id, &vec[i]);
|
|
|
|
if (!ca.status)
|
|
|
|
goto out;
|
2002-05-27 05:53:57 +00:00
|
|
|
|
|
|
|
/* We know JSVAL_IS_STRING yields 0 or 1, so avoid a branch via &=. */
|
2004-10-05 10:19:07 +00:00
|
|
|
all_strings &= JSVAL_IS_STRING(vec[i]);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ca.context = cx;
|
|
|
|
ca.fval = fval;
|
|
|
|
ca.status = JS_TRUE;
|
2002-05-27 05:53:57 +00:00
|
|
|
if (!js_HeapSort(vec, (size_t) len, sizeof(jsval),
|
|
|
|
all_strings ? sort_compare_strings : sort_compare,
|
|
|
|
&ca)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
JS_ReportOutOfMemory(cx);
|
|
|
|
ca.status = JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ca.status) {
|
2003-06-12 00:26:40 +00:00
|
|
|
ca.status = InitArrayElements(cx, obj, newlen, vec);
|
|
|
|
if (ca.status)
|
|
|
|
*rval = OBJECT_TO_JSVAL(obj);
|
1999-04-27 15:18:57 +00:00
|
|
|
#if JS_HAS_SPARSE_ARRAYS
|
|
|
|
/* set length of newly-created array object to old length. */
|
|
|
|
if (ca.status && newlen < len) {
|
|
|
|
ca.status = js_SetLengthProperty(cx, obj, len);
|
|
|
|
|
|
|
|
/* Delete any leftover properties greater than newlen. */
|
|
|
|
while (ca.status && newlen < len) {
|
|
|
|
jsval junk;
|
|
|
|
|
|
|
|
ca.status = !IndexToId(cx, newlen, &id) ||
|
|
|
|
!OBJ_DELETE_PROPERTY(cx, obj, id, &junk);
|
|
|
|
newlen++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1999-04-27 15:18:57 +00:00
|
|
|
|
1998-03-28 02:44:41 +00:00
|
|
|
out:
|
1998-04-24 00:31:11 +00:00
|
|
|
if (vec)
|
2003-06-12 00:26:40 +00:00
|
|
|
JS_free(cx, vec);
|
1998-03-28 02:44:41 +00:00
|
|
|
return ca.status;
|
|
|
|
}
|
2003-06-12 00:26:40 +00:00
|
|
|
#endif /* JS_HAS_SOME_PERL_FUN */
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2003-06-12 00:26:40 +00:00
|
|
|
#if JS_HAS_MORE_PERL_FUN
|
1998-03-28 02:44:41 +00:00
|
|
|
/*
|
2003-06-12 00:26:40 +00:00
|
|
|
* Perl-inspired push, pop, shift, unshift, and splice methods.
|
1998-03-28 02:44:41 +00:00
|
|
|
*/
|
|
|
|
static JSBool
|
|
|
|
array_push(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length;
|
1998-03-28 02:44:41 +00:00
|
|
|
uintN i;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsid id;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
for (i = 0; i < argc; i++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!IndexToId(cx, length + i, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id, &argv[i]))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If JS1.2, follow Perl4 by returning the last thing pushed. Otherwise,
|
|
|
|
* return the new array length.
|
|
|
|
*/
|
|
|
|
length += argc;
|
2004-12-24 00:03:59 +00:00
|
|
|
if (JS_VERSION_IS_1_2(cx)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
*rval = argc ? argv[argc-1] : JSVAL_VOID;
|
1998-04-24 00:31:11 +00:00
|
|
|
} else {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!IndexToValue(cx, length, rval))
|
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
|
|
|
return js_SetLengthProperty(cx, obj, length);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_pop(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint index;
|
|
|
|
jsid id;
|
|
|
|
jsval junk;
|
|
|
|
|
|
|
|
if (!js_GetLengthProperty(cx, obj, &index))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
if (index > 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
index--;
|
|
|
|
if (!IndexToId(cx, index, &id))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2003-06-12 00:26:40 +00:00
|
|
|
/* Get the to-be-deleted property's value into rval. */
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, rval))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!OBJ_DELETE_PROPERTY(cx, obj, id, &junk))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
return js_SetLengthProperty(cx, obj, index);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_shift(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length, i;
|
|
|
|
jsid id, id2;
|
|
|
|
jsval v, junk;
|
|
|
|
|
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
if (length > 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
length--;
|
|
|
|
id = JSVAL_ZERO;
|
|
|
|
|
|
|
|
/* Get the to-be-deleted property's value into rval ASAP. */
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, rval))
|
|
|
|
return JS_FALSE;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Slide down the array above the first element.
|
|
|
|
*/
|
|
|
|
if (length > 0) {
|
|
|
|
for (i = 1; i <= length; i++) {
|
|
|
|
if (!IndexToId(cx, i, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, i - 1, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id2, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Delete the only or last element. */
|
|
|
|
if (!OBJ_DELETE_PROPERTY(cx, obj, id, &junk))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
return js_SetLengthProperty(cx, obj, length);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_unshift(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
|
2003-06-12 00:26:40 +00:00
|
|
|
jsval *rval)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length, last;
|
1998-03-28 02:44:41 +00:00
|
|
|
uintN i;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsid id, id2;
|
|
|
|
jsval v;
|
1999-04-27 15:18:57 +00:00
|
|
|
#if JS_HAS_SPARSE_ARRAYS
|
|
|
|
JSObject *obj2;
|
|
|
|
JSProperty *prop;
|
|
|
|
#endif
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
if (argc > 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
/* Slide up the array to make room for argc at the bottom. */
|
|
|
|
if (length > 0) {
|
|
|
|
last = length;
|
|
|
|
while (last--) {
|
|
|
|
if (!IndexToId(cx, last, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, last + argc, &id2))
|
|
|
|
return JS_FALSE;
|
1999-04-27 15:18:57 +00:00
|
|
|
#if JS_HAS_SPARSE_ARRAYS
|
|
|
|
if (!OBJ_LOOKUP_PROPERTY(cx, obj, id, &obj2, &prop))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!prop) {
|
|
|
|
OBJ_DELETE_PROPERTY(cx, obj, id2, &v); /* v is junk. */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
OBJ_DROP_PROPERTY(cx, obj2, prop);
|
|
|
|
#endif
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id2, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy from argv to the bottom of the array. */
|
|
|
|
for (i = 0; i < argc; i++) {
|
|
|
|
if (!IndexToId(cx, i, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id, &argv[i]))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Follow Perl by returning the new array length. */
|
|
|
|
length += argc;
|
|
|
|
if (!js_SetLengthProperty(cx, obj, length))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
return IndexToValue(cx, length, rval);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_splice(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length, begin, end, count, delta, last;
|
1998-03-28 02:44:41 +00:00
|
|
|
uintN i;
|
|
|
|
jsdouble d;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsid id, id2;
|
|
|
|
jsval v;
|
1998-03-28 02:44:41 +00:00
|
|
|
JSObject *obj2;
|
|
|
|
|
|
|
|
/* Nothing to do if no args. Otherwise lock and load length. */
|
|
|
|
if (argc == 0)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_TRUE;
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
/* Convert the first argument into a starting index. */
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_ValueToNumber(cx, *argv, &d))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
d = js_DoubleToInteger(d);
|
|
|
|
if (d < 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
d += length;
|
|
|
|
if (d < 0)
|
|
|
|
d = 0;
|
1998-04-24 00:31:11 +00:00
|
|
|
} else if (d > length) {
|
2003-06-12 00:26:40 +00:00
|
|
|
d = length;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
begin = (jsuint)d; /* d has been clamped to uint32 */
|
1998-03-28 02:44:41 +00:00
|
|
|
argc--;
|
|
|
|
argv++;
|
|
|
|
|
|
|
|
/* Convert the second argument from a count into a fencepost index. */
|
|
|
|
delta = length - begin;
|
|
|
|
if (argc == 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
count = delta;
|
|
|
|
end = length;
|
1998-03-28 02:44:41 +00:00
|
|
|
} else {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!js_ValueToNumber(cx, *argv, &d))
|
|
|
|
return JS_FALSE;
|
|
|
|
d = js_DoubleToInteger(d);
|
|
|
|
if (d < 0)
|
|
|
|
d = 0;
|
|
|
|
else if (d > delta)
|
|
|
|
d = delta;
|
|
|
|
count = (jsuint)d;
|
|
|
|
end = begin + count;
|
|
|
|
argc--;
|
|
|
|
argv++;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
2004-12-24 00:03:59 +00:00
|
|
|
if (count == 1 && JS_VERSION_IS_1_2(cx)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
/*
|
|
|
|
* JS lacks "list context", whereby in Perl one turns the single
|
|
|
|
* scalar that's spliced out into an array just by assigning it to
|
|
|
|
* @single instead of $single, or by using it as Perl push's first
|
|
|
|
* argument, for instance.
|
|
|
|
*
|
|
|
|
* JS1.2 emulated Perl too closely and returned a non-Array for
|
|
|
|
* the single-splice-out case, requiring callers to test and wrap
|
|
|
|
* in [] if necessary. So JS1.3, default, and other versions all
|
|
|
|
* return an array of length 1 for uniformity.
|
|
|
|
*/
|
|
|
|
if (!IndexToId(cx, begin, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, rval))
|
|
|
|
return JS_FALSE;
|
1998-04-24 00:31:11 +00:00
|
|
|
} else {
|
2004-12-24 00:03:59 +00:00
|
|
|
if (!JS_VERSION_IS_1_2(cx) || count > 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
/*
|
|
|
|
* Create a new array value to return. Our ECMA v2 proposal specs
|
|
|
|
* that splice always returns an array value, even when given no
|
|
|
|
* arguments. We think this is best because it eliminates the need
|
|
|
|
* for callers to do an extra test to handle the empty splice case.
|
|
|
|
*/
|
|
|
|
obj2 = js_NewArrayObject(cx, 0, NULL);
|
|
|
|
if (!obj2)
|
|
|
|
return JS_FALSE;
|
|
|
|
*rval = OBJECT_TO_JSVAL(obj2);
|
1998-04-24 00:31:11 +00:00
|
|
|
|
1998-11-05 00:08:43 +00:00
|
|
|
/* If there are elements to remove, put them into the return value. */
|
|
|
|
if (count > 0) {
|
|
|
|
for (last = begin; last < end; last++) {
|
|
|
|
if (!IndexToId(cx, last, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, last - begin, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj2, id2, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
}
|
2003-06-12 00:26:40 +00:00
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Find the direction (up or down) to copy and make way for argv. */
|
1998-04-24 00:31:11 +00:00
|
|
|
if (argc > count) {
|
2003-06-12 00:26:40 +00:00
|
|
|
delta = (jsuint)argc - count;
|
|
|
|
last = length;
|
|
|
|
/* (uint) end could be 0, so can't use vanilla >= test */
|
|
|
|
while (last-- > end) {
|
|
|
|
if (!IndexToId(cx, last, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, last + delta, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id2, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
length += delta;
|
1998-04-24 00:31:11 +00:00
|
|
|
} else if (argc < count) {
|
2003-06-12 00:26:40 +00:00
|
|
|
delta = count - (jsuint)argc;
|
|
|
|
for (last = end; last < length; last++) {
|
|
|
|
if (!IndexToId(cx, last, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, last - delta, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id2, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
length -= delta;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy from argv into the hole to complete the splice. */
|
|
|
|
for (i = 0; i < argc; i++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!IndexToId(cx, begin + i, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, obj, id, &argv[i]))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Update length in case we deleted elements from the end. */
|
1998-04-24 00:31:11 +00:00
|
|
|
return js_SetLengthProperty(cx, obj, length);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
2003-06-12 00:26:40 +00:00
|
|
|
#endif /* JS_HAS_MORE_PERL_FUN */
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
#if JS_HAS_SEQUENCE_OPS
|
|
|
|
/*
|
|
|
|
* Python-esque sequence operations.
|
|
|
|
*/
|
|
|
|
static JSBool
|
|
|
|
array_concat(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
|
|
|
JSObject *nobj, *aobj;
|
2002-09-20 18:45:05 +00:00
|
|
|
jsuint length, alength, slot;
|
1998-03-28 02:44:41 +00:00
|
|
|
uintN i;
|
2002-09-20 18:45:05 +00:00
|
|
|
jsval v;
|
|
|
|
jsid id, id2;
|
|
|
|
|
|
|
|
/* Treat obj as the first argument; see ECMA 15.4.4.4. */
|
|
|
|
--argv;
|
|
|
|
JS_ASSERT(obj == JSVAL_TO_OBJECT(argv[0]));
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2002-09-20 18:45:05 +00:00
|
|
|
/* Create a new Array object and store it in the rval local root. */
|
1998-04-24 00:31:11 +00:00
|
|
|
nobj = js_NewArrayObject(cx, 0, NULL);
|
1998-03-28 02:44:41 +00:00
|
|
|
if (!nobj)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
2002-09-20 18:45:05 +00:00
|
|
|
*rval = OBJECT_TO_JSVAL(nobj);
|
1998-03-28 02:44:41 +00:00
|
|
|
|
2002-09-20 18:45:05 +00:00
|
|
|
/* Loop over [0, argc] to concat args into nobj, expanding all Arrays. */
|
|
|
|
length = 0;
|
|
|
|
for (i = 0; i <= argc; i++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
v = argv[i];
|
|
|
|
if (JSVAL_IS_OBJECT(v)) {
|
|
|
|
aobj = JSVAL_TO_OBJECT(v);
|
|
|
|
if (aobj && OBJ_GET_CLASS(cx, aobj) == &js_ArrayClass) {
|
2002-09-20 18:45:05 +00:00
|
|
|
if (!OBJ_GET_PROPERTY(cx, aobj,
|
2004-10-05 10:19:07 +00:00
|
|
|
ATOM_TO_JSID(cx->runtime->atomState
|
|
|
|
.lengthAtom),
|
2002-09-20 18:45:05 +00:00
|
|
|
&v)) {
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
if (!ValueIsLength(cx, v, &alength))
|
|
|
|
return JS_FALSE;
|
2003-06-12 00:26:40 +00:00
|
|
|
for (slot = 0; slot < alength; slot++) {
|
|
|
|
if (!IndexToId(cx, slot, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, length + slot, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, aobj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, nobj, id2, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
length += alength;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IndexToId(cx, length, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, nobj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
length++;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
return JS_TRUE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
array_slice(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
|
|
|
JSObject *nobj;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length, begin, end, slot;
|
1998-03-28 02:44:41 +00:00
|
|
|
jsdouble d;
|
1998-04-24 00:31:11 +00:00
|
|
|
jsid id, id2;
|
|
|
|
jsval v;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
nobj = js_NewArrayObject(cx, 0, NULL);
|
1998-03-28 02:44:41 +00:00
|
|
|
if (!nobj)
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
|
1998-04-24 00:31:11 +00:00
|
|
|
if (!js_GetLengthProperty(cx, obj, &length))
|
2003-06-12 00:26:40 +00:00
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
begin = 0;
|
|
|
|
end = length;
|
|
|
|
|
|
|
|
if (argc > 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!js_ValueToNumber(cx, argv[0], &d))
|
|
|
|
return JS_FALSE;
|
|
|
|
d = js_DoubleToInteger(d);
|
|
|
|
if (d < 0) {
|
|
|
|
d += length;
|
|
|
|
if (d < 0)
|
|
|
|
d = 0;
|
|
|
|
} else if (d > length) {
|
|
|
|
d = length;
|
|
|
|
}
|
|
|
|
begin = (jsuint)d;
|
|
|
|
|
|
|
|
if (argc > 1) {
|
|
|
|
if (!js_ValueToNumber(cx, argv[1], &d))
|
|
|
|
return JS_FALSE;
|
|
|
|
d = js_DoubleToInteger(d);
|
|
|
|
if (d < 0) {
|
|
|
|
d += length;
|
|
|
|
if (d < 0)
|
|
|
|
d = 0;
|
|
|
|
} else if (d > length) {
|
|
|
|
d = length;
|
|
|
|
}
|
|
|
|
end = (jsuint)d;
|
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (slot = begin; slot < end; slot++) {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!IndexToId(cx, slot, &id))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!IndexToId(cx, slot - begin, &id2))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_GET_PROPERTY(cx, obj, id, &v))
|
|
|
|
return JS_FALSE;
|
|
|
|
if (!OBJ_SET_PROPERTY(cx, nobj, id2, &v))
|
|
|
|
return JS_FALSE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
*rval = OBJECT_TO_JSVAL(nobj);
|
1998-04-24 00:31:11 +00:00
|
|
|
return JS_TRUE;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
#endif /* JS_HAS_SEQUENCE_OPS */
|
|
|
|
|
|
|
|
static JSFunctionSpec array_methods[] = {
|
1998-04-24 00:31:11 +00:00
|
|
|
#if JS_HAS_TOSOURCE
|
2000-08-09 21:46:03 +00:00
|
|
|
{js_toSource_str, array_toSource, 0,0,0},
|
1998-04-24 00:31:11 +00:00
|
|
|
#endif
|
2000-08-09 21:46:03 +00:00
|
|
|
{js_toString_str, array_toString, 0,0,0},
|
|
|
|
{js_toLocaleString_str, array_toLocaleString, 0,0,0},
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
/* Perl-ish methods. */
|
2003-06-12 00:26:40 +00:00
|
|
|
#if JS_HAS_SOME_PERL_FUN
|
2000-08-09 21:46:03 +00:00
|
|
|
{"join", array_join, 1,0,0},
|
|
|
|
{"reverse", array_reverse, 0,0,0},
|
|
|
|
{"sort", array_sort, 1,0,0},
|
1998-03-28 02:44:41 +00:00
|
|
|
#endif
|
2003-06-12 00:26:40 +00:00
|
|
|
#if JS_HAS_MORE_PERL_FUN
|
2000-08-09 21:46:03 +00:00
|
|
|
{"push", array_push, 1,0,0},
|
|
|
|
{"pop", array_pop, 0,0,0},
|
|
|
|
{"shift", array_shift, 0,0,0},
|
|
|
|
{"unshift", array_unshift, 1,0,0},
|
|
|
|
{"splice", array_splice, 1,0,0},
|
2003-06-12 00:26:40 +00:00
|
|
|
#endif
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
/* Python-esque sequence methods. */
|
|
|
|
#if JS_HAS_SEQUENCE_OPS
|
2000-08-09 21:46:03 +00:00
|
|
|
{"concat", array_concat, 0,0,0},
|
|
|
|
{"slice", array_slice, 0,0,0},
|
1998-03-28 02:44:41 +00:00
|
|
|
#endif
|
|
|
|
|
1999-11-11 21:52:35 +00:00
|
|
|
{0,0,0,0,0}
|
1998-03-28 02:44:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
Array(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
jsuint length;
|
1998-03-28 02:44:41 +00:00
|
|
|
jsval *vector;
|
1998-04-24 00:31:11 +00:00
|
|
|
|
|
|
|
/* If called without new, replace obj with a new Array object. */
|
Fix for bug 99663 (for loop resolves properties of the object being enumerated
with JSRESOLVE_ASSIGNING, wrongly), plus a few miscellaneous bugfixes.
- Combine the JSStackFrame members constructing, special, overrides, and
reserved into a uint32 flags member.
- Separate JOF_ASSIGNING from the JOF_SET bytecode format flag, and impute
JSRESOLVE_ASSIGNING from the presence of JOF_ASSIGNING among the current
opcode's format flags. To handle the for-in loop opcodes, which do more
than simply assign -- in particular, they do property lookups whose resolve
hook outcalls should not be flagged with JSRESOLVE_ASSIGNING -- a new frame
flag, JSFRAME_ASSIGNING, has been added.
- Fix interpreter version selection to respect JS_SetVersion, whose effect on
cx->version is "sticky".
- Fix js_DecompileValueGenerator to deal with JSOP_ENUMELEM -- it never had,
as this testcase shows (it crashes without this patch):
version(120);
eval("function fe(s) { for (it[s] in this); }");
try { fe('rdonly'); } catch (e) { print(e); }
2001-10-03 06:39:30 +00:00
|
|
|
if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
obj = js_NewObject(cx, &js_ArrayClass, NULL, NULL);
|
|
|
|
if (!obj)
|
|
|
|
return JS_FALSE;
|
|
|
|
*rval = OBJECT_TO_JSVAL(obj);
|
1998-04-24 00:31:11 +00:00
|
|
|
}
|
1998-03-28 02:44:41 +00:00
|
|
|
|
|
|
|
if (argc == 0) {
|
2003-06-12 00:26:40 +00:00
|
|
|
length = 0;
|
|
|
|
vector = NULL;
|
2004-12-24 00:03:59 +00:00
|
|
|
} else if (JS_VERSION_IS_1_2(cx)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
length = (jsuint) argc;
|
|
|
|
vector = argv;
|
|
|
|
} else if (argc > 1) {
|
|
|
|
length = (jsuint) argc;
|
|
|
|
vector = argv;
|
|
|
|
} else if (!JSVAL_IS_NUMBER(argv[0])) {
|
|
|
|
length = 1;
|
|
|
|
vector = argv;
|
1998-03-28 02:44:41 +00:00
|
|
|
} else {
|
2003-06-12 00:26:40 +00:00
|
|
|
if (!ValueIsLength(cx, argv[0], &length))
|
|
|
|
return JS_FALSE;
|
|
|
|
vector = NULL;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
1998-04-24 00:31:11 +00:00
|
|
|
return InitArrayObject(cx, obj, length, vector);
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
JSObject *
|
|
|
|
js_InitArrayClass(JSContext *cx, JSObject *obj)
|
|
|
|
{
|
1998-04-24 00:31:11 +00:00
|
|
|
JSObject *proto;
|
|
|
|
|
|
|
|
proto = JS_InitClass(cx, obj, NULL, &js_ArrayClass, Array, 1,
|
2003-06-12 00:26:40 +00:00
|
|
|
NULL, array_methods, NULL, NULL);
|
1998-04-24 00:31:11 +00:00
|
|
|
|
|
|
|
/* Initialize the Array prototype object so it gets a length property. */
|
|
|
|
if (!proto || !InitArrayObject(cx, proto, 0, NULL))
|
2003-06-12 00:26:40 +00:00
|
|
|
return NULL;
|
1998-04-24 00:31:11 +00:00
|
|
|
return proto;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
JSObject *
|
1998-04-24 00:31:11 +00:00
|
|
|
js_NewArrayObject(JSContext *cx, jsuint length, jsval *vector)
|
1998-03-28 02:44:41 +00:00
|
|
|
{
|
|
|
|
JSObject *obj;
|
|
|
|
|
|
|
|
obj = js_NewObject(cx, &js_ArrayClass, NULL, NULL);
|
|
|
|
if (!obj)
|
2003-06-12 00:26:40 +00:00
|
|
|
return NULL;
|
1998-03-28 02:44:41 +00:00
|
|
|
if (!InitArrayObject(cx, obj, length, vector)) {
|
2003-06-12 00:26:40 +00:00
|
|
|
cx->newborn[GCX_OBJECT] = NULL;
|
|
|
|
return NULL;
|
1998-03-28 02:44:41 +00:00
|
|
|
}
|
|
|
|
return obj;
|
|
|
|
}
|