/js/lib/Socket.IO-node/support/expresso/deps/jscoverage/js/jsstr.cpp
http://github.com/onedayitwillmake/RealtimeMultiplayerNodeJs · C++ · 2220 lines · 1751 code · 236 blank · 233 comment · 405 complexity · c3cf6b74a7ab61e8451b13f010618873 MD5 · raw file
- /* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
- * vim: set ts=8 sw=4 et tw=99:
- *
- * ***** BEGIN LICENSE BLOCK *****
- * Version: MPL 1.1/GPL 2.0/LGPL 2.1
- *
- * The contents of this file are subject to the Mozilla Public License Version
- * 1.1 (the "License"); you may not use this file except in compliance with
- * the License. You may obtain a copy of the License at
- * http://www.mozilla.org/MPL/
- *
- * Software distributed under the License is distributed on an "AS IS" basis,
- * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
- * for the specific language governing rights and limitations under the
- * License.
- *
- * The Original Code is Mozilla 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 ***** */
- /*
- * JS string type implementation.
- *
- * In order to avoid unnecessary js_LockGCThing/js_UnlockGCThing calls, these
- * native methods store strings (possibly newborn) converted from their 'this'
- * parameter and arguments on the stack: 'this' conversions at argv[-1], arg
- * conversions at their index (argv[0], argv[1]). This is a legitimate method
- * of rooting things that might lose their newborn root due to subsequent GC
- * allocations in the same native method.
- */
- #include "jsstddef.h"
- #include <stdlib.h>
- #include <string.h>
- #include "jstypes.h"
- #include "jsutil.h" /* Added by JSIFY */
- #include "jshash.h" /* Added by JSIFY */
- #include "jsprf.h"
- #include "jsapi.h"
- #include "jsarray.h"
- #include "jsatom.h"
- #include "jsbool.h"
- #include "jsbuiltins.h"
- #include "jscntxt.h"
- #include "jsversion.h"
- #include "jsgc.h"
- #include "jsinterp.h"
- #include "jslock.h"
- #include "jsnum.h"
- #include "jsobj.h"
- #include "jsopcode.h"
- #include "jsregexp.h"
- #include "jsscope.h"
- #include "jsstr.h"
- #include "jsbit.h"
- #define JSSTRDEP_RECURSION_LIMIT 100
- size_t
- js_MinimizeDependentStrings(JSString *str, int level, JSString **basep)
- {
- JSString *base;
- size_t start, length;
- JS_ASSERT(JSSTRING_IS_DEPENDENT(str));
- base = JSSTRDEP_BASE(str);
- start = JSSTRDEP_START(str);
- if (JSSTRING_IS_DEPENDENT(base)) {
- if (level < JSSTRDEP_RECURSION_LIMIT) {
- start += js_MinimizeDependentStrings(base, level + 1, &base);
- } else {
- do {
- start += JSSTRDEP_START(base);
- base = JSSTRDEP_BASE(base);
- } while (JSSTRING_IS_DEPENDENT(base));
- }
- if (start == 0) {
- JS_ASSERT(JSSTRDEP_IS_PREFIX(str));
- JSPREFIX_SET_BASE(str, base);
- } else if (start <= JSSTRDEP_START_MASK) {
- length = JSSTRDEP_LENGTH(str);
- JSSTRDEP_INIT(str, base, start, length);
- }
- }
- *basep = base;
- return start;
- }
- jschar *
- js_GetDependentStringChars(JSString *str)
- {
- size_t start;
- JSString *base;
- start = js_MinimizeDependentStrings(str, 0, &base);
- JS_ASSERT(start < JSFLATSTR_LENGTH(base));
- return JSFLATSTR_CHARS(base) + start;
- }
- const jschar *
- js_GetStringChars(JSContext *cx, JSString *str)
- {
- if (!js_MakeStringImmutable(cx, str))
- return NULL;
- return JSFLATSTR_CHARS(str);
- }
- JSString * JS_FASTCALL
- js_ConcatStrings(JSContext *cx, JSString *left, JSString *right)
- {
- size_t rn, ln, lrdist, n;
- jschar *rs, *ls, *s;
- JSString *ldep; /* non-null if left should become dependent */
- JSString *str;
- JSSTRING_CHARS_AND_LENGTH(right, rs, rn);
- if (rn == 0)
- return left;
- JSSTRING_CHARS_AND_LENGTH(left, ls, ln);
- if (ln == 0)
- return right;
- if (!JSSTRING_IS_MUTABLE(left)) {
- /* We must copy if left does not own a buffer to realloc. */
- s = (jschar *) JS_malloc(cx, (ln + rn + 1) * sizeof(jschar));
- if (!s)
- return NULL;
- js_strncpy(s, ls, ln);
- ldep = NULL;
- } else {
- /* We can realloc left's space and make it depend on our result. */
- JS_ASSERT(JSSTRING_IS_FLAT(left));
- s = (jschar *) JS_realloc(cx, ls, (ln + rn + 1) * sizeof(jschar));
- if (!s)
- return NULL;
- /* Take care: right could depend on left! */
- lrdist = (size_t)(rs - ls);
- if (lrdist < ln)
- rs = s + lrdist;
- left->u.chars = ls = s;
- ldep = left;
- }
- js_strncpy(s + ln, rs, rn);
- n = ln + rn;
- s[n] = 0;
- str = js_NewString(cx, s, n);
- if (!str) {
- /* Out of memory: clean up any space we (re-)allocated. */
- if (!ldep) {
- JS_free(cx, s);
- } else {
- s = (jschar *) JS_realloc(cx, ls, (ln + 1) * sizeof(jschar));
- if (s)
- left->u.chars = s;
- }
- } else {
- JSFLATSTR_SET_MUTABLE(str);
- /* Morph left into a dependent prefix if we realloc'd its buffer. */
- if (ldep) {
- JSPREFIX_INIT(ldep, str, ln);
- #ifdef DEBUG
- {
- JSRuntime *rt = cx->runtime;
- JS_RUNTIME_METER(rt, liveDependentStrings);
- JS_RUNTIME_METER(rt, totalDependentStrings);
- JS_LOCK_RUNTIME_VOID(rt,
- (rt->strdepLengthSum += (double)ln,
- rt->strdepLengthSquaredSum += (double)ln * (double)ln));
- }
- #endif
- }
- }
- return str;
- }
- const jschar *
- js_UndependString(JSContext *cx, JSString *str)
- {
- size_t n, size;
- jschar *s;
- if (JSSTRING_IS_DEPENDENT(str)) {
- n = JSSTRDEP_LENGTH(str);
- size = (n + 1) * sizeof(jschar);
- s = (jschar *) JS_malloc(cx, size);
- if (!s)
- return NULL;
- js_strncpy(s, JSSTRDEP_CHARS(str), n);
- s[n] = 0;
- JSFLATSTR_INIT(str, s, n);
- #ifdef DEBUG
- {
- JSRuntime *rt = cx->runtime;
- JS_RUNTIME_UNMETER(rt, liveDependentStrings);
- JS_RUNTIME_UNMETER(rt, totalDependentStrings);
- JS_LOCK_RUNTIME_VOID(rt,
- (rt->strdepLengthSum -= (double)n,
- rt->strdepLengthSquaredSum -= (double)n * (double)n));
- }
- #endif
- }
- return JSFLATSTR_CHARS(str);
- }
- JSBool
- js_MakeStringImmutable(JSContext *cx, JSString *str)
- {
- if (JSSTRING_IS_DEPENDENT(str) && !js_UndependString(cx, str)) {
- JS_RUNTIME_METER(cx->runtime, badUndependStrings);
- return JS_FALSE;
- }
- JSFLATSTR_CLEAR_MUTABLE(str);
- return JS_TRUE;
- }
- static JSString *
- ArgToRootedString(JSContext *cx, uintN argc, jsval *vp, uintN arg)
- {
- JSObject *obj;
- JSString *str;
- if (arg >= argc)
- return ATOM_TO_STRING(cx->runtime->atomState.typeAtoms[JSTYPE_VOID]);
- vp += 2 + arg;
- if (JSVAL_IS_OBJECT(*vp)) {
- obj = JSVAL_TO_OBJECT(*vp);
- if (!obj)
- return ATOM_TO_STRING(cx->runtime->atomState.nullAtom);
- if (!OBJ_DEFAULT_VALUE(cx, obj, JSTYPE_STRING, vp))
- return NULL;
- }
- if (JSVAL_IS_STRING(*vp))
- return JSVAL_TO_STRING(*vp);
- if (JSVAL_IS_INT(*vp)) {
- str = js_NumberToString(cx, JSVAL_TO_INT(*vp));
- } else if (JSVAL_IS_DOUBLE(*vp)) {
- str = js_NumberToString(cx, *JSVAL_TO_DOUBLE(*vp));
- } else if (JSVAL_IS_BOOLEAN(*vp)) {
- return ATOM_TO_STRING(cx->runtime->atomState.booleanAtoms[
- JSVAL_TO_BOOLEAN(*vp)? 1 : 0]);
- } else {
- JS_ASSERT(JSVAL_IS_VOID(*vp));
- return ATOM_TO_STRING(cx->runtime->atomState.typeAtoms[JSTYPE_VOID]);
- }
- if (str)
- *vp = STRING_TO_JSVAL(str);
- return str;
- }
- /*
- * Forward declarations for URI encode/decode and helper routines
- */
- static JSBool
- str_decodeURI(JSContext *cx, uintN argc, jsval *vp);
- static JSBool
- str_decodeURI_Component(JSContext *cx, uintN argc, jsval *vp);
- static JSBool
- str_encodeURI(JSContext *cx, uintN argc, jsval *vp);
- static JSBool
- str_encodeURI_Component(JSContext *cx, uintN argc, jsval *vp);
- static uint32
- Utf8ToOneUcs4Char(const uint8 *utf8Buffer, int utf8Length);
- /*
- * Contributions from the String class to the set of methods defined for the
- * global object. escape and unescape used to be defined in the Mocha library,
- * but as ECMA decided to spec them, they've been moved to the core engine
- * and made ECMA-compliant. (Incomplete escapes are interpreted as literal
- * characters by unescape.)
- */
- /*
- * Stuff to emulate the old libmocha escape, which took a second argument
- * giving the type of escape to perform. Retained for compatibility, and
- * copied here to avoid reliance on net.h, mkparse.c/NET_EscapeBytes.
- */
- #define URL_XALPHAS ((uint8) 1)
- #define URL_XPALPHAS ((uint8) 2)
- #define URL_PATH ((uint8) 4)
- static const uint8 urlCharType[256] =
- /* Bit 0 xalpha -- the alphas
- * Bit 1 xpalpha -- as xalpha but
- * converts spaces to plus and plus to %20
- * Bit 2 ... path -- as xalphas but doesn't escape '/'
- */
- /* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
- { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x */
- 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 1x */
- 0,0,0,0,0,0,0,0,0,0,7,4,0,7,7,4, /* 2x !"#$%&'()*+,-./ */
- 7,7,7,7,7,7,7,7,7,7,0,0,0,0,0,0, /* 3x 0123456789:;<=>? */
- 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7, /* 4x @ABCDEFGHIJKLMNO */
- 7,7,7,7,7,7,7,7,7,7,7,0,0,0,0,7, /* 5X PQRSTUVWXYZ[\]^_ */
- 0,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7, /* 6x `abcdefghijklmno */
- 7,7,7,7,7,7,7,7,7,7,7,0,0,0,0,0, /* 7X pqrstuvwxyz{\}~ DEL */
- 0, };
- /* This matches the ECMA escape set when mask is 7 (default.) */
- #define IS_OK(C, mask) (urlCharType[((uint8) (C))] & (mask))
- /* See ECMA-262 Edition 3 B.2.1 */
- JSBool
- js_str_escape(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
- {
- JSString *str;
- size_t i, ni, length, newlength;
- const jschar *chars;
- jschar *newchars;
- jschar ch;
- jsint mask;
- jsdouble d;
- const char digits[] = {'0', '1', '2', '3', '4', '5', '6', '7',
- '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
- mask = URL_XALPHAS | URL_XPALPHAS | URL_PATH;
- if (argc > 1) {
- d = js_ValueToNumber(cx, &argv[1]);
- if (JSVAL_IS_NULL(argv[1]))
- return JS_FALSE;
- if (!JSDOUBLE_IS_FINITE(d) ||
- (mask = (jsint)d) != d ||
- mask & ~(URL_XALPHAS | URL_XPALPHAS | URL_PATH))
- {
- char numBuf[12];
- JS_snprintf(numBuf, sizeof numBuf, "%lx", (unsigned long) mask);
- JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
- JSMSG_BAD_STRING_MASK, numBuf);
- return JS_FALSE;
- }
- }
- str = ArgToRootedString(cx, argc, argv - 2, 0);
- if (!str)
- return JS_FALSE;
- JSSTRING_CHARS_AND_LENGTH(str, chars, length);
- newlength = length;
- /* Take a first pass and see how big the result string will need to be. */
- for (i = 0; i < length; i++) {
- if ((ch = chars[i]) < 128 && IS_OK(ch, mask))
- continue;
- if (ch < 256) {
- if (mask == URL_XPALPHAS && ch == ' ')
- continue; /* The character will be encoded as '+' */
- newlength += 2; /* The character will be encoded as %XX */
- } else {
- newlength += 5; /* The character will be encoded as %uXXXX */
- }
- /*
- * This overflow test works because newlength is incremented by at
- * most 5 on each iteration.
- */
- if (newlength < length) {
- js_ReportAllocationOverflow(cx);
- return JS_FALSE;
- }
- }
- if (newlength >= ~(size_t)0 / sizeof(jschar)) {
- js_ReportAllocationOverflow(cx);
- return JS_FALSE;
- }
- newchars = (jschar *) JS_malloc(cx, (newlength + 1) * sizeof(jschar));
- if (!newchars)
- return JS_FALSE;
- for (i = 0, ni = 0; i < length; i++) {
- if ((ch = chars[i]) < 128 && IS_OK(ch, mask)) {
- newchars[ni++] = ch;
- } else if (ch < 256) {
- if (mask == URL_XPALPHAS && ch == ' ') {
- newchars[ni++] = '+'; /* convert spaces to pluses */
- } else {
- newchars[ni++] = '%';
- newchars[ni++] = digits[ch >> 4];
- newchars[ni++] = digits[ch & 0xF];
- }
- } else {
- newchars[ni++] = '%';
- newchars[ni++] = 'u';
- newchars[ni++] = digits[ch >> 12];
- newchars[ni++] = digits[(ch & 0xF00) >> 8];
- newchars[ni++] = digits[(ch & 0xF0) >> 4];
- newchars[ni++] = digits[ch & 0xF];
- }
- }
- JS_ASSERT(ni == newlength);
- newchars[newlength] = 0;
- str = js_NewString(cx, newchars, newlength);
- if (!str) {
- JS_free(cx, newchars);
- return JS_FALSE;
- }
- *rval = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #undef IS_OK
- static JSBool
- str_escape(JSContext *cx, uintN argc, jsval *vp)
- {
- JSObject *obj;
- obj = JS_THIS_OBJECT(cx, vp);
- return obj && js_str_escape(cx, obj, argc, vp + 2, vp);
- }
- /* See ECMA-262 Edition 3 B.2.2 */
- static JSBool
- str_unescape(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- size_t i, ni, length;
- const jschar *chars;
- jschar *newchars;
- jschar ch;
- str = ArgToRootedString(cx, argc, vp, 0);
- if (!str)
- return JS_FALSE;
- JSSTRING_CHARS_AND_LENGTH(str, chars, length);
- /* Don't bother allocating less space for the new string. */
- newchars = (jschar *) JS_malloc(cx, (length + 1) * sizeof(jschar));
- if (!newchars)
- return JS_FALSE;
- ni = i = 0;
- while (i < length) {
- ch = chars[i++];
- if (ch == '%') {
- if (i + 1 < length &&
- JS7_ISHEX(chars[i]) && JS7_ISHEX(chars[i + 1]))
- {
- ch = JS7_UNHEX(chars[i]) * 16 + JS7_UNHEX(chars[i + 1]);
- i += 2;
- } else if (i + 4 < length && chars[i] == 'u' &&
- JS7_ISHEX(chars[i + 1]) && JS7_ISHEX(chars[i + 2]) &&
- JS7_ISHEX(chars[i + 3]) && JS7_ISHEX(chars[i + 4]))
- {
- ch = (((((JS7_UNHEX(chars[i + 1]) << 4)
- + JS7_UNHEX(chars[i + 2])) << 4)
- + JS7_UNHEX(chars[i + 3])) << 4)
- + JS7_UNHEX(chars[i + 4]);
- i += 5;
- }
- }
- newchars[ni++] = ch;
- }
- newchars[ni] = 0;
- str = js_NewString(cx, newchars, ni);
- if (!str) {
- JS_free(cx, newchars);
- return JS_FALSE;
- }
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #if JS_HAS_UNEVAL
- static JSBool
- str_uneval(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- str = js_ValueToSource(cx, argc != 0 ? vp[2] : JSVAL_VOID);
- if (!str)
- return JS_FALSE;
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #endif
- const char js_escape_str[] = "escape";
- const char js_unescape_str[] = "unescape";
- #if JS_HAS_UNEVAL
- const char js_uneval_str[] = "uneval";
- #endif
- const char js_decodeURI_str[] = "decodeURI";
- const char js_encodeURI_str[] = "encodeURI";
- const char js_decodeURIComponent_str[] = "decodeURIComponent";
- const char js_encodeURIComponent_str[] = "encodeURIComponent";
- static JSFunctionSpec string_functions[] = {
- JS_FN(js_escape_str, str_escape, 1,0),
- JS_FN(js_unescape_str, str_unescape, 1,0),
- #if JS_HAS_UNEVAL
- JS_FN(js_uneval_str, str_uneval, 1,0),
- #endif
- JS_FN(js_decodeURI_str, str_decodeURI, 1,0),
- JS_FN(js_encodeURI_str, str_encodeURI, 1,0),
- JS_FN(js_decodeURIComponent_str, str_decodeURI_Component, 1,0),
- JS_FN(js_encodeURIComponent_str, str_encodeURI_Component, 1,0),
- JS_FS_END
- };
- jschar js_empty_ucstr[] = {0};
- JSSubString js_EmptySubString = {0, js_empty_ucstr};
- enum string_tinyid {
- STRING_LENGTH = -1
- };
- static JSPropertySpec string_props[] = {
- {js_length_str, STRING_LENGTH,
- JSPROP_READONLY|JSPROP_PERMANENT|JSPROP_SHARED, 0,0},
- {0,0,0,0,0}
- };
- static JSBool
- str_getProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
- {
- jsval v;
- JSString *str;
- jsint slot;
- if (!JSVAL_IS_INT(id))
- return JS_TRUE;
- slot = JSVAL_TO_INT(id);
- if (slot == STRING_LENGTH) {
- if (OBJ_GET_CLASS(cx, obj) == &js_StringClass) {
- /* Follow ECMA-262 by fetching intrinsic length of our string. */
- v = OBJ_GET_SLOT(cx, obj, JSSLOT_PRIVATE);
- JS_ASSERT(JSVAL_IS_STRING(v));
- str = JSVAL_TO_STRING(v);
- } else {
- /* Preserve compatibility: convert obj to a string primitive. */
- str = js_ValueToString(cx, OBJECT_TO_JSVAL(obj));
- if (!str)
- return JS_FALSE;
- }
- *vp = INT_TO_JSVAL((jsint) JSSTRING_LENGTH(str));
- }
- return JS_TRUE;
- }
- #define STRING_ELEMENT_ATTRS (JSPROP_ENUMERATE|JSPROP_READONLY|JSPROP_PERMANENT)
- static JSBool
- str_enumerate(JSContext *cx, JSObject *obj)
- {
- jsval v;
- JSString *str, *str1;
- size_t i, length;
- v = OBJ_GET_SLOT(cx, obj, JSSLOT_PRIVATE);
- JS_ASSERT(JSVAL_IS_STRING(v));
- str = JSVAL_TO_STRING(v);
- length = JSSTRING_LENGTH(str);
- for (i = 0; i < length; i++) {
- str1 = js_NewDependentString(cx, str, i, 1);
- if (!str1)
- return JS_FALSE;
- if (!OBJ_DEFINE_PROPERTY(cx, obj, INT_TO_JSID(i),
- STRING_TO_JSVAL(str1), NULL, NULL,
- STRING_ELEMENT_ATTRS, NULL)) {
- return JS_FALSE;
- }
- }
- return JS_TRUE;
- }
- static JSBool
- str_resolve(JSContext *cx, JSObject *obj, jsval id, uintN flags,
- JSObject **objp)
- {
- jsval v;
- JSString *str, *str1;
- jsint slot;
- if (!JSVAL_IS_INT(id) || (flags & JSRESOLVE_ASSIGNING))
- return JS_TRUE;
- v = OBJ_GET_SLOT(cx, obj, JSSLOT_PRIVATE);
- JS_ASSERT(JSVAL_IS_STRING(v));
- str = JSVAL_TO_STRING(v);
- slot = JSVAL_TO_INT(id);
- if ((size_t)slot < JSSTRING_LENGTH(str)) {
- str1 = js_GetUnitString(cx, str, (size_t)slot);
- if (!str1)
- return JS_FALSE;
- if (!OBJ_DEFINE_PROPERTY(cx, obj, INT_TO_JSID(slot),
- STRING_TO_JSVAL(str1), NULL, NULL,
- STRING_ELEMENT_ATTRS, NULL)) {
- return JS_FALSE;
- }
- *objp = obj;
- }
- return JS_TRUE;
- }
- JSClass js_StringClass = {
- js_String_str,
- JSCLASS_HAS_PRIVATE | JSCLASS_NEW_RESOLVE |
- JSCLASS_HAS_CACHED_PROTO(JSProto_String),
- JS_PropertyStub, JS_PropertyStub, str_getProperty, JS_PropertyStub,
- str_enumerate, (JSResolveOp)str_resolve, JS_ConvertStub, JS_FinalizeStub,
- JSCLASS_NO_OPTIONAL_MEMBERS
- };
- #define NORMALIZE_THIS(cx,vp,str) \
- JS_BEGIN_MACRO \
- if (JSVAL_IS_STRING(vp[1])) { \
- str = JSVAL_TO_STRING(vp[1]); \
- } else { \
- str = NormalizeThis(cx, vp); \
- if (!str) \
- return JS_FALSE; \
- } \
- JS_END_MACRO
- static JSString *
- NormalizeThis(JSContext *cx, jsval *vp)
- {
- JSString *str;
- if (JSVAL_IS_NULL(vp[1]) && JSVAL_IS_NULL(JS_THIS(cx, vp)))
- return NULL;
- str = js_ValueToString(cx, vp[1]);
- if (!str)
- return NULL;
- vp[1] = STRING_TO_JSVAL(str);
- return str;
- }
- #if JS_HAS_TOSOURCE
- /*
- * String.prototype.quote is generic (as are most string methods), unlike
- * toSource, toString, and valueOf.
- */
- static JSBool
- str_quote(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- NORMALIZE_THIS(cx, vp, str);
- str = js_QuoteString(cx, str, '"');
- if (!str)
- return JS_FALSE;
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- static JSBool
- str_toSource(JSContext *cx, uintN argc, jsval *vp)
- {
- jsval v;
- JSString *str;
- size_t i, j, k, n;
- char buf[16];
- jschar *s, *t;
- if (!js_GetPrimitiveThis(cx, vp, &js_StringClass, &v))
- return JS_FALSE;
- JS_ASSERT(JSVAL_IS_STRING(v));
- str = js_QuoteString(cx, JSVAL_TO_STRING(v), '"');
- if (!str)
- return JS_FALSE;
- j = JS_snprintf(buf, sizeof buf, "(new %s(", js_StringClass.name);
- JSSTRING_CHARS_AND_LENGTH(str, s, k);
- n = j + k + 2;
- t = (jschar *) JS_malloc(cx, (n + 1) * sizeof(jschar));
- if (!t)
- return JS_FALSE;
- for (i = 0; i < j; i++)
- t[i] = buf[i];
- for (j = 0; j < k; i++, j++)
- t[i] = s[j];
- t[i++] = ')';
- t[i++] = ')';
- t[i] = 0;
- str = js_NewString(cx, t, n);
- if (!str) {
- JS_free(cx, t);
- return JS_FALSE;
- }
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #endif /* JS_HAS_TOSOURCE */
- static JSBool
- str_toString(JSContext *cx, uintN argc, jsval *vp)
- {
- return js_GetPrimitiveThis(cx, vp, &js_StringClass, vp);
- }
- /*
- * Java-like string native methods.
- */
- static JSString *
- SubstringTail(JSContext *cx, JSString *str, jsdouble length, jsdouble begin, jsdouble end)
- {
- if (begin < 0)
- begin = 0;
- else if (begin > length)
- begin = length;
- if (end < 0)
- end = 0;
- else if (end > length)
- end = length;
- if (end < begin) {
- /* ECMA emulates old JDK1.0 java.lang.String.substring. */
- jsdouble tmp = begin;
- begin = end;
- end = tmp;
- }
- return js_NewDependentString(cx, str, (size_t)begin, (size_t)(end - begin));
- }
- static JSBool
- str_substring(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- jsdouble d;
- jsdouble length, begin, end;
- NORMALIZE_THIS(cx, vp, str);
- if (argc != 0) {
- d = js_ValueToNumber(cx, &vp[2]);
- if (JSVAL_IS_NULL(vp[2]))
- return JS_FALSE;
- length = JSSTRING_LENGTH(str);
- begin = js_DoubleToInteger(d);
- if (argc == 1) {
- end = length;
- } else {
- d = js_ValueToNumber(cx, &vp[3]);
- if (JSVAL_IS_NULL(vp[3]))
- return JS_FALSE;
- end = js_DoubleToInteger(d);
- }
- str = SubstringTail(cx, str, length, begin, end);
- if (!str)
- return JS_FALSE;
- }
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #ifdef JS_TRACER
- static JSString* FASTCALL
- String_p_toString(JSContext* cx, JSObject* obj)
- {
- if (!JS_InstanceOf(cx, obj, &js_StringClass, NULL))
- return NULL;
- jsval v = OBJ_GET_SLOT(cx, obj, JSSLOT_PRIVATE);
- JS_ASSERT(JSVAL_IS_STRING(v));
- return JSVAL_TO_STRING(v);
- }
- static JSString* FASTCALL
- String_p_substring(JSContext* cx, JSString* str, int32 begin, int32 end)
- {
- JS_ASSERT(JS_ON_TRACE(cx));
- size_t length = JSSTRING_LENGTH(str);
- return SubstringTail(cx, str, length, begin, end);
- }
- static JSString* FASTCALL
- String_p_substring_1(JSContext* cx, JSString* str, int32 begin)
- {
- JS_ASSERT(JS_ON_TRACE(cx));
- size_t length = JSSTRING_LENGTH(str);
- return SubstringTail(cx, str, length, begin, length);
- }
- #endif
- JSString* JS_FASTCALL
- js_toLowerCase(JSContext *cx, JSString *str)
- {
- size_t i, n;
- jschar *s, *news;
- JSSTRING_CHARS_AND_LENGTH(str, s, n);
- news = (jschar *) JS_malloc(cx, (n + 1) * sizeof(jschar));
- if (!news)
- return NULL;
- for (i = 0; i < n; i++)
- news[i] = JS_TOLOWER(s[i]);
- news[n] = 0;
- str = js_NewString(cx, news, n);
- if (!str) {
- JS_free(cx, news);
- return NULL;
- }
- return str;
- }
- static JSBool
- str_toLowerCase(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- NORMALIZE_THIS(cx, vp, str);
- str = js_toLowerCase(cx, str);
- if (!str)
- return JS_FALSE;
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- static JSBool
- str_toLocaleLowerCase(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- /*
- * Forcefully ignore the first (or any) argument and return toLowerCase(),
- * ECMA has reserved that argument, presumably for defining the locale.
- */
- if (cx->localeCallbacks && cx->localeCallbacks->localeToLowerCase) {
- NORMALIZE_THIS(cx, vp, str);
- return cx->localeCallbacks->localeToLowerCase(cx, str, vp);
- }
- return str_toLowerCase(cx, 0, vp);
- }
- JSString* JS_FASTCALL
- js_toUpperCase(JSContext *cx, JSString *str)
- {
- size_t i, n;
- jschar *s, *news;
- JSSTRING_CHARS_AND_LENGTH(str, s, n);
- news = (jschar *) JS_malloc(cx, (n + 1) * sizeof(jschar));
- if (!news)
- return NULL;
- for (i = 0; i < n; i++)
- news[i] = JS_TOUPPER(s[i]);
- news[n] = 0;
- str = js_NewString(cx, news, n);
- if (!str) {
- JS_free(cx, news);
- return NULL;
- }
- return str;
- }
- static JSBool
- str_toUpperCase(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- NORMALIZE_THIS(cx, vp, str);
- str = js_toUpperCase(cx, str);
- if (!str)
- return JS_FALSE;
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- static JSBool
- str_toLocaleUpperCase(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- /*
- * Forcefully ignore the first (or any) argument and return toUpperCase(),
- * ECMA has reserved that argument, presumably for defining the locale.
- */
- if (cx->localeCallbacks && cx->localeCallbacks->localeToUpperCase) {
- NORMALIZE_THIS(cx, vp, str);
- return cx->localeCallbacks->localeToUpperCase(cx, str, vp);
- }
- return str_toUpperCase(cx, 0, vp);
- }
- static JSBool
- str_localeCompare(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str, *thatStr;
- NORMALIZE_THIS(cx, vp, str);
- if (argc == 0) {
- *vp = JSVAL_ZERO;
- } else {
- thatStr = js_ValueToString(cx, vp[2]);
- if (!thatStr)
- return JS_FALSE;
- if (cx->localeCallbacks && cx->localeCallbacks->localeCompare) {
- vp[2] = STRING_TO_JSVAL(thatStr);
- return cx->localeCallbacks->localeCompare(cx, str, thatStr, vp);
- }
- *vp = INT_TO_JSVAL(js_CompareStrings(str, thatStr));
- }
- return JS_TRUE;
- }
- static JSBool
- str_charAt(JSContext *cx, uintN argc, jsval *vp)
- {
- jsval t;
- JSString *str;
- jsint i;
- jsdouble d;
- t = vp[1];
- if (JSVAL_IS_STRING(t) && argc != 0 && JSVAL_IS_INT(vp[2])) {
- str = JSVAL_TO_STRING(t);
- i = JSVAL_TO_INT(vp[2]);
- if ((size_t)i >= JSSTRING_LENGTH(str))
- goto out_of_range;
- } else {
- str = NormalizeThis(cx, vp);
- if (!str)
- return JS_FALSE;
- if (argc == 0) {
- d = 0.0;
- } else {
- d = js_ValueToNumber(cx, &vp[2]);
- if (JSVAL_IS_NULL(vp[2]))
- return JS_FALSE;
- d = js_DoubleToInteger(d);
- }
- if (d < 0 || JSSTRING_LENGTH(str) <= d)
- goto out_of_range;
- i = (jsint) d;
- }
- str = js_GetUnitString(cx, str, (size_t)i);
- if (!str)
- return JS_FALSE;
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- out_of_range:
- *vp = JS_GetEmptyStringValue(cx);
- return JS_TRUE;
- }
- static JSBool
- str_charCodeAt(JSContext *cx, uintN argc, jsval *vp)
- {
- jsval t;
- JSString *str;
- jsint i;
- jsdouble d;
- t = vp[1];
- if (JSVAL_IS_STRING(t) && argc != 0 && JSVAL_IS_INT(vp[2])) {
- str = JSVAL_TO_STRING(t);
- i = JSVAL_TO_INT(vp[2]);
- if ((size_t)i >= JSSTRING_LENGTH(str))
- goto out_of_range;
- } else {
- str = NormalizeThis(cx, vp);
- if (!str)
- return JS_FALSE;
- if (argc == 0) {
- d = 0.0;
- } else {
- d = js_ValueToNumber(cx, &vp[2]);
- if (JSVAL_IS_NULL(vp[2]))
- return JS_FALSE;
- d = js_DoubleToInteger(d);
- }
- if (d < 0 || JSSTRING_LENGTH(str) <= d)
- goto out_of_range;
- i = (jsint) d;
- }
- *vp = INT_TO_JSVAL(JSSTRING_CHARS(str)[i]);
- return JS_TRUE;
- out_of_range:
- *vp = JS_GetNaNValue(cx);
- return JS_TRUE;
- }
- #ifdef JS_TRACER
- int32 FASTCALL
- js_String_p_charCodeAt(JSString* str, int32 i)
- {
- if (i < 0 || (int32)JSSTRING_LENGTH(str) <= i)
- return -1;
- return JSSTRING_CHARS(str)[i];
- }
- #endif
- jsint
- js_BoyerMooreHorspool(const jschar *text, jsint textlen,
- const jschar *pat, jsint patlen,
- jsint start)
- {
- jsint i, j, k, m;
- uint8 skip[BMH_CHARSET_SIZE];
- jschar c;
- JS_ASSERT(0 < patlen && patlen <= BMH_PATLEN_MAX);
- for (i = 0; i < BMH_CHARSET_SIZE; i++)
- skip[i] = (uint8)patlen;
- m = patlen - 1;
- for (i = 0; i < m; i++) {
- c = pat[i];
- if (c >= BMH_CHARSET_SIZE)
- return BMH_BAD_PATTERN;
- skip[c] = (uint8)(m - i);
- }
- for (k = start + m;
- k < textlen;
- k += ((c = text[k]) >= BMH_CHARSET_SIZE) ? patlen : skip[c]) {
- for (i = k, j = m; ; i--, j--) {
- if (j < 0)
- return i + 1;
- if (text[i] != pat[j])
- break;
- }
- }
- return -1;
- }
- static JSBool
- str_indexOf(JSContext *cx, uintN argc, jsval *vp)
- {
- jsval t;
- JSString *str, *str2;
- const jschar *text, *pat;
- jsint i, j, index, textlen, patlen;
- jsdouble d;
- t = vp[1];
- if (JSVAL_IS_STRING(t) && argc != 0 && JSVAL_IS_STRING(vp[2])) {
- str = JSVAL_TO_STRING(t);
- str2 = JSVAL_TO_STRING(vp[2]);
- } else {
- str = NormalizeThis(cx, vp);
- if (!str)
- return JS_FALSE;
- str2 = ArgToRootedString(cx, argc, vp, 0);
- if (!str2)
- return JS_FALSE;
- }
- text = JSSTRING_CHARS(str);
- textlen = (jsint) JSSTRING_LENGTH(str);
- pat = JSSTRING_CHARS(str2);
- patlen = (jsint) JSSTRING_LENGTH(str2);
- if (argc > 1) {
- d = js_ValueToNumber(cx, &vp[3]);
- if (JSVAL_IS_NULL(vp[3]))
- return JS_FALSE;
- d = js_DoubleToInteger(d);
- if (d < 0)
- i = 0;
- else if (d > textlen)
- i = textlen;
- else
- i = (jsint)d;
- } else {
- i = 0;
- }
- if (patlen == 0) {
- *vp = INT_TO_JSVAL(i);
- return JS_TRUE;
- }
- /* XXX tune the BMH threshold (512) */
- if (textlen - i >= 512 && (jsuint)(patlen - 2) <= BMH_PATLEN_MAX - 2) {
- index = js_BoyerMooreHorspool(text, textlen, pat, patlen, i);
- if (index != BMH_BAD_PATTERN)
- goto out;
- }
- index = -1;
- j = 0;
- while (i + j < textlen) {
- if (text[i + j] == pat[j]) {
- if (++j == patlen) {
- index = i;
- break;
- }
- } else {
- i++;
- j = 0;
- }
- }
- out:
- *vp = INT_TO_JSVAL(index);
- return JS_TRUE;
- }
- static JSBool
- str_lastIndexOf(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str, *str2;
- const jschar *text, *pat;
- jsint i, j, textlen, patlen;
- jsdouble d;
- NORMALIZE_THIS(cx, vp, str);
- text = JSSTRING_CHARS(str);
- textlen = (jsint) JSSTRING_LENGTH(str);
- str2 = ArgToRootedString(cx, argc, vp, 0);
- if (!str2)
- return JS_FALSE;
- pat = JSSTRING_CHARS(str2);
- patlen = (jsint) JSSTRING_LENGTH(str2);
- if (argc > 1) {
- d = js_ValueToNumber(cx, &vp[3]);
- if (JSVAL_IS_NULL(vp[3]))
- return JS_FALSE;
- if (JSDOUBLE_IS_NaN(d)) {
- i = textlen;
- } else {
- d = js_DoubleToInteger(d);
- if (d < 0)
- i = 0;
- else if (d > textlen)
- i = textlen;
- else
- i = (jsint)d;
- }
- } else {
- i = textlen;
- }
- if (patlen == 0) {
- *vp = INT_TO_JSVAL(i);
- return JS_TRUE;
- }
- j = 0;
- while (i >= 0) {
- /* Don't assume that text is NUL-terminated: it could be dependent. */
- if (i + j < textlen && text[i + j] == pat[j]) {
- if (++j == patlen)
- break;
- } else {
- i--;
- j = 0;
- }
- }
- *vp = INT_TO_JSVAL(i);
- return JS_TRUE;
- }
- static JSBool
- js_TrimString(JSContext *cx, jsval *vp, JSBool trimLeft, JSBool trimRight)
- {
- JSString *str;
- const jschar *chars;
- size_t length, begin, end;
- NORMALIZE_THIS(cx, vp, str);
- JSSTRING_CHARS_AND_LENGTH(str, chars, length);
- begin = 0;
- end = length;
- if (trimLeft) {
- while (begin < length && JS_ISSPACE(chars[begin]))
- ++begin;
- }
- if (trimRight) {
- while (end > begin && JS_ISSPACE(chars[end-1]))
- --end;
- }
- str = js_NewDependentString(cx, str, begin, end - begin);
- if (!str)
- return JS_FALSE;
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- static JSBool
- str_trim(JSContext *cx, uintN argc, jsval *vp)
- {
- return js_TrimString(cx, vp, JS_TRUE, JS_TRUE);
- }
- static JSBool
- str_trimLeft(JSContext *cx, uintN argc, jsval *vp)
- {
- return js_TrimString(cx, vp, JS_TRUE, JS_FALSE);
- }
- static JSBool
- str_trimRight(JSContext *cx, uintN argc, jsval *vp)
- {
- return js_TrimString(cx, vp, JS_FALSE, JS_TRUE);
- }
- /*
- * Perl-inspired string functions.
- */
- typedef struct GlobData {
- jsbytecode *pc; /* in: program counter resulting in us matching */
- uintN flags; /* inout: mode and flag bits, see below */
- uintN optarg; /* in: index of optional flags argument */
- JSString *str; /* out: 'this' parameter object as string */
- JSRegExp *regexp; /* out: regexp parameter object private data */
- } GlobData;
- /*
- * Mode and flag bit definitions for match_or_replace's GlobData.flags field.
- */
- #define MODE_MATCH 0x00 /* in: return match array on success */
- #define MODE_REPLACE 0x01 /* in: match and replace */
- #define MODE_SEARCH 0x02 /* in: search only, return match index or -1 */
- #define GET_MODE(f) ((f) & 0x03)
- #define FORCE_FLAT 0x04 /* in: force flat (non-regexp) string match */
- #define KEEP_REGEXP 0x08 /* inout: keep GlobData.regexp alive for caller
- of match_or_replace; if set on input
- but clear on output, regexp ownership
- does not pass to caller */
- #define GLOBAL_REGEXP 0x10 /* out: regexp had the 'g' flag */
- static JSBool
- match_or_replace(JSContext *cx,
- JSBool (*glob)(JSContext *cx, jsint count, GlobData *data),
- void (*destroy)(JSContext *cx, GlobData *data),
- GlobData *data, uintN argc, jsval *vp)
- {
- JSString *str, *src, *opt;
- JSObject *reobj;
- JSRegExp *re;
- size_t index, length;
- JSBool ok, test;
- jsint count;
- NORMALIZE_THIS(cx, vp, str);
- data->str = str;
- if (argc != 0 && VALUE_IS_REGEXP(cx, vp[2])) {
- reobj = JSVAL_TO_OBJECT(vp[2]);
- re = (JSRegExp *) JS_GetPrivate(cx, reobj);
- } else {
- src = ArgToRootedString(cx, argc, vp, 0);
- if (!src)
- return JS_FALSE;
- if (data->optarg < argc) {
- opt = js_ValueToString(cx, vp[2 + data->optarg]);
- if (!opt)
- return JS_FALSE;
- } else {
- opt = NULL;
- }
- re = js_NewRegExpOpt(cx, src, opt, (data->flags & FORCE_FLAT) != 0);
- if (!re)
- return JS_FALSE;
- reobj = NULL;
- }
- /* From here on, all control flow must reach the matching DROP. */
- data->regexp = re;
- HOLD_REGEXP(cx, re);
- if (re->flags & JSREG_GLOB)
- data->flags |= GLOBAL_REGEXP;
- index = 0;
- if (GET_MODE(data->flags) == MODE_SEARCH) {
- ok = js_ExecuteRegExp(cx, re, str, &index, JS_TRUE, vp);
- if (ok) {
- *vp = (*vp == JSVAL_TRUE)
- ? INT_TO_JSVAL(cx->regExpStatics.leftContext.length)
- : INT_TO_JSVAL(-1);
- }
- } else if (data->flags & GLOBAL_REGEXP) {
- if (reobj) {
- /* Set the lastIndex property's reserved slot to 0. */
- ok = js_SetLastIndex(cx, reobj, 0);
- } else {
- ok = JS_TRUE;
- }
- if (ok) {
- length = JSSTRING_LENGTH(str);
- for (count = 0; index <= length; count++) {
- ok = js_ExecuteRegExp(cx, re, str, &index, JS_TRUE, vp);
- if (!ok || *vp != JSVAL_TRUE)
- break;
- ok = glob(cx, count, data);
- if (!ok)
- break;
- if (cx->regExpStatics.lastMatch.length == 0) {
- if (index == length)
- break;
- index++;
- }
- }
- if (!ok && destroy)
- destroy(cx, data);
- }
- } else {
- if (GET_MODE(data->flags) == MODE_REPLACE) {
- test = JS_TRUE;
- } else {
- /*
- * MODE_MATCH implies str_match is being called from a script or a
- * scripted function. If the caller cares only about testing null
- * vs. non-null return value, optimize away the array object that
- * would normally be returned in *vp.
- *
- * Assume a full array result is required, then prove otherwise.
- */
- test = JS_FALSE;
- if (data->pc && (*data->pc == JSOP_CALL || *data->pc == JSOP_NEW)) {
- JS_ASSERT(js_CodeSpec[*data->pc].length == 3);
- switch (data->pc[3]) {
- case JSOP_POP:
- case JSOP_IFEQ:
- case JSOP_IFNE:
- case JSOP_IFEQX:
- case JSOP_IFNEX:
- test = JS_TRUE;
- break;
- default:;
- }
- }
- }
- ok = js_ExecuteRegExp(cx, re, str, &index, test, vp);
- }
- DROP_REGEXP(cx, re);
- if (reobj) {
- /* Tell our caller that it doesn't need to destroy data->regexp. */
- data->flags &= ~KEEP_REGEXP;
- } else if (!ok || !(data->flags & KEEP_REGEXP)) {
- /* Caller didn't want to keep data->regexp, so null and destroy it. */
- data->regexp = NULL;
- js_DestroyRegExp(cx, re);
- }
- return ok;
- }
- typedef struct MatchData {
- GlobData base;
- jsval *arrayval; /* NB: local root pointer */
- } MatchData;
- static JSBool
- match_glob(JSContext *cx, jsint count, GlobData *data)
- {
- MatchData *mdata;
- JSObject *arrayobj;
- JSSubString *matchsub;
- JSString *matchstr;
- jsval v;
- mdata = (MatchData *)data;
- arrayobj = JSVAL_TO_OBJECT(*mdata->arrayval);
- if (!arrayobj) {
- arrayobj = js_ConstructObject(cx, &js_ArrayClass, NULL, NULL, 0, NULL);
- if (!arrayobj)
- return JS_FALSE;
- *mdata->arrayval = OBJECT_TO_JSVAL(arrayobj);
- }
- matchsub = &cx->regExpStatics.lastMatch;
- matchstr = js_NewStringCopyN(cx, matchsub->chars, matchsub->length);
- if (!matchstr)
- return JS_FALSE;
- v = STRING_TO_JSVAL(matchstr);
- JS_ASSERT(count <= JSVAL_INT_MAX);
- JSAutoResolveFlags rf(cx, JSRESOLVE_QUALIFIED | JSRESOLVE_ASSIGNING);
- return OBJ_SET_PROPERTY(cx, arrayobj, INT_TO_JSID(count), &v);
- }
- JSBool
- js_StringMatchHelper(JSContext *cx, uintN argc, jsval *vp, jsbytecode *pc)
- {
- JSTempValueRooter tvr;
- MatchData mdata;
- JSBool ok;
- JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
- mdata.base.pc = pc;
- mdata.base.flags = MODE_MATCH;
- mdata.base.optarg = 1;
- mdata.arrayval = &tvr.u.value;
- ok = match_or_replace(cx, match_glob, NULL, &mdata.base, argc, vp);
- if (ok && !JSVAL_IS_NULL(*mdata.arrayval))
- *vp = *mdata.arrayval;
- JS_POP_TEMP_ROOT(cx, &tvr);
- return ok;
- }
- static JSBool
- str_match(JSContext *cx, uintN argc, jsval *vp)
- {
- JSStackFrame *fp;
- for (fp = cx->fp; fp && !fp->regs; fp = fp->down)
- JS_ASSERT(!fp->script);
- return js_StringMatchHelper(cx, argc, vp, fp ? fp->regs->pc : NULL);
- }
- #ifdef JS_TRACER
- static JSObject* FASTCALL
- String_p_match(JSContext* cx, JSString* str, jsbytecode *pc, JSObject* regexp)
- {
- jsval vp[3] = { JSVAL_NULL, STRING_TO_JSVAL(str), OBJECT_TO_JSVAL(regexp) };
- if (!js_StringMatchHelper(cx, 1, vp, pc))
- return (JSObject*) JSVAL_TO_BOOLEAN(JSVAL_VOID);
- JS_ASSERT(JSVAL_IS_NULL(vp[0]) ||
- (!JSVAL_IS_PRIMITIVE(vp[0]) && OBJ_IS_ARRAY(cx, JSVAL_TO_OBJECT(vp[0]))));
- return JSVAL_TO_OBJECT(vp[0]);
- }
- static JSObject* FASTCALL
- String_p_match_obj(JSContext* cx, JSObject* str, jsbytecode *pc, JSObject* regexp)
- {
- jsval vp[3] = { JSVAL_NULL, OBJECT_TO_JSVAL(str), OBJECT_TO_JSVAL(regexp) };
- if (!js_StringMatchHelper(cx, 1, vp, pc))
- return (JSObject*) JSVAL_TO_BOOLEAN(JSVAL_VOID);
- JS_ASSERT(JSVAL_IS_NULL(vp[0]) ||
- (!JSVAL_IS_PRIMITIVE(vp[0]) && OBJ_IS_ARRAY(cx, JSVAL_TO_OBJECT(vp[0]))));
- return JSVAL_TO_OBJECT(vp[0]);
- }
- #endif
- static JSBool
- str_search(JSContext *cx, uintN argc, jsval *vp)
- {
- GlobData data;
- data.flags = MODE_SEARCH;
- data.optarg = 1;
- return match_or_replace(cx, NULL, NULL, &data, argc, vp);
- }
- typedef struct ReplaceData {
- GlobData base; /* base struct state */
- JSObject *lambda; /* replacement function object or null */
- JSString *repstr; /* replacement string */
- jschar *dollar; /* null or pointer to first $ in repstr */
- jschar *dollarEnd; /* limit pointer for js_strchr_limit */
- jschar *chars; /* result chars, null initially */
- size_t length; /* result length, 0 initially */
- jsint index; /* index in result of next replacement */
- jsint leftIndex; /* left context index in base.str->chars */
- JSSubString dollarStr; /* for "$$" interpret_dollar result */
- } ReplaceData;
- static JSSubString *
- interpret_dollar(JSContext *cx, jschar *dp, jschar *ep, ReplaceData *rdata,
- size_t *skip)
- {
- JSRegExpStatics *res;
- jschar dc, *cp;
- uintN num, tmp;
- JS_ASSERT(*dp == '$');
- /* If there is only a dollar, bail now */
- if (dp + 1 >= ep)
- return NULL;
- /* Interpret all Perl match-induced dollar variables. */
- res = &cx->regExpStatics;
- dc = dp[1];
- if (JS7_ISDEC(dc)) {
- /* ECMA-262 Edition 3: 1-9 or 01-99 */
- num = JS7_UNDEC(dc);
- if (num > res->parenCount)
- return NULL;
- cp = dp + 2;
- if (cp < ep && (dc = *cp, JS7_ISDEC(dc))) {
- tmp = 10 * num + JS7_UNDEC(dc);
- if (tmp <= res->parenCount) {
- cp++;
- num = tmp;
- }
- }
- if (num == 0)
- return NULL;
- /* Adjust num from 1 $n-origin to 0 array-index-origin. */
- num--;
- *skip = cp - dp;
- return REGEXP_PAREN_SUBSTRING(res, num);
- }
- *skip = 2;
- switch (dc) {
- case '$':
- rdata->dollarStr.chars = dp;
- rdata->dollarStr.length = 1;
- return &rdata->dollarStr;
- case '&':
- return &res->lastMatch;
- case '+':
- return &res->lastParen;
- case '`':
- return &res->leftContext;
- case '\'':
- return &res->rightContext;
- }
- return NULL;
- }
- static JSBool
- find_replen(JSContext *cx, ReplaceData *rdata, size_t *sizep)
- {
- JSString *repstr;
- size_t replen, skip;
- jschar *dp, *ep;
- JSSubString *sub;
- JSObject *lambda;
- lambda = rdata->lambda;
- if (lambda) {
- uintN argc, i, j, m, n, p;
- jsval *invokevp, *sp;
- void *mark;
- JSBool ok;
- /*
- * Save the regExpStatics from the current regexp, since they may be
- * clobbered by a RegExp usage in the lambda function. Note that all
- * members of JSRegExpStatics are JSSubStrings, so not GC roots, save
- * input, which is rooted otherwise via vp[1] in str_replace.
- */
- JSRegExpStatics save = cx->regExpStatics;
- JSBool freeMoreParens = JS_FALSE;
- /*
- * In the lambda case, not only do we find the replacement string's
- * length, we compute repstr and return it via rdata for use within
- * do_replace. The lambda is called with arguments ($&, $1, $2, ...,
- * index, input), i.e., all the properties of a regexp match array.
- * For $&, etc., we must create string jsvals from cx->regExpStatics.
- * We grab up stack space to keep the newborn strings GC-rooted.
- */
- p = rdata->base.regexp->parenCount;
- argc = 1 + p + 2;
- invokevp = js_AllocStack(cx, 2 + argc, &mark);
- if (!invokevp)
- return JS_FALSE;
- /* Push lambda and its 'this' parameter. */
- sp = invokevp;
- *sp++ = OBJECT_TO_JSVAL(lambda);
- *sp++ = OBJECT_TO_JSVAL(OBJ_GET_PARENT(cx, lambda));
- #define PUSH_REGEXP_STATIC(sub) \
- JS_BEGIN_MACRO \
- JSString *str = js_NewStringCopyN(cx, \
- cx->regExpStatics.sub.chars, \
- cx->regExpStatics.sub.length); \
- if (!str) { \
- ok = JS_FALSE; \
- goto lambda_out; \
- } \
- *sp++ = STRING_TO_JSVAL(str); \
- JS_END_MACRO
- /* Push $&, $1, $2, ... */
- PUSH_REGEXP_STATIC(lastMatch);
- i = 0;
- m = cx->regExpStatics.parenCount;
- n = JS_MIN(m, 9);
- for (j = 0; i < n; i++, j++)
- PUSH_REGEXP_STATIC(parens[j]);
- for (j = 0; i < m; i++, j++)
- PUSH_REGEXP_STATIC(moreParens[j]);
- /*
- * We need to clear moreParens in the top-of-stack cx->regExpStatics
- * to it won't be possibly realloc'ed, leaving the bottom-of-stack
- * moreParens pointing to freed memory.
- */
- cx->regExpStatics.moreParens = NULL;
- freeMoreParens = JS_TRUE;
- #undef PUSH_REGEXP_STATIC
- /* Make sure to push undefined for any unmatched parens. */
- for (; i < p; i++)
- *sp++ = JSVAL_VOID;
- /* Push match index and input string. */
- *sp++ = INT_TO_JSVAL((jsint)cx->regExpStatics.leftContext.length);
- *sp++ = STRING_TO_JSVAL(rdata->base.str);
- ok = js_Invoke(cx, argc, invokevp, 0);
- if (ok) {
- /*
- * NB: we count on the newborn string root to hold any string
- * created by this js_ValueToString that would otherwise be GC-
- * able, until we use rdata->repstr in do_replace.
- */
- repstr = js_ValueToString(cx, *invokevp);
- if (!repstr) {
- ok = JS_FALSE;
- } else {
- rdata->repstr = repstr;
- *sizep = JSSTRING_LENGTH(repstr);
- }
- }
- lambda_out:
- js_FreeStack(cx, mark);
- if (freeMoreParens)
- JS_free(cx, cx->regExpStatics.moreParens);
- cx->regExpStatics = save;
- return ok;
- }
- repstr = rdata->repstr;
- replen = JSSTRING_LENGTH(repstr);
- for (dp = rdata->dollar, ep = rdata->dollarEnd; dp;
- dp = js_strchr_limit(dp, '$', ep)) {
- sub = interpret_dollar(cx, dp, ep, rdata, &skip);
- if (sub) {
- replen += sub->length - skip;
- dp += skip;
- }
- else
- dp++;
- }
- *sizep = replen;
- return JS_TRUE;
- }
- static void
- do_replace(JSContext *cx, ReplaceData *rdata, jschar *chars)
- {
- JSString *repstr;
- jschar *bp, *cp, *dp, *ep;
- size_t len, skip;
- JSSubString *sub;
- repstr = rdata->repstr;
- bp = cp = JSSTRING_CHARS(repstr);
- for (dp = rdata->dollar, ep = rdata->dollarEnd; dp;
- dp = js_strchr_limit(dp, '$', ep)) {
- len = dp - cp;
- js_strncpy(chars, cp, len);
- chars += len;
- cp = dp;
- sub = interpret_dollar(cx, dp, ep, rdata, &skip);
- if (sub) {
- len = sub->length;
- js_strncpy(chars, sub->chars, len);
- chars += len;
- cp += skip;
- dp += skip;
- } else {
- dp++;
- }
- }
- js_strncpy(chars, cp, JSSTRING_LENGTH(repstr) - (cp - bp));
- }
- static void
- replace_destroy(JSContext *cx, GlobData *data)
- {
- ReplaceData *rdata;
- rdata = (ReplaceData *)data;
- JS_free(cx, rdata->chars);
- rdata->chars = NULL;
- }
- static JSBool
- replace_glob(JSContext *cx, jsint count, GlobData *data)
- {
- ReplaceData *rdata;
- JSString *str;
- size_t leftoff, leftlen, replen, growth;
- const jschar *left;
- jschar *chars;
- rdata = (ReplaceData *)data;
- str = data->str;
- leftoff = rdata->leftIndex;
- left = JSSTRING_CHARS(str) + leftoff;
- leftlen = cx->regExpStatics.lastMatch.chars - left;
- rdata->leftIndex = cx->regExpStatics.lastMatch.chars - JSSTRING_CHARS(str);
- rdata->leftIndex += cx->regExpStatics.lastMatch.length;
- if (!find_replen(cx, rdata, &replen))
- return JS_FALSE;
- growth = leftlen + replen;
- chars = (jschar *)
- (rdata->chars
- ? JS_realloc(cx, rdata->chars, (rdata->length + growth + 1)
- * sizeof(jschar))
- : JS_malloc(cx, (growth + 1) * sizeof(jschar)));
- if (!chars)
- return JS_FALSE;
- rdata->chars = chars;
- rdata->length += growth;
- chars += rdata->index;
- rdata->index += growth;
- js_strncpy(chars, left, leftlen);
- chars += leftlen;
- do_replace(cx, rdata, chars);
- return JS_TRUE;
- }
- static JSBool
- str_replace(JSContext *cx, uintN argc, jsval *vp)
- {
- JSObject *lambda;
- JSString *repstr;
- if (argc >= 2 && JS_TypeOfValue(cx, vp[3]) == JSTYPE_FUNCTION) {
- lambda = JSVAL_TO_OBJECT(vp[3]);
- repstr = NULL;
- } else {
- lambda = NULL;
- repstr = ArgToRootedString(cx, argc, vp, 1);
- if (!repstr)
- return JS_FALSE;
- }
- return js_StringReplaceHelper(cx, argc, lambda, repstr, vp);
- }
- #ifdef JS_TRACER
- static JSString* FASTCALL
- String_p_replace_str(JSContext* cx, JSString* str, JSObject* regexp, JSString* repstr)
- {
- jsval vp[4] = {
- JSVAL_NULL, STRING_TO_JSVAL(str), OBJECT_TO_JSVAL(regexp), STRING_TO_JSVAL(repstr)
- };
- if (!js_StringReplaceHelper(cx, 2, NULL, repstr, vp))
- return NULL;
- JS_ASSERT(JSVAL_IS_STRING(vp[0]));
- return JSVAL_TO_STRING(vp[0]);
- }
- static JSString* FASTCALL
- String_p_replace_str2(JSContext* cx, JSString* str, JSString* patstr, JSString* repstr)
- {
- jsval vp[4] = {
- JSVAL_NULL, STRING_TO_JSVAL(str), STRING_TO_JSVAL(patstr), STRING_TO_JSVAL(repstr)
- };
- if (!js_StringReplaceHelper(cx, 2, NULL, repstr, vp))
- return NULL;
- JS_ASSERT(JSVAL_IS_STRING(vp[0]));
- return JSVAL_TO_STRING(vp[0]);
- }
- static JSString* FASTCALL
- String_p_replace_str3(JSContext* cx, JSString* str, JSString* patstr, JSString* repstr,
- JSString* flagstr)
- {
- jsval vp[5] = {
- JSVAL_NULL, STRING_TO_JSVAL(str), STRING_TO_JSVAL(patstr), STRING_TO_JSVAL(repstr),
- STRING_TO_JSVAL(flagstr)
- };
- if (!js_StringReplaceHelper(cx, 3, NULL, repstr, vp))
- return NULL;
- JS_ASSERT(JSVAL_IS_STRING(vp[0]));
- return JSVAL_TO_STRING(vp[0]);
- }
- #endif
- JSBool
- js_StringReplaceHelper(JSContext *cx, uintN argc, JSObject *lambda,
- JSString *repstr, jsval *vp)
- {
- ReplaceData rdata;
- JSBool ok;
- size_t leftlen, rightlen, length;
- jschar *chars;
- JSString *str;
- /*
- * For ECMA Edition 3, the first argument is to be converted to a string
- * to match in a "flat" sense (without regular expression metachars having
- * special meanings) UNLESS the first arg is a RegExp object.
- */
- rdata.base.flags = MODE_REPLACE | KEEP_REGEXP | FORCE_FLAT;
- rdata.base.optarg = 2;
- rdata.lambda = lambda;
- rdata.repstr = repstr;
- if (repstr) {
- rdata.dollarEnd = JSSTRING_CHARS(repstr) + JSSTRING_LENGTH(repstr);
- rdata.dollar = js_strchr_limit(JSSTRING_CHARS(repstr), '$',
- rdata.dollarEnd);
- } else {
- rdata.dollar = rdata.dollarEnd = NULL;
- }
- rdata.chars = NULL;
- rdata.length = 0;
- rdata.index = 0;
- rdata.leftIndex = 0;
- ok = match_or_replace(cx, replace_glob, replace_destroy, &rdata.base,
- argc, vp);
- if (!ok)
- return JS_FALSE;
- if (!rdata.chars) {
- if ((rdata.base.flags & GLOBAL_REGEXP) || *vp != JSVAL_TRUE) {
- /* Didn't match even once. */
- *vp = STRING_TO_JSVAL(rdata.base.str);
- goto out;
- }
- leftlen = cx->regExpStatics.leftContext.length;
- ok = find_replen(cx, &rdata, &length);
- if (!ok)
- goto out;
- length += leftlen;
- chars = (jschar *) JS_malloc(cx, (length + 1) * sizeof(jschar));
- if (!chars) {
- ok = JS_FALSE;
- goto out;
- }
- js_strncpy(chars, cx->regExpStatics.leftContext.chars, leftlen);
- do_replace(cx, &rdata, chars + leftlen);
- rdata.chars = chars;
- rdata.length = length;
- }
- rightlen = cx->regExpStatics.rightContext.length;
- length = rdata.length + rightlen;
- chars = (jschar *)
- JS_realloc(cx, rdata.chars, (length + 1) * sizeof(jschar));
- if (!chars) {
- JS_free(cx, rdata.chars);
- ok = JS_FALSE;
- goto out;
- }
- js_strncpy(chars + rdata.length, cx->regExpStatics.rightContext.chars,
- rightlen);
- chars[length] = 0;
- str = js_NewString(cx, chars, length);
- if (!str) {
- JS_free(cx, chars);
- ok = JS_FALSE;
- goto out;
- }
- *vp = STRING_TO_JSVAL(str);
- out:
- /* If KEEP_REGEXP is still set, it's our job to destroy regexp now. */
- if (rdata.base.flags & KEEP_REGEXP)
- js_DestroyRegExp(cx, rdata.base.regexp);
- return ok;
- }
- /*
- * Subroutine used by str_split to find the next split point in str, starting
- * at offset *ip and looking either for the separator substring given by sep, or
- * for the next re match. In the re case, return the matched separator in *sep,
- * and the possibly updated offset in *ip.
- *
- * Return -2 on error, -1 on end of string, >= 0 for a valid index of the next
- * separator occurrence if found, or str->length if no separator is found.
- */
- static jsint
- find_split(JSContext *cx, JSString *str, JSRegExp *re, jsint *ip,
- JSSubString *sep)
- {
- jsint i, j, k;
- size_t length;
- jschar *chars;
- /*
- * Stop if past end of string. If at end of string, we will compare the
- * null char stored there (by js_NewString*) to sep->chars[j] in the while
- * loop at the end of this function, so that
- *
- * "ab,".split(',') => ["ab", ""]
- *
- * and the resulting array converts back to the string "ab," for symmetry.
- * However, we ape Perl and do this only if there is a sufficiently large
- * limit argument (see str_split).
- */
- i = *ip;
- length = JSSTRING_LENGTH(str);
- if ((size_t)i > length)
- return -1;
- chars = JSSTRING_CHARS(str);
- /*
- * Match a regular expression against the separator at or above index i.
- * Call js_ExecuteRegExp with true for the test argument. On successful
- * match, get the separator from cx->regExpStatics.lastMatch.
- */
- if (re) {
- size_t index;
- jsval rval;
- again:
- /* JS1.2 deviated from Perl by never matching at end of string. */
- index = (size_t)i;
- if (!js_ExecuteRegExp(cx, re, str, &index, JS_TRUE, &rval))
- return -2;
- if (rval != JSVAL_TRUE) {
- /* Mismatch: ensure our caller advances i past end of string. */
- sep->length = 1;
- return length;
- }
- i = (jsint)index;
- *sep = cx->regExpStatics.lastMatch;
- if (sep->length == 0) {
- /*
- * Empty string match: never split on an empty match at the start
- * of a find_split cycle. Same rule as for an empty global match
- * in match_or_replace.
- */
- if (i == *ip) {
- /*
- * "Bump-along" to avoid sticking at an empty match, but don't
- * bump past end of string -- our caller must do that by adding
- * sep->length to our return value.
- */
- if ((size_t)i == length)
- return -1;
- i++;
- goto again;
- }
- if ((size_t)i == length) {
- /*
- * If there was a trivial zero-length match at the end of the
- * split, then we shouldn't output the matched string at the end
- * of the split array. See ECMA-262 Ed. 3, 15.5.4.14, Step 15.
- */
- sep->chars = NULL;
- }
- }
- JS_ASSERT((size_t)i >= sep->length);
- return i - sep->length;
- }
- /*
- * Special case: if sep is the empty string, split str into one character
- * substrings. Let our caller worry about whether to split once at end of
- * string into an empty substring.
- */
- if (sep->length == 0)
- return ((size_t)i == length) ? -1 : i + 1;
- /*
- * Now that we know sep is non-empty, search starting at i in str for an
- * occurrence of all of sep's chars. If we find them, return the index of
- * the first separator char. Otherwise, return length.
- */
- j = 0;
- while ((size_t)(k = i + j) < length) {
- if (chars[k] == sep->chars[j]) {
- if ((size_t)++j == sep->length)
- return i;
- } else {
- i++;
- j = 0;
- }
- }
- return k;
- }
- static JSBool
- str_split(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str, *sub;
- JSObject *arrayobj;
- jsval v;
- JSBool ok, limited;
- JSRegExp *re;
- JSSubString *sep, tmp;
- jsdouble d;
- jsint i, j;
- uint32 len, limit;
- NORMALIZE_THIS(cx, vp, str);
- arrayobj = js_ConstructObject(cx, &js_ArrayClass, NULL, NULL, 0, NULL);
- if (!arrayobj)
- return JS_FALSE;
- *vp = OBJECT_TO_JSVAL(arrayobj);
- if (argc == 0) {
- v = STRING_TO_JSVAL(str);
- ok = OBJ_SET_PROPERTY(cx, arrayobj, INT_TO_JSID(0), &v);
- } else {
- if (VALUE_IS_REGEXP(cx, vp[2])) {
- re = (JSRegExp *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(vp[2]));
- sep = &tmp;
- /* Set a magic value so we can detect a successful re match. */
- sep->chars = NULL;
- sep->length = 0;
- } else {
- JSString *str2 = js_ValueToString(cx, vp[2]);
- if (!str2)
- return JS_FALSE;
- vp[2] = STRING_TO_JSVAL(str2);
- /*
- * Point sep at a local copy of str2's header because find_split
- * will modify sep->length.
- */
- JSSTRING_CHARS_AND_LENGTH(str2, tmp.chars, tmp.length);
- sep = &tmp;
- re = NULL;
- }
- /* Use the second argument as the split limit, if given. */
- limited = (argc > 1) && !JSVAL_IS_VOID(vp[3]);
- limit = 0; /* Avoid warning. */
- if (limited) {
- d = js_ValueToNumber(cx, &vp[3]);
- if (JSVAL_IS_NULL(vp[3]))
- return JS_FALSE;
- /* Clamp limit between 0 and 1 + string length. */
- limit = js_DoubleToECMAUint32(d);
- if (limit > JSSTRING_LENGTH(str))
- limit = 1 + JSSTRING_LENGTH(str);
- }
- len = i = 0;
- while ((j = find_split(cx, str, re, &i, sep)) >= 0) {
- if (limited && len >= limit)
- break;
- sub = js_NewDependentString(cx, str, i, (size_t)(j - i));
- if (!sub)
- return JS_FALSE;
- v = STRING_TO_JSVAL(sub);
- if (!JS_SetElement(cx, arrayobj, len, &v))
- return JS_FALSE;
- len++;
- /*
- * Imitate perl's feature of including parenthesized substrings
- * that matched part of the delimiter in the new array, after the
- * split substring that was delimited.
- */
- if (re && sep->chars) {
- uintN num;
- JSSubString *parsub;
- for (num = 0; num < cx->regExpStatics.parenCount; num++) {
- if (limited && len >= limit)
- break;
- parsub = REGEXP_PAREN_SUBSTRING(&cx->regExpStatics, num);
- sub = js_NewStringCopyN(cx, parsub->chars, parsub->length);
- if (!sub)
- return JS_FALSE;
- v = STRING_TO_JSVAL(sub);
- if (!JS_SetElement(cx, arrayobj, len, &v))
- return JS_FALSE;
- len++;
- }
- sep->chars = NULL;
- }
- i = j + sep->length;
- }
- ok = (j != -2);
- }
- return ok;
- }
- #ifdef JS_TRACER
- static JSObject* FASTCALL
- String_p_split(JSContext* cx, JSString* str, JSString* sepstr)
- {
- // FIXME: Avoid building and then parsing this array.
- jsval vp[4] = { JSVAL_NULL, STRING_TO_JSVAL(str), STRING_TO_JSVAL(sepstr), JSVAL_VOID };
- if (!str_split(cx, 2, vp))
- return NULL;
- JS_ASSERT(JSVAL_IS_OBJECT(vp[0]));
- return JSVAL_TO_OBJECT(vp[0]);
- }
- #endif
- #if JS_HAS_PERL_SUBSTR
- static JSBool
- str_substr(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str;
- jsdouble d;
- jsdouble length, begin, end;
- NORMALIZE_THIS(cx, vp, str);
- if (argc != 0) {
- d = js_ValueToNumber(cx, &vp[2]);
- if (JSVAL_IS_NULL(vp[2]))
- return JS_FALSE;
- length = JSSTRING_LENGTH(str);
- begin = js_DoubleToInteger(d);
- if (begin < 0) {
- begin += length;
- if (begin < 0)
- begin = 0;
- } else if (begin > length) {
- begin = length;
- }
- if (argc == 1) {
- end = length;
- } else {
- d = js_ValueToNumber(cx, &vp[3]);
- if (JSVAL_IS_NULL(vp[3]))
- return JS_FALSE;
- end = js_DoubleToInteger(d);
- if (end < 0)
- end = 0;
- end += begin;
- if (end > length)
- end = length;
- }
- str = js_NewDependentString(cx, str,
- (size_t)begin,
- (size_t)(end - begin));
- if (!str)
- return JS_FALSE;
- }
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #endif /* JS_HAS_PERL_SUBSTR */
- /*
- * Python-esque sequence operations.
- */
- static JSBool
- str_concat(JSContext *cx, uintN argc, jsval *vp)
- {
- JSString *str, *str2;
- jsval *argv;
- uintN i;
- NORMALIZE_THIS(cx, vp, str);
- for (i = 0, argv = vp + 2; i < argc; i++) {
- str2 = js_ValueToString(cx, argv[i]);
- if (!str2)
- return JS_FALSE;
- argv[i] = STRING_TO_JSVAL(str2);
- str = js_ConcatStrings(cx, str, str2);
- if (!str)
- return JS_FALSE;
- }
- *vp = STRING_TO_JSVAL(str);
- return JS_TRUE;
- }
- #ifdef JS_TRACER
- static JSString* FASTCALL
- String_p_concat_1int(JSContext* cx, JSString* str, int32 i)
- {
- // FIXME: should be able to use stack buffer and avoid istr...
- JSString* istr = js_NumberToString(cx, i);
- if (!istr)
- return NULL;
- return js_ConcatStrings(cx, str, istr);
- }
- static JSString* FASTCALL
- String_p_concat_2str(JSContext* cx, JSString* str, JSString* a, JSString* b)
- {
- str = js_ConcatStrings(cx, str, a);
- if (str)
- return js_ConcatStrings(cx, str, b);
- r