2009-06-10 18:29:44 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2008-10-21 15:21:23 -07:00
|
|
|
* vim: set ts=8 sw=4 et tw=99:
|
2007-03-22 10:30:00 -07:00
|
|
|
*
|
|
|
|
* ***** 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 <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "jstypes.h"
|
2009-03-18 11:38:16 -07:00
|
|
|
#include "jsstdint.h"
|
2010-10-01 16:46:54 -07:00
|
|
|
#include "jsutil.h"
|
|
|
|
#include "jshash.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "jsprf.h"
|
|
|
|
#include "jsapi.h"
|
|
|
|
#include "jsarray.h"
|
|
|
|
#include "jsatom.h"
|
|
|
|
#include "jsbool.h"
|
2008-10-08 15:08:33 -07:00
|
|
|
#include "jsbuiltins.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "jscntxt.h"
|
|
|
|
#include "jsgc.h"
|
|
|
|
#include "jsinterp.h"
|
|
|
|
#include "jslock.h"
|
|
|
|
#include "jsnum.h"
|
|
|
|
#include "jsobj.h"
|
|
|
|
#include "jsopcode.h"
|
2010-09-20 12:43:53 -07:00
|
|
|
#include "jsprobes.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "jsregexp.h"
|
2008-03-04 15:40:10 -08:00
|
|
|
#include "jsscope.h"
|
2009-01-30 15:40:05 -08:00
|
|
|
#include "jsstaticcheck.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "jsstr.h"
|
2008-01-29 18:36:33 -08:00
|
|
|
#include "jsbit.h"
|
2009-06-30 17:19:42 -07:00
|
|
|
#include "jsvector.h"
|
2010-03-30 00:44:28 -07:00
|
|
|
#include "jsversion.h"
|
2010-04-14 18:57:30 -07:00
|
|
|
|
2011-06-15 10:15:37 -07:00
|
|
|
#include "vm/GlobalObject.h"
|
|
|
|
|
2010-09-28 15:23:43 -07:00
|
|
|
#include "jsinterpinlines.h"
|
2010-04-14 18:57:30 -07:00
|
|
|
#include "jsobjinlines.h"
|
2010-08-11 13:30:07 -07:00
|
|
|
#include "jsregexpinlines.h"
|
2010-10-28 14:33:32 -07:00
|
|
|
#include "jsautooplen.h" // generated headers last
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-04-20 13:33:05 -07:00
|
|
|
#include "vm/StringObject-inl.h"
|
2011-06-20 11:44:20 -07:00
|
|
|
#include "vm/String-inl.h"
|
2011-04-20 13:33:05 -07:00
|
|
|
|
2010-01-22 14:49:18 -08:00
|
|
|
using namespace js;
|
2010-09-24 10:54:39 -07:00
|
|
|
using namespace js::gc;
|
2011-07-26 14:10:33 -07:00
|
|
|
using namespace js::unicode;
|
2010-01-22 14:49:18 -08:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
#ifdef JS_TRACER
|
|
|
|
|
|
|
|
JSBool JS_FASTCALL
|
|
|
|
js_FlattenOnTrace(JSContext *cx, JSString* str)
|
2007-08-15 23:23:06 -07:00
|
|
|
{
|
2011-03-14 13:59:53 -07:00
|
|
|
return !!str->ensureLinear(cx);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2011-03-14 13:59:53 -07:00
|
|
|
JS_DEFINE_CALLINFO_2(extern, BOOL, js_FlattenOnTrace, CONTEXT, STRING, 0, nanojit::ACCSET_STORE_ANY)
|
|
|
|
|
|
|
|
#endif /* !JS_TRACER */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
static JSLinearString *
|
2010-07-14 23:19:36 -07:00
|
|
|
ArgToRootedString(JSContext *cx, uintN argc, Value *vp, uintN arg)
|
2008-08-08 09:02:50 -07:00
|
|
|
{
|
|
|
|
if (arg >= argc)
|
2011-03-14 13:59:53 -07:00
|
|
|
return cx->runtime->atomState.typeAtoms[JSTYPE_VOID];
|
2008-08-08 09:02:50 -07:00
|
|
|
vp += 2 + arg;
|
|
|
|
|
2011-04-01 15:24:21 -07:00
|
|
|
if (!ToPrimitive(cx, JSTYPE_STRING, vp))
|
2009-06-30 17:19:42 -07:00
|
|
|
return NULL;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str;
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp->isString()) {
|
2010-12-06 10:26:58 -08:00
|
|
|
str = vp->toString()->ensureLinear(cx);
|
2010-07-14 23:19:36 -07:00
|
|
|
} else if (vp->isBoolean()) {
|
2011-03-14 13:59:53 -07:00
|
|
|
str = cx->runtime->atomState.booleanAtoms[(int)vp->toBoolean()];
|
2010-07-14 23:19:36 -07:00
|
|
|
} else if (vp->isNull()) {
|
2011-03-14 13:59:53 -07:00
|
|
|
str = cx->runtime->atomState.nullAtom;
|
2010-07-14 23:19:36 -07:00
|
|
|
} else if (vp->isUndefined()) {
|
2011-03-14 13:59:53 -07:00
|
|
|
str = cx->runtime->atomState.typeAtoms[JSTYPE_VOID];
|
2009-06-30 17:19:42 -07:00
|
|
|
}
|
|
|
|
else {
|
2010-12-06 10:26:58 -08:00
|
|
|
str = NumberToString(cx, vp->toNumber());
|
|
|
|
if (!str)
|
|
|
|
return NULL;
|
|
|
|
vp->setString(str);
|
2008-08-08 09:02:50 -07:00
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/*
|
|
|
|
* Forward declarations for URI encode/decode and helper routines
|
|
|
|
*/
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_decodeURI(JSContext *cx, uintN argc, Value *vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_decodeURI_Component(JSContext *cx, uintN argc, Value *vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_encodeURI(JSContext *cx, uintN argc, Value *vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_encodeURI_Component(JSContext *cx, uintN argc, Value *vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-07-05 09:38:35 -07:00
|
|
|
static const uint32 INVALID_UTF8 = UINT32_MAX;
|
2009-10-01 17:10:14 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
static uint32
|
|
|
|
Utf8ToOneUcs4Char(const uint8 *utf8Buffer, int utf8Length);
|
|
|
|
|
|
|
|
/*
|
2011-07-05 16:51:12 -07:00
|
|
|
* Global string methods
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2011-07-05 16:51:12 -07:00
|
|
|
/* ES5 B.2.1 */
|
|
|
|
static JSBool
|
|
|
|
str_escape(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
const char digits[] = {'0', '1', '2', '3', '4', '5', '6', '7',
|
|
|
|
'8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
|
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
JSLinearString *str = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length = str->length();
|
|
|
|
const jschar *chars = str->chars();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-07-05 16:51:12 -07:00
|
|
|
static const uint8 shouldPassThrough[256] = {
|
|
|
|
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
|
|
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
|
|
|
|
0,0,0,0,0,0,0,0,0,0,1,1,0,1,1,1, /* !"#$%&'()*+,-./ */
|
|
|
|
1,1,1,1,1,1,1,1,1,1,0,0,0,0,0,0, /* 0123456789:;<=>? */
|
|
|
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* @ABCDEFGHIJKLMNO */
|
|
|
|
1,1,1,1,1,1,1,1,1,1,1,0,0,0,0,1, /* PQRSTUVWXYZ[\]^_ */
|
|
|
|
0,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* `abcdefghijklmno */
|
|
|
|
1,1,1,1,1,1,1,1,1,1,1,0,0,0,0,0,0, /* pqrstuvwxyz{\}~ DEL */
|
|
|
|
};
|
|
|
|
|
|
|
|
/* In step 7, exactly 69 characters should pass through unencoded. */
|
|
|
|
#ifdef DEBUG
|
|
|
|
int count = 0;
|
|
|
|
for (uint i = 0; i < sizeof(shouldPassThrough); i++) {
|
|
|
|
if (shouldPassThrough[i]) {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
JS_ASSERT(count == 69);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/* Take a first pass and see how big the result string will need to be. */
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t newlength = length;
|
|
|
|
for (size_t i = 0; i < length; i++) {
|
2011-07-05 16:51:12 -07:00
|
|
|
jschar ch = chars[i];
|
|
|
|
if (ch < 128 && shouldPassThrough[ch])
|
2007-03-22 10:30:00 -07:00
|
|
|
continue;
|
2011-07-05 16:51:12 -07:00
|
|
|
|
|
|
|
/* The character will be encoded as %XX or %uXXXX. */
|
|
|
|
newlength += (ch < 256) ? 2 : 5;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This overflow test works because newlength is incremented by at
|
|
|
|
* most 5 on each iteration.
|
|
|
|
*/
|
|
|
|
if (newlength < length) {
|
2008-03-12 16:07:47 -07:00
|
|
|
js_ReportAllocationOverflow(cx);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newlength >= ~(size_t)0 / sizeof(jschar)) {
|
2008-03-12 16:07:47 -07:00
|
|
|
js_ReportAllocationOverflow(cx);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *newchars = (jschar *) cx->malloc_((newlength + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!newchars)
|
|
|
|
return JS_FALSE;
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t i, ni;
|
2007-03-22 10:30:00 -07:00
|
|
|
for (i = 0, ni = 0; i < length; i++) {
|
2011-07-05 16:51:12 -07:00
|
|
|
jschar ch = chars[i];
|
|
|
|
if (ch < 128 && shouldPassThrough[ch]) {
|
2007-03-22 10:30:00 -07:00
|
|
|
newchars[ni++] = ch;
|
|
|
|
} else if (ch < 256) {
|
2011-07-05 16:51:12 -07:00
|
|
|
newchars[ni++] = '%';
|
|
|
|
newchars[ni++] = digits[ch >> 4];
|
|
|
|
newchars[ni++] = digits[ch & 0xF];
|
2007-03-22 10:30:00 -07:00
|
|
|
} 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;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSString *retstr = js_NewString(cx, newchars, newlength);
|
|
|
|
if (!retstr) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(newchars);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2011-07-05 16:51:12 -07:00
|
|
|
vp->setString(retstr);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2011-07-05 16:51:12 -07:00
|
|
|
/* ES5 B.2.2 */
|
2007-08-01 21:33:52 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_unescape(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length = str->length();
|
|
|
|
const jschar *chars = str->chars();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/* Don't bother allocating less space for the new string. */
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *newchars = (jschar *) cx->malloc_((length + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!newchars)
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
|
|
|
size_t ni = 0, i = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
while (i < length) {
|
2010-12-06 10:26:58 -08:00
|
|
|
jschar ch = chars[i++];
|
2007-03-22 10:30:00 -07:00
|
|
|
if (ch == '%') {
|
2011-07-05 16:51:12 -07:00
|
|
|
/* Incomplete escapes are interpreted as literal characters. */
|
2007-03-22 10:30:00 -07:00
|
|
|
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;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSString *retstr = js_NewString(cx, newchars, ni);
|
|
|
|
if (!retstr) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(newchars);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
vp->setString(retstr);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if JS_HAS_UNEVAL
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_uneval(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
JSString *str;
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
str = js_ValueToSource(cx, argc != 0 ? vp[2] : UndefinedValue());
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2007-03-22 10:30:00 -07:00
|
|
|
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[] = {
|
2008-08-08 09:02:50 -07:00
|
|
|
JS_FN(js_escape_str, str_escape, 1,0),
|
|
|
|
JS_FN(js_unescape_str, str_unescape, 1,0),
|
2007-03-22 10:30:00 -07:00
|
|
|
#if JS_HAS_UNEVAL
|
2008-08-08 09:02:50 -07:00
|
|
|
JS_FN(js_uneval_str, str_uneval, 1,0),
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
2008-08-08 09:02:50 -07:00
|
|
|
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),
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-08-01 21:33:52 -07:00
|
|
|
JS_FS_END
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
jschar js_empty_ucstr[] = {0};
|
|
|
|
JSSubString js_EmptySubString = {0, js_empty_ucstr};
|
|
|
|
|
|
|
|
#define STRING_ELEMENT_ATTRS (JSPROP_ENUMERATE|JSPROP_READONLY|JSPROP_PERMANENT)
|
|
|
|
|
|
|
|
static JSBool
|
|
|
|
str_enumerate(JSContext *cx, JSObject *obj)
|
|
|
|
{
|
2011-03-13 20:38:34 -07:00
|
|
|
JSString *str = obj->getPrimitiveThis().toString();
|
|
|
|
for (size_t i = 0, length = str->length(); i < length; i++) {
|
|
|
|
JSString *str1 = js_NewDependentString(cx, str, i, 1);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str1)
|
2011-03-13 20:38:34 -07:00
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
if (!obj->defineProperty(cx, INT_TO_JSID(i), StringValue(str1),
|
2011-02-09 11:31:40 -08:00
|
|
|
PropertyStub, StrictPropertyStub,
|
2009-08-26 14:28:36 -07:00
|
|
|
STRING_ELEMENT_ATTRS)) {
|
2011-03-13 20:38:34 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
2009-09-24 14:33:14 -07:00
|
|
|
|
2011-03-13 20:38:34 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_resolve(JSContext *cx, JSObject *obj, jsid id, uintN flags,
|
2007-03-22 10:30:00 -07:00
|
|
|
JSObject **objp)
|
|
|
|
{
|
2010-09-15 13:43:55 -07:00
|
|
|
if (!JSID_IS_INT(id))
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
JSString *str = obj->getPrimitiveThis().toString();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
jsint slot = JSID_TO_INT(id);
|
2009-06-10 18:29:44 -07:00
|
|
|
if ((size_t)slot < str->length()) {
|
2011-03-14 13:59:53 -07:00
|
|
|
JSString *str1 = JSAtom::getUnitStringForElement(cx, str, size_t(slot));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str1)
|
|
|
|
return JS_FALSE;
|
2010-07-14 23:19:36 -07:00
|
|
|
if (!obj->defineProperty(cx, id, StringValue(str1), NULL, NULL,
|
2009-08-26 14:28:36 -07:00
|
|
|
STRING_ELEMENT_ATTRS)) {
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
*objp = obj;
|
|
|
|
}
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
Class js_StringClass = {
|
2007-03-22 10:30:00 -07:00
|
|
|
js_String_str,
|
2011-04-20 13:33:05 -07:00
|
|
|
JSCLASS_HAS_RESERVED_SLOTS(StringObject::RESERVED_SLOTS) |
|
2011-03-13 20:38:34 -07:00
|
|
|
JSCLASS_NEW_RESOLVE | JSCLASS_HAS_CACHED_PROTO(JSProto_String),
|
2011-02-09 11:31:40 -08:00
|
|
|
PropertyStub, /* addProperty */
|
|
|
|
PropertyStub, /* delProperty */
|
2011-03-13 20:38:34 -07:00
|
|
|
PropertyStub, /* getProperty */
|
2011-02-09 11:31:40 -08:00
|
|
|
StrictPropertyStub, /* setProperty */
|
2010-06-12 09:29:04 -07:00
|
|
|
str_enumerate,
|
|
|
|
(JSResolveOp)str_resolve,
|
|
|
|
ConvertStub
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
/*
|
|
|
|
* Returns a JSString * for the |this| value associated with vp, or throws a
|
|
|
|
* TypeError if |this| is null or undefined. This algorithm is the same as
|
|
|
|
* calling CheckObjectCoercible(this), then returning ToString(this), as all
|
|
|
|
* String.prototype.* methods do.
|
|
|
|
*/
|
|
|
|
static JS_ALWAYS_INLINE JSString *
|
|
|
|
ThisToStringForStringProto(JSContext *cx, Value *vp)
|
|
|
|
{
|
2011-05-18 10:57:52 -07:00
|
|
|
JS_CHECK_RECURSION(cx, return NULL);
|
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
if (vp[1].isString())
|
|
|
|
return vp[1].toString();
|
2009-12-03 11:39:55 -08:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp[1].isObject()) {
|
|
|
|
JSObject *obj = &vp[1].toObject();
|
2011-01-26 13:37:45 -08:00
|
|
|
if (obj->getClass() == &js_StringClass &&
|
|
|
|
ClassMethodIsNative(cx, obj,
|
|
|
|
&js_StringClass,
|
|
|
|
ATOM_TO_JSID(cx->runtime->atomState.toStringAtom),
|
|
|
|
js_str_toString))
|
|
|
|
{
|
2010-04-14 18:57:30 -07:00
|
|
|
vp[1] = obj->getPrimitiveThis();
|
2010-07-14 23:19:36 -07:00
|
|
|
return vp[1].toString();
|
2009-12-03 20:35:36 -08:00
|
|
|
}
|
2011-01-26 13:37:45 -08:00
|
|
|
} else if (vp[1].isNullOrUndefined()) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_CANT_CONVERT_TO,
|
|
|
|
vp[1].isNull() ? "null" : "undefined", "object");
|
|
|
|
return NULL;
|
2009-12-03 11:39:55 -08:00
|
|
|
}
|
2009-12-03 20:35:36 -08:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
JSString *str = js_ValueToString(cx, vp[1]);
|
2009-12-03 20:35:36 -08:00
|
|
|
if (!str)
|
|
|
|
return NULL;
|
2010-07-14 23:19:36 -07:00
|
|
|
vp[1].setString(str);
|
2008-02-17 16:12:33 -08:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
#if JS_HAS_TOSOURCE
|
|
|
|
|
|
|
|
/*
|
|
|
|
* String.prototype.quote is generic (as are most string methods), unlike
|
|
|
|
* toSource, toString, and valueOf.
|
|
|
|
*/
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_quote(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
str = js_QuoteString(cx, str, '"');
|
|
|
|
if (!str)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_toSource(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
JSString *str;
|
2010-10-12 11:50:02 -07:00
|
|
|
if (!GetPrimitiveThis(cx, vp, &str))
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-10-12 11:50:02 -07:00
|
|
|
str = js_QuoteString(cx, str, '"');
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2010-10-12 11:50:02 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
char buf[16];
|
|
|
|
size_t j = JS_snprintf(buf, sizeof buf, "(new String(");
|
|
|
|
|
2011-02-24 18:52:27 -08:00
|
|
|
JS::Anchor<JSString *> anchor(str);
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t k = str->length();
|
|
|
|
const jschar *s = str->getChars(cx);
|
|
|
|
if (!s)
|
|
|
|
return false;
|
2010-10-12 11:50:02 -07:00
|
|
|
|
|
|
|
size_t n = j + k + 2;
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *t = (jschar *) cx->malloc_((n + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!t)
|
2010-10-12 11:50:02 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
size_t i;
|
2007-03-22 10:30:00 -07:00
|
|
|
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;
|
2010-10-12 11:50:02 -07:00
|
|
|
|
2007-08-15 23:23:06 -07:00
|
|
|
str = js_NewString(cx, t, n);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(t);
|
2010-10-12 11:50:02 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2010-10-12 11:50:02 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* JS_HAS_TOSOURCE */
|
|
|
|
|
2009-07-15 17:36:24 -07:00
|
|
|
JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
js_str_toString(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-10-12 11:50:02 -07:00
|
|
|
JSString *str;
|
|
|
|
if (!GetPrimitiveThis(cx, vp, &str))
|
2010-07-14 23:19:36 -07:00
|
|
|
return false;
|
2010-10-12 11:50:02 -07:00
|
|
|
vp->setString(str);
|
2010-07-14 23:19:36 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Java-like string native methods.
|
|
|
|
*/
|
2010-12-29 07:22:45 -08:00
|
|
|
|
|
|
|
JS_ALWAYS_INLINE bool
|
|
|
|
ValueToIntegerRange(JSContext *cx, const Value &v, int32 *out)
|
2008-10-21 15:21:23 -07:00
|
|
|
{
|
2010-12-29 07:22:45 -08:00
|
|
|
if (v.isInt32()) {
|
|
|
|
*out = v.toInt32();
|
|
|
|
} else {
|
|
|
|
double d;
|
2011-04-02 11:33:20 -07:00
|
|
|
if (!ToInteger(cx, v, &d))
|
2010-12-29 07:22:45 -08:00
|
|
|
return false;
|
|
|
|
if (d > INT32_MAX)
|
|
|
|
*out = INT32_MAX;
|
|
|
|
else if (d < INT32_MIN)
|
|
|
|
*out = INT32_MIN;
|
|
|
|
else
|
|
|
|
*out = int32(d);
|
2008-10-21 15:21:23 -07:00
|
|
|
}
|
|
|
|
|
2010-12-29 07:22:45 -08:00
|
|
|
return true;
|
2008-10-21 15:21:23 -07:00
|
|
|
}
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_substring(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2010-12-29 07:22:45 -08:00
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
int32 length, begin, end;
|
2010-12-29 07:22:45 -08:00
|
|
|
if (argc > 0) {
|
|
|
|
end = length = int32(str->length());
|
|
|
|
|
|
|
|
if (!ValueToIntegerRange(cx, vp[2], &begin))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (begin < 0)
|
|
|
|
begin = 0;
|
|
|
|
else if (begin > length)
|
|
|
|
begin = length;
|
|
|
|
|
|
|
|
if (argc > 1 && !vp[3].isUndefined()) {
|
|
|
|
if (!ValueToIntegerRange(cx, vp[3], &end))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (end > length) {
|
|
|
|
end = length;
|
|
|
|
} else {
|
|
|
|
if (end < 0)
|
|
|
|
end = 0;
|
|
|
|
if (end < begin) {
|
|
|
|
int32_t tmp = begin;
|
|
|
|
begin = end;
|
|
|
|
end = tmp;
|
|
|
|
}
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-12-29 07:22:45 -08:00
|
|
|
str = js_NewDependentString(cx, str, size_t(begin), size_t(end - begin));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2010-12-29 07:22:45 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-12-29 07:22:45 -08:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2010-12-29 07:22:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-08-29 17:35:00 -07:00
|
|
|
JSString* JS_FASTCALL
|
|
|
|
js_toLowerCase(JSContext *cx, JSString *str)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t n = str->length();
|
|
|
|
const jschar *s = str->getChars(cx);
|
|
|
|
if (!s)
|
|
|
|
return NULL;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *news = (jschar *) cx->malloc_((n + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!news)
|
2008-08-29 17:35:00 -07:00
|
|
|
return NULL;
|
2010-12-06 10:26:58 -08:00
|
|
|
for (size_t i = 0; i < n; i++)
|
2011-07-26 14:10:33 -07:00
|
|
|
news[i] = unicode::ToLowerCase(s[i]);
|
2007-03-22 10:30:00 -07:00
|
|
|
news[n] = 0;
|
2007-08-15 23:23:06 -07:00
|
|
|
str = js_NewString(cx, news, n);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(news);
|
2008-08-29 17:35:00 -07:00
|
|
|
return NULL;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2008-08-29 17:35:00 -07:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_toLowerCase(JSContext *cx, uintN argc, Value *vp)
|
2008-08-29 17:35:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2008-08-29 17:35:00 -07:00
|
|
|
str = js_toLowerCase(cx, str);
|
|
|
|
if (!str)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_toLocaleLowerCase(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* 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) {
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
return cx->localeCallbacks->localeToLowerCase(cx, str, Jsvalify(vp));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2011-01-26 13:37:45 -08:00
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
return str_toLowerCase(cx, 0, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-08-29 17:35:00 -07:00
|
|
|
JSString* JS_FASTCALL
|
|
|
|
js_toUpperCase(JSContext *cx, JSString *str)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t n = str->length();
|
|
|
|
const jschar *s = str->getChars(cx);
|
|
|
|
if (!s)
|
|
|
|
return NULL;
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *news = (jschar *) cx->malloc_((n + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!news)
|
2008-08-29 17:35:00 -07:00
|
|
|
return NULL;
|
2010-12-06 10:26:58 -08:00
|
|
|
for (size_t i = 0; i < n; i++)
|
2011-07-26 14:10:33 -07:00
|
|
|
news[i] = unicode::ToUpperCase(s[i]);
|
2007-03-22 10:30:00 -07:00
|
|
|
news[n] = 0;
|
2007-08-15 23:23:06 -07:00
|
|
|
str = js_NewString(cx, news, n);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(news);
|
2008-08-29 17:35:00 -07:00
|
|
|
return NULL;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2008-08-29 17:35:00 -07:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_toUpperCase(JSContext *cx, uintN argc, Value *vp)
|
2008-08-29 17:35:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2008-08-29 17:35:00 -07:00
|
|
|
str = js_toUpperCase(cx, str);
|
|
|
|
if (!str)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_toLocaleUpperCase(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Forcefully ignore the first (or any) argument and return toUpperCase(),
|
2008-02-18 13:01:47 -08:00
|
|
|
* ECMA has reserved that argument, presumably for defining the locale.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
|
|
|
if (cx->localeCallbacks && cx->localeCallbacks->localeToUpperCase) {
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
return cx->localeCallbacks->localeToUpperCase(cx, str, Jsvalify(vp));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2011-01-26 13:37:45 -08:00
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
return str_toUpperCase(cx, 0, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_localeCompare(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (argc == 0) {
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32(0);
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *thatStr = js_ValueToString(cx, vp[2]);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!thatStr)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (cx->localeCallbacks && cx->localeCallbacks->localeCompare) {
|
2010-07-14 23:19:36 -07:00
|
|
|
vp[2].setString(thatStr);
|
|
|
|
return cx->localeCallbacks->localeCompare(cx, str, thatStr, Jsvalify(vp));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
int32 result;
|
|
|
|
if (!CompareStrings(cx, str, thatStr, &result))
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2010-12-06 10:26:58 -08:00
|
|
|
vp->setInt32(result);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-08-05 22:54:34 -07:00
|
|
|
JSBool
|
|
|
|
js_str_charAt(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
JSString *str;
|
2007-08-01 21:33:52 -07:00
|
|
|
jsint i;
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp[1].isString() && argc != 0 && vp[2].isInt32()) {
|
|
|
|
str = vp[1].toString();
|
|
|
|
i = vp[2].toInt32();
|
2009-06-10 18:29:44 -07:00
|
|
|
if ((size_t)i >= str->length())
|
2007-08-01 21:33:52 -07:00
|
|
|
goto out_of_range;
|
2007-07-21 16:21:20 -07:00
|
|
|
} else {
|
2011-01-26 13:37:45 -08:00
|
|
|
str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2011-04-02 11:33:20 -07:00
|
|
|
double d = 0.0;
|
|
|
|
if (argc > 0 && !ToInteger(cx, vp[2], &d))
|
|
|
|
return false;
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2009-06-10 18:29:44 -07:00
|
|
|
if (d < 0 || str->length() <= d)
|
2007-08-01 21:33:52 -07:00
|
|
|
goto out_of_range;
|
|
|
|
i = (jsint) d;
|
2007-07-21 16:21:20 -07:00
|
|
|
}
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
str = JSAtom::getUnitStringForElement(cx, str, size_t(i));
|
2007-08-01 21:33:52 -07:00
|
|
|
if (!str)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
out_of_range:
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(cx->runtime->emptyString);
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-08-05 22:54:34 -07:00
|
|
|
JSBool
|
|
|
|
js_str_charCodeAt(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
JSString *str;
|
2007-08-01 21:33:52 -07:00
|
|
|
jsint i;
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp[1].isString() && argc != 0 && vp[2].isInt32()) {
|
|
|
|
str = vp[1].toString();
|
|
|
|
i = vp[2].toInt32();
|
2009-06-10 18:29:44 -07:00
|
|
|
if ((size_t)i >= str->length())
|
2007-08-01 21:33:52 -07:00
|
|
|
goto out_of_range;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
2011-01-26 13:37:45 -08:00
|
|
|
str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-07-21 14:39:42 -07:00
|
|
|
|
2011-04-02 11:33:20 -07:00
|
|
|
double d = 0.0;
|
|
|
|
if (argc > 0 && !ToInteger(cx, vp[2], &d))
|
|
|
|
return false;
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2009-06-10 18:29:44 -07:00
|
|
|
if (d < 0 || str->length() <= d)
|
2007-08-01 21:33:52 -07:00
|
|
|
goto out_of_range;
|
|
|
|
i = (jsint) d;
|
2007-07-21 16:21:20 -07:00
|
|
|
}
|
2007-08-01 21:33:52 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *chars;
|
|
|
|
chars = str->getChars(cx);
|
|
|
|
if (!chars)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
vp->setInt32(chars[i]);
|
|
|
|
return true;
|
2007-08-01 21:33:52 -07:00
|
|
|
|
|
|
|
out_of_range:
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setDouble(js_NaN);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
/*
|
|
|
|
* Boyer-Moore-Horspool superlinear search for pat:patlen in text:textlen.
|
|
|
|
* The patlen argument must be positive and no greater than sBMHPatLenMax.
|
|
|
|
*
|
|
|
|
* Return the index of pat in text, or -1 if not found.
|
|
|
|
*/
|
|
|
|
static const jsuint sBMHCharSetSize = 256; /* ISO-Latin-1 */
|
|
|
|
static const jsuint sBMHPatLenMax = 255; /* skip table element is uint8 */
|
|
|
|
static const jsint sBMHBadPattern = -2; /* return value if pat is not ISO-Latin-1 */
|
|
|
|
|
2007-08-02 02:14:06 -07:00
|
|
|
jsint
|
2009-09-23 08:54:50 -07:00
|
|
|
js_BoyerMooreHorspool(const jschar *text, jsuint textlen,
|
|
|
|
const jschar *pat, jsuint patlen)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-09-23 08:54:50 -07:00
|
|
|
uint8 skip[sBMHCharSetSize];
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
JS_ASSERT(0 < patlen && patlen <= sBMHPatLenMax);
|
|
|
|
for (jsuint i = 0; i < sBMHCharSetSize; i++)
|
2007-03-22 10:30:00 -07:00
|
|
|
skip[i] = (uint8)patlen;
|
2009-09-23 08:54:50 -07:00
|
|
|
jsuint m = patlen - 1;
|
|
|
|
for (jsuint i = 0; i < m; i++) {
|
|
|
|
jschar c = pat[i];
|
|
|
|
if (c >= sBMHCharSetSize)
|
|
|
|
return sBMHBadPattern;
|
2007-03-22 10:30:00 -07:00
|
|
|
skip[c] = (uint8)(m - i);
|
|
|
|
}
|
2009-09-23 08:54:50 -07:00
|
|
|
jschar c;
|
|
|
|
for (jsuint k = m;
|
2007-03-22 10:30:00 -07:00
|
|
|
k < textlen;
|
2009-09-23 08:54:50 -07:00
|
|
|
k += ((c = text[k]) >= sBMHCharSetSize) ? patlen : skip[c]) {
|
|
|
|
for (jsuint i = k, j = m; ; i--, j--) {
|
2007-03-22 10:30:00 -07:00
|
|
|
if (text[i] != pat[j])
|
|
|
|
break;
|
2009-09-23 08:54:50 -07:00
|
|
|
if (j == 0)
|
|
|
|
return static_cast<jsint>(i); /* safe: max string size */
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-03-16 21:43:48 -07:00
|
|
|
struct MemCmp {
|
|
|
|
typedef jsuint Extent;
|
|
|
|
static JS_ALWAYS_INLINE Extent computeExtent(const jschar *, jsuint patlen) {
|
|
|
|
return (patlen - 1) * sizeof(jschar);
|
|
|
|
}
|
|
|
|
static JS_ALWAYS_INLINE bool match(const jschar *p, const jschar *t, Extent extent) {
|
|
|
|
return memcmp(p, t, extent) == 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ManualCmp {
|
|
|
|
typedef const jschar *Extent;
|
|
|
|
static JS_ALWAYS_INLINE Extent computeExtent(const jschar *pat, jsuint patlen) {
|
|
|
|
return pat + patlen;
|
|
|
|
}
|
|
|
|
static JS_ALWAYS_INLINE bool match(const jschar *p, const jschar *t, Extent extent) {
|
|
|
|
for (; p != extent; ++p, ++t) {
|
|
|
|
if (*p != *t)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class InnerMatch>
|
|
|
|
static jsint
|
2010-04-30 18:48:55 -07:00
|
|
|
UnrolledMatch(const jschar *text, jsuint textlen, const jschar *pat, jsuint patlen)
|
2010-03-16 21:43:48 -07:00
|
|
|
{
|
|
|
|
JS_ASSERT(patlen > 0 && textlen > 0);
|
|
|
|
const jschar *textend = text + textlen - (patlen - 1);
|
|
|
|
const jschar p0 = *pat;
|
|
|
|
const jschar *const patNext = pat + 1;
|
|
|
|
const typename InnerMatch::Extent extent = InnerMatch::computeExtent(pat, patlen);
|
|
|
|
uint8 fixup;
|
|
|
|
|
|
|
|
const jschar *t = text;
|
|
|
|
switch ((textend - t) & 7) {
|
2010-04-30 18:48:55 -07:00
|
|
|
case 0: if (*t++ == p0) { fixup = 8; goto match; }
|
|
|
|
case 7: if (*t++ == p0) { fixup = 7; goto match; }
|
|
|
|
case 6: if (*t++ == p0) { fixup = 6; goto match; }
|
|
|
|
case 5: if (*t++ == p0) { fixup = 5; goto match; }
|
|
|
|
case 4: if (*t++ == p0) { fixup = 4; goto match; }
|
|
|
|
case 3: if (*t++ == p0) { fixup = 3; goto match; }
|
|
|
|
case 2: if (*t++ == p0) { fixup = 2; goto match; }
|
|
|
|
case 1: if (*t++ == p0) { fixup = 1; goto match; }
|
|
|
|
}
|
|
|
|
while (t != textend) {
|
|
|
|
if (t[0] == p0) { t += 1; fixup = 8; goto match; }
|
|
|
|
if (t[1] == p0) { t += 2; fixup = 7; goto match; }
|
|
|
|
if (t[2] == p0) { t += 3; fixup = 6; goto match; }
|
|
|
|
if (t[3] == p0) { t += 4; fixup = 5; goto match; }
|
|
|
|
if (t[4] == p0) { t += 5; fixup = 4; goto match; }
|
|
|
|
if (t[5] == p0) { t += 6; fixup = 3; goto match; }
|
|
|
|
if (t[6] == p0) { t += 7; fixup = 2; goto match; }
|
|
|
|
if (t[7] == p0) { t += 8; fixup = 1; goto match; }
|
|
|
|
t += 8;
|
|
|
|
continue;
|
2010-03-16 21:43:48 -07:00
|
|
|
do {
|
2010-04-30 18:48:55 -07:00
|
|
|
if (*t++ == p0) {
|
|
|
|
match:
|
|
|
|
if (!InnerMatch::match(patNext, t, extent))
|
|
|
|
goto failed_match;
|
|
|
|
return t - text - 1;
|
|
|
|
}
|
|
|
|
failed_match:;
|
|
|
|
} while (--fixup > 0);
|
2010-03-16 21:43:48 -07:00
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
static JS_ALWAYS_INLINE jsint
|
|
|
|
StringMatch(const jschar *text, jsuint textlen,
|
|
|
|
const jschar *pat, jsuint patlen)
|
2009-09-16 18:23:27 -07:00
|
|
|
{
|
2009-09-23 08:54:50 -07:00
|
|
|
if (patlen == 0)
|
|
|
|
return 0;
|
|
|
|
if (textlen < patlen)
|
|
|
|
return -1;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-03-31 08:21:07 -07:00
|
|
|
#if defined(__i386__) || defined(_M_IX86) || defined(__i386)
|
2009-11-17 22:40:50 -08:00
|
|
|
/*
|
|
|
|
* Given enough registers, the unrolled loop below is faster than the
|
|
|
|
* following loop. 32-bit x86 does not have enough registers.
|
|
|
|
*/
|
|
|
|
if (patlen == 1) {
|
|
|
|
const jschar p0 = *pat;
|
|
|
|
for (const jschar *c = text, *end = text + textlen; c != end; ++c) {
|
|
|
|
if (*c == p0)
|
|
|
|
return c - text;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
2009-11-19 10:34:28 -08:00
|
|
|
* If the text or pattern string is short, BMH will be more expensive than
|
|
|
|
* the basic linear scan due to initialization cost and a more complex loop
|
|
|
|
* body. While the correct threshold is input-dependent, we can make a few
|
|
|
|
* conservative observations:
|
|
|
|
* - When |textlen| is "big enough", the initialization time will be
|
|
|
|
* proportionally small, so the worst-case slowdown is minimized.
|
|
|
|
* - When |patlen| is "too small", even the best case for BMH will be
|
|
|
|
* slower than a simple scan for large |textlen| due to the more complex
|
|
|
|
* loop body of BMH.
|
|
|
|
* From this, the values for "big enough" and "too small" are determined
|
|
|
|
* empirically. See bug 526348.
|
2009-11-17 22:40:50 -08:00
|
|
|
*/
|
2009-11-19 10:34:28 -08:00
|
|
|
if (textlen >= 512 && patlen >= 11 && patlen <= sBMHPatLenMax) {
|
2009-09-23 08:54:50 -07:00
|
|
|
jsint index = js_BoyerMooreHorspool(text, textlen, pat, patlen);
|
|
|
|
if (index != sBMHBadPattern)
|
|
|
|
return index;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-11-17 22:40:50 -08:00
|
|
|
/*
|
2010-03-16 21:43:48 -07:00
|
|
|
* For big patterns with large potential overlap we want the SIMD-optimized
|
|
|
|
* speed of memcmp. For small patterns, a simple loop is faster.
|
|
|
|
*
|
|
|
|
* FIXME: Linux memcmp performance is sad and the manual loop is faster.
|
2009-11-17 22:40:50 -08:00
|
|
|
*/
|
2010-03-16 21:43:48 -07:00
|
|
|
return
|
|
|
|
#if !defined(__linux__)
|
2010-04-30 18:48:55 -07:00
|
|
|
patlen > 128 ? UnrolledMatch<MemCmp>(text, textlen, pat, patlen)
|
2010-03-16 21:43:48 -07:00
|
|
|
:
|
2009-11-17 22:40:50 -08:00
|
|
|
#endif
|
2010-04-30 18:48:55 -07:00
|
|
|
UnrolledMatch<ManualCmp>(text, textlen, pat, patlen);
|
2009-09-23 08:54:50 -07:00
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-07-22 17:54:39 -07:00
|
|
|
static const size_t sRopeMatchThresholdRatioLog2 = 5;
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
/*
|
|
|
|
* RopeMatch takes the text to search, the patern to search for in the text.
|
|
|
|
* RopeMatch returns false on OOM and otherwise returns the match index through
|
|
|
|
* the 'match' outparam (-1 for not found).
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
RopeMatch(JSContext *cx, JSString *textstr, const jschar *pat, jsuint patlen, jsint *match)
|
2010-07-22 17:54:39 -07:00
|
|
|
{
|
2010-11-30 18:41:32 -08:00
|
|
|
JS_ASSERT(textstr->isRope());
|
2010-07-22 17:54:39 -07:00
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
if (patlen == 0) {
|
|
|
|
*match = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (textstr->length() < patlen) {
|
|
|
|
*match = -1;
|
|
|
|
return true;
|
|
|
|
}
|
2010-07-22 17:54:39 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* List of leaf nodes in the rope. If we run out of memory when trying to
|
|
|
|
* append to this list, we can still fall back to StringMatch, so use the
|
|
|
|
* system allocator so we don't report OOM in that case.
|
|
|
|
*/
|
2011-03-14 13:59:53 -07:00
|
|
|
Vector<JSLinearString *, 16, SystemAllocPolicy> strs;
|
2010-07-22 17:54:39 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We don't want to do rope matching if there is a poor node-to-char ratio,
|
|
|
|
* since this means spending a lot of time in the match loop below. We also
|
|
|
|
* need to build the list of leaf nodes. Do both here: iterate over the
|
|
|
|
* nodes so long as there are not too many.
|
|
|
|
*/
|
2010-11-30 18:41:32 -08:00
|
|
|
{
|
|
|
|
size_t textstrlen = textstr->length();
|
|
|
|
size_t threshold = textstrlen >> sRopeMatchThresholdRatioLog2;
|
|
|
|
StringSegmentRange r(cx);
|
|
|
|
if (!r.init(textstr))
|
|
|
|
return false;
|
|
|
|
while (!r.empty()) {
|
|
|
|
if (threshold-- == 0 || !strs.append(r.front())) {
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *chars = textstr->getChars(cx);
|
|
|
|
if (!chars)
|
|
|
|
return false;
|
|
|
|
*match = StringMatch(chars, textstrlen, pat, patlen);
|
2010-11-30 18:41:32 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (!r.popFront())
|
|
|
|
return false;
|
|
|
|
}
|
2010-07-22 17:54:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Absolute offset from the beginning of the logical string textstr. */
|
|
|
|
jsint pos = 0;
|
|
|
|
|
|
|
|
// TODO: consider branching to a simple loop if patlen == 1
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
for (JSLinearString **outerp = strs.begin(); outerp != strs.end(); ++outerp) {
|
2010-07-22 17:54:39 -07:00
|
|
|
/* First try to match without spanning two nodes. */
|
2011-03-14 13:59:53 -07:00
|
|
|
JSLinearString *outer = *outerp;
|
|
|
|
const jschar *chars = outer->chars();
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t len = outer->length();
|
2010-07-22 17:54:39 -07:00
|
|
|
jsint matchResult = StringMatch(chars, len, pat, patlen);
|
2010-11-30 18:41:32 -08:00
|
|
|
if (matchResult != -1) {
|
|
|
|
*match = pos + matchResult;
|
|
|
|
return true;
|
|
|
|
}
|
2010-07-22 17:54:39 -07:00
|
|
|
|
|
|
|
/* Test the overlap. */
|
2011-03-14 13:59:53 -07:00
|
|
|
JSLinearString **innerp = outerp;
|
2010-07-22 17:54:39 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Start searching at the first place where StringMatch wouldn't have
|
|
|
|
* found the match.
|
|
|
|
*/
|
|
|
|
const jschar *const text = chars + (patlen > len ? 0 : len - patlen + 1);
|
|
|
|
const jschar *const textend = chars + len;
|
|
|
|
const jschar p0 = *pat;
|
|
|
|
const jschar *const p1 = pat + 1;
|
|
|
|
const jschar *const patend = pat + patlen;
|
|
|
|
for (const jschar *t = text; t != textend; ) {
|
|
|
|
if (*t++ != p0)
|
|
|
|
continue;
|
|
|
|
const jschar *ttend = textend;
|
|
|
|
for (const jschar *pp = p1, *tt = t; pp != patend; ++pp, ++tt) {
|
|
|
|
while (tt == ttend) {
|
2010-11-30 18:41:32 -08:00
|
|
|
if (++innerp == strs.end()) {
|
|
|
|
*match = -1;
|
|
|
|
return true;
|
|
|
|
}
|
2011-03-14 13:59:53 -07:00
|
|
|
JSLinearString *inner = *innerp;
|
|
|
|
tt = inner->chars();
|
2010-12-06 10:26:58 -08:00
|
|
|
ttend = tt + inner->length();
|
2010-07-22 17:54:39 -07:00
|
|
|
}
|
|
|
|
if (*pp != *tt)
|
|
|
|
goto break_continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Matched! */
|
2010-11-30 18:41:32 -08:00
|
|
|
*match = pos + (t - chars) - 1; /* -1 because of *t++ above */
|
|
|
|
return true;
|
2010-07-22 17:54:39 -07:00
|
|
|
|
|
|
|
break_continue:;
|
|
|
|
}
|
|
|
|
|
|
|
|
pos += len;
|
|
|
|
}
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
*match = -1;
|
|
|
|
return true;
|
2010-07-22 17:54:39 -07:00
|
|
|
}
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_indexOf(JSContext *cx, uintN argc, Value *vp)
|
2009-09-23 08:54:50 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2009-09-22 10:19:59 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *patstr = ArgToRootedString(cx, argc, vp, 0);
|
2009-09-23 08:54:50 -07:00
|
|
|
if (!patstr)
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2009-09-22 10:19:59 -07:00
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
jsuint textlen = str->length();
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *text = str->getChars(cx);
|
|
|
|
if (!text)
|
|
|
|
return false;
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
jsuint patlen = patstr->length();
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *pat = patstr->chars();
|
2009-09-23 08:54:50 -07:00
|
|
|
|
|
|
|
jsuint start;
|
|
|
|
if (argc > 1) {
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp[3].isInt32()) {
|
|
|
|
jsint i = vp[3].toInt32();
|
2009-09-23 08:54:50 -07:00
|
|
|
if (i <= 0) {
|
|
|
|
start = 0;
|
|
|
|
} else if (jsuint(i) > textlen) {
|
2010-11-17 03:21:59 -08:00
|
|
|
start = textlen;
|
2009-09-23 08:54:50 -07:00
|
|
|
textlen = 0;
|
|
|
|
} else {
|
|
|
|
start = i;
|
|
|
|
text += start;
|
|
|
|
textlen -= start;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
} else {
|
2010-04-07 13:18:50 -07:00
|
|
|
jsdouble d;
|
2011-04-02 11:33:20 -07:00
|
|
|
if (!ToInteger(cx, vp[3], &d))
|
|
|
|
return false;
|
2009-09-23 08:54:50 -07:00
|
|
|
if (d <= 0) {
|
|
|
|
start = 0;
|
|
|
|
} else if (d > textlen) {
|
2010-11-17 03:21:59 -08:00
|
|
|
start = textlen;
|
2009-09-23 08:54:50 -07:00
|
|
|
textlen = 0;
|
|
|
|
} else {
|
|
|
|
start = (jsint)d;
|
|
|
|
text += start;
|
|
|
|
textlen -= start;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-09-23 08:54:50 -07:00
|
|
|
} else {
|
|
|
|
start = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
jsint match = StringMatch(text, textlen, pat, patlen);
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32((match == -1) ? -1 : start + match);
|
2009-09-23 08:54:50 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_lastIndexOf(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *textstr = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!textstr)
|
|
|
|
return false;
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t textlen = textstr->length();
|
|
|
|
const jschar *text = textstr->getChars(cx);
|
|
|
|
if (!text)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *patstr = ArgToRootedString(cx, argc, vp, 0);
|
|
|
|
if (!patstr)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t patlen = patstr->length();
|
|
|
|
const jschar *pat = patstr->chars();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
jsint i = textlen - patlen; // Start searching here
|
2009-03-23 10:46:16 -07:00
|
|
|
if (i < 0) {
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32(-1);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
2009-03-23 10:46:16 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
if (argc > 1) {
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp[3].isInt32()) {
|
2010-12-06 10:26:58 -08:00
|
|
|
jsint j = vp[3].toInt32();
|
2009-03-23 10:46:16 -07:00
|
|
|
if (j <= 0)
|
2007-03-22 10:30:00 -07:00
|
|
|
i = 0;
|
2009-03-23 10:46:16 -07:00
|
|
|
else if (j < i)
|
|
|
|
i = j;
|
|
|
|
} else {
|
2010-12-06 10:26:58 -08:00
|
|
|
double d;
|
2011-06-13 21:49:59 -07:00
|
|
|
if (!ToNumber(cx, vp[3], &d))
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2009-03-23 10:46:16 -07:00
|
|
|
if (!JSDOUBLE_IS_NaN(d)) {
|
|
|
|
d = js_DoubleToInteger(d);
|
|
|
|
if (d <= 0)
|
|
|
|
i = 0;
|
|
|
|
else if (d < i)
|
|
|
|
i = (jsint)d;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (patlen == 0) {
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32(i);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-11-17 22:40:50 -08:00
|
|
|
const jschar *t = text + i;
|
|
|
|
const jschar *textend = text - 1;
|
|
|
|
const jschar p0 = *pat;
|
|
|
|
const jschar *patNext = pat + 1;
|
|
|
|
const jschar *patEnd = pat + patlen;
|
|
|
|
|
|
|
|
for (; t != textend; --t) {
|
|
|
|
if (*t == p0) {
|
|
|
|
const jschar *t1 = t + 1;
|
|
|
|
for (const jschar *p1 = patNext; p1 != patEnd; ++p1, ++t1) {
|
|
|
|
if (*t1 != *p1)
|
|
|
|
goto break_continue;
|
|
|
|
}
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32(t - text);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-11-17 22:40:50 -08:00
|
|
|
break_continue:;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-11-17 22:40:50 -08:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32(-1);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-09-07 10:22:41 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
js_TrimString(JSContext *cx, Value *vp, JSBool trimLeft, JSBool trimRight)
|
2008-09-07 10:22:41 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length = str->length();
|
|
|
|
const jschar *chars = str->getChars(cx);
|
|
|
|
if (!chars)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
size_t begin = 0;
|
|
|
|
size_t end = length;
|
2008-09-07 10:22:41 -07:00
|
|
|
|
|
|
|
if (trimLeft) {
|
2011-07-26 14:10:33 -07:00
|
|
|
while (begin < length && unicode::IsSpace(chars[begin]))
|
2008-09-07 10:22:41 -07:00
|
|
|
++begin;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trimRight) {
|
2011-07-26 14:10:33 -07:00
|
|
|
while (end > begin && unicode::IsSpace(chars[end - 1]))
|
2008-09-07 10:22:41 -07:00
|
|
|
--end;
|
|
|
|
}
|
|
|
|
|
|
|
|
str = js_NewDependentString(cx, str, begin, end - begin);
|
|
|
|
if (!str)
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2008-09-07 10:22:41 -07:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
2008-09-07 10:22:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_trim(JSContext *cx, uintN argc, Value *vp)
|
2008-09-07 10:22:41 -07:00
|
|
|
{
|
|
|
|
return js_TrimString(cx, vp, JS_TRUE, JS_TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_trimLeft(JSContext *cx, uintN argc, Value *vp)
|
2008-09-07 10:22:41 -07:00
|
|
|
{
|
|
|
|
return js_TrimString(cx, vp, JS_TRUE, JS_FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_trimRight(JSContext *cx, uintN argc, Value *vp)
|
2008-09-07 10:22:41 -07:00
|
|
|
{
|
|
|
|
return js_TrimString(cx, vp, JS_FALSE, JS_TRUE);
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/*
|
|
|
|
* Perl-inspired string functions.
|
|
|
|
*/
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
/* Result of a successfully performed flat match. */
|
|
|
|
class FlatMatch
|
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *patstr;
|
2010-08-24 14:46:19 -07:00
|
|
|
const jschar *pat;
|
|
|
|
size_t patlen;
|
|
|
|
int32 match_;
|
|
|
|
|
|
|
|
friend class RegExpGuard;
|
|
|
|
|
|
|
|
public:
|
2010-09-16 11:46:58 -07:00
|
|
|
FlatMatch() : patstr(NULL) {} /* Old GCC wants this initialization. */
|
2010-08-24 14:46:19 -07:00
|
|
|
JSString *pattern() const { return patstr; }
|
|
|
|
size_t patternLength() const { return patlen; }
|
|
|
|
|
|
|
|
/*
|
2011-01-05 18:44:30 -08:00
|
|
|
* Note: The match is -1 when the match is performed successfully,
|
|
|
|
* but no match is found.
|
2010-08-24 14:46:19 -07:00
|
|
|
*/
|
|
|
|
int32 match() const { return match_; }
|
|
|
|
};
|
|
|
|
|
|
|
|
/* A regexp and optional associated object. */
|
|
|
|
class RegExpPair
|
|
|
|
{
|
2011-01-17 14:58:53 -08:00
|
|
|
AutoRefCount<RegExp> re_;
|
2011-01-05 18:44:30 -08:00
|
|
|
JSObject *reobj_;
|
2010-08-24 14:46:19 -07:00
|
|
|
|
2011-01-05 18:44:30 -08:00
|
|
|
explicit RegExpPair(RegExpPair &);
|
2010-08-24 14:46:19 -07:00
|
|
|
|
|
|
|
public:
|
2011-01-17 14:58:53 -08:00
|
|
|
explicit RegExpPair(JSContext *cx) : re_(cx) {}
|
2011-01-05 18:44:30 -08:00
|
|
|
|
2011-01-17 14:58:53 -08:00
|
|
|
void reset(JSObject &obj) {
|
|
|
|
reobj_ = &obj;
|
|
|
|
RegExp *re = RegExp::extractFrom(reobj_);
|
|
|
|
JS_ASSERT(re);
|
|
|
|
re_.reset(NeedsIncRef<RegExp>(re));
|
2011-01-05 18:44:30 -08:00
|
|
|
}
|
|
|
|
|
2011-01-17 14:58:53 -08:00
|
|
|
void reset(AlreadyIncRefed<RegExp> re) {
|
2011-01-05 18:44:30 -08:00
|
|
|
reobj_ = NULL;
|
2011-01-17 14:58:53 -08:00
|
|
|
re_.reset(re);
|
2011-01-05 18:44:30 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Note: May be null. */
|
2010-08-24 14:46:19 -07:00
|
|
|
JSObject *reobj() const { return reobj_; }
|
2011-01-17 14:58:53 -08:00
|
|
|
bool hasRegExp() const { return !re_.null(); }
|
|
|
|
RegExp &re() const { JS_ASSERT(hasRegExp()); return *re_; }
|
2010-08-24 14:46:19 -07:00
|
|
|
};
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
/*
|
2010-08-24 14:46:19 -07:00
|
|
|
* RegExpGuard factors logic out of String regexp operations.
|
2009-09-23 08:54:50 -07:00
|
|
|
*
|
2010-08-24 14:46:19 -07:00
|
|
|
* @param optarg Indicates in which argument position RegExp
|
|
|
|
* flags will be found, if present. This is a Mozilla
|
|
|
|
* extension and not part of any ECMA spec.
|
2009-09-23 08:54:50 -07:00
|
|
|
*/
|
2009-08-12 13:01:34 -07:00
|
|
|
class RegExpGuard
|
2009-06-25 12:12:19 -07:00
|
|
|
{
|
2009-08-12 13:01:34 -07:00
|
|
|
RegExpGuard(const RegExpGuard &);
|
|
|
|
void operator=(const RegExpGuard &);
|
2009-06-25 12:12:19 -07:00
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
JSContext *cx;
|
|
|
|
RegExpPair rep;
|
|
|
|
FlatMatch fm;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Upper bound on the number of characters we are willing to potentially
|
|
|
|
* waste on searching for RegExp meta-characters.
|
|
|
|
*/
|
|
|
|
static const size_t MAX_FLAT_PAT_LEN = 256;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
static JSString *flattenPattern(JSContext *cx, JSLinearString *patstr) {
|
2011-01-12 15:28:58 -08:00
|
|
|
StringBuffer sb(cx);
|
|
|
|
if (!sb.reserve(patstr->length()))
|
2010-09-22 10:27:38 -07:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
static const jschar ESCAPE_CHAR = '\\';
|
|
|
|
const jschar *chars = patstr->chars();
|
|
|
|
size_t len = patstr->length();
|
|
|
|
for (const jschar *it = chars; it != chars + len; ++it) {
|
|
|
|
if (RegExp::isMetaChar(*it)) {
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(ESCAPE_CHAR) || !sb.append(*it))
|
2010-09-22 10:27:38 -07:00
|
|
|
return NULL;
|
|
|
|
} else {
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(*it))
|
2010-09-22 10:27:38 -07:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
2011-01-12 15:28:58 -08:00
|
|
|
return sb.finishString();
|
2010-09-22 10:27:38 -07:00
|
|
|
}
|
|
|
|
|
2009-08-12 13:01:34 -07:00
|
|
|
public:
|
2011-01-17 14:58:53 -08:00
|
|
|
explicit RegExpGuard(JSContext *cx) : cx(cx), rep(cx) {}
|
2011-01-05 18:44:30 -08:00
|
|
|
~RegExpGuard() {}
|
2009-09-23 08:54:50 -07:00
|
|
|
|
|
|
|
/* init must succeed in order to call tryFlatMatch or normalizeRegExp. */
|
2009-08-12 13:01:34 -07:00
|
|
|
bool
|
2011-07-22 07:46:07 -07:00
|
|
|
init(uintN argc, Value *vp, bool convertVoid = false)
|
2009-08-12 13:01:34 -07:00
|
|
|
{
|
2010-08-24 14:46:19 -07:00
|
|
|
if (argc != 0 && VALUE_IS_REGEXP(cx, vp[2])) {
|
2011-01-05 18:44:30 -08:00
|
|
|
rep.reset(vp[2].toObject());
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
2011-07-22 07:46:07 -07:00
|
|
|
if (convertVoid && (argc == 0 || vp[2].isUndefined())) {
|
|
|
|
fm.patstr = cx->runtime->emptyString;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
fm.patstr = ArgToRootedString(cx, argc, vp, 0);
|
|
|
|
if (!fm.patstr)
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-08-12 13:01:34 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
/*
|
2010-08-24 14:46:19 -07:00
|
|
|
* Attempt to match |patstr| to |textstr|. A flags argument, metachars in the
|
|
|
|
* pattern string, or a lengthy pattern string can thwart this process.
|
|
|
|
*
|
|
|
|
* @param checkMetaChars Look for regexp metachars in the pattern string.
|
|
|
|
* @return Whether flat matching could be used.
|
2010-11-30 18:41:32 -08:00
|
|
|
*
|
2011-01-07 02:03:14 -08:00
|
|
|
* N.B. tryFlatMatch returns NULL on OOM, so the caller must check cx->isExceptionPending().
|
2009-09-23 08:54:50 -07:00
|
|
|
*/
|
2010-08-24 14:46:19 -07:00
|
|
|
const FlatMatch *
|
2010-11-30 18:41:32 -08:00
|
|
|
tryFlatMatch(JSContext *cx, JSString *textstr, uintN optarg, uintN argc,
|
|
|
|
bool checkMetaChars = true)
|
2009-09-23 08:54:50 -07:00
|
|
|
{
|
2011-01-05 18:44:30 -08:00
|
|
|
if (rep.hasRegExp())
|
2010-08-24 14:46:19 -07:00
|
|
|
return NULL;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
fm.pat = fm.patstr->chars();
|
|
|
|
fm.patlen = fm.patstr->length();
|
2010-08-24 14:46:19 -07:00
|
|
|
|
|
|
|
if (optarg < argc)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (checkMetaChars &&
|
|
|
|
(fm.patlen > MAX_FLAT_PAT_LEN || RegExp::hasMetaChars(fm.pat, fm.patlen))) {
|
|
|
|
return NULL;
|
2009-09-23 08:54:50 -07:00
|
|
|
}
|
2010-08-24 14:46:19 -07:00
|
|
|
|
2010-07-22 17:54:39 -07:00
|
|
|
/*
|
|
|
|
* textstr could be a rope, so we want to avoid flattening it for as
|
|
|
|
* long as possible.
|
|
|
|
*/
|
2010-11-30 18:41:32 -08:00
|
|
|
if (textstr->isRope()) {
|
|
|
|
if (!RopeMatch(cx, textstr, fm.pat, fm.patlen, &fm.match_))
|
|
|
|
return NULL;
|
2010-07-22 17:54:39 -07:00
|
|
|
} else {
|
2011-03-14 13:59:53 -07:00
|
|
|
const jschar *text = textstr->asLinear().chars();
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t textlen = textstr->length();
|
2010-08-24 14:46:19 -07:00
|
|
|
fm.match_ = StringMatch(text, textlen, fm.pat, fm.patlen);
|
2010-07-22 17:54:39 -07:00
|
|
|
}
|
2010-08-24 14:46:19 -07:00
|
|
|
return &fm;
|
2009-09-23 08:54:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If the pattern is not already a regular expression, make it so. */
|
2010-08-24 14:46:19 -07:00
|
|
|
const RegExpPair *
|
2010-07-14 23:19:36 -07:00
|
|
|
normalizeRegExp(bool flat, uintN optarg, uintN argc, Value *vp)
|
2009-09-23 08:54:50 -07:00
|
|
|
{
|
2011-01-05 18:44:30 -08:00
|
|
|
if (rep.hasRegExp())
|
2010-08-24 14:46:19 -07:00
|
|
|
return &rep;
|
|
|
|
|
2011-01-05 18:44:30 -08:00
|
|
|
/* Build RegExp from pattern string. */
|
2009-09-23 08:54:50 -07:00
|
|
|
JSString *opt;
|
|
|
|
if (optarg < argc) {
|
2010-08-24 14:46:19 -07:00
|
|
|
opt = js_ValueToString(cx, vp[2 + optarg]);
|
2009-09-23 08:54:50 -07:00
|
|
|
if (!opt)
|
2010-08-24 14:46:19 -07:00
|
|
|
return NULL;
|
2009-09-23 08:54:50 -07:00
|
|
|
} else {
|
|
|
|
opt = NULL;
|
|
|
|
}
|
2010-08-24 14:46:19 -07:00
|
|
|
|
2010-09-22 10:27:38 -07:00
|
|
|
JSString *patstr;
|
|
|
|
if (flat) {
|
|
|
|
patstr = flattenPattern(cx, fm.patstr);
|
|
|
|
if (!patstr)
|
2011-04-14 13:45:06 -07:00
|
|
|
return NULL;
|
2010-09-22 10:27:38 -07:00
|
|
|
} else {
|
|
|
|
patstr = fm.patstr;
|
|
|
|
}
|
|
|
|
JS_ASSERT(patstr);
|
|
|
|
|
2011-06-27 18:53:26 -07:00
|
|
|
AlreadyIncRefed<RegExp> re = RegExp::createFlagged(cx, patstr, opt, NULL);
|
2011-01-05 18:44:30 -08:00
|
|
|
if (!re)
|
2010-08-24 14:46:19 -07:00
|
|
|
return NULL;
|
2011-01-17 14:58:53 -08:00
|
|
|
rep.reset(re);
|
2010-08-24 14:46:19 -07:00
|
|
|
return &rep;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
#if DEBUG
|
2011-01-05 18:44:30 -08:00
|
|
|
bool hasRegExpPair() const { return rep.hasRegExp(); }
|
2010-08-24 14:46:19 -07:00
|
|
|
#endif
|
2009-08-12 13:01:34 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/* js_ExecuteRegExp indicates success in two ways, based on the 'test' flag. */
|
|
|
|
static JS_ALWAYS_INLINE bool
|
2010-07-14 23:19:36 -07:00
|
|
|
Matched(bool test, const Value &v)
|
2009-08-12 13:01:34 -07:00
|
|
|
{
|
2010-07-14 23:19:36 -07:00
|
|
|
return test ? v.isTrue() : !v.isNull();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-09-13 15:53:50 -07:00
|
|
|
typedef bool (*DoMatchCallback)(JSContext *cx, RegExpStatics *res, size_t count, void *data);
|
2009-08-12 13:01:34 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* BitOR-ing these flags allows the DoMatch caller to control when how the
|
|
|
|
* RegExp engine is called and when callbacks are fired.
|
|
|
|
*/
|
|
|
|
enum MatchControlFlags {
|
|
|
|
TEST_GLOBAL_BIT = 0x1, /* use RegExp.test for global regexps */
|
|
|
|
TEST_SINGLE_BIT = 0x2, /* use RegExp.test for non-global regexps */
|
|
|
|
CALLBACK_ON_SINGLE_BIT = 0x4, /* fire callback on non-global match */
|
|
|
|
|
|
|
|
MATCH_ARGS = TEST_GLOBAL_BIT,
|
|
|
|
MATCHALL_ARGS = CALLBACK_ON_SINGLE_BIT,
|
|
|
|
REPLACE_ARGS = TEST_GLOBAL_BIT | TEST_SINGLE_BIT | CALLBACK_ON_SINGLE_BIT
|
2009-08-18 14:13:28 -07:00
|
|
|
};
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-08-12 13:01:34 -07:00
|
|
|
/* Factor out looping and matching logic. */
|
|
|
|
static bool
|
2011-05-13 08:56:26 -07:00
|
|
|
DoMatch(JSContext *cx, RegExpStatics *res, JSString *str, const RegExpPair &rep,
|
|
|
|
DoMatchCallback callback, void *data, MatchControlFlags flags, Value *rval)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-08-24 14:46:19 -07:00
|
|
|
RegExp &re = rep.re();
|
2010-08-11 13:30:07 -07:00
|
|
|
if (re.global()) {
|
2009-08-12 13:01:34 -07:00
|
|
|
/* global matching ('g') */
|
|
|
|
bool testGlobal = flags & TEST_GLOBAL_BIT;
|
2010-08-24 14:46:19 -07:00
|
|
|
if (rep.reobj())
|
|
|
|
rep.reobj()->zeroRegExpLastIndex();
|
2009-08-12 13:01:34 -07:00
|
|
|
for (size_t count = 0, i = 0, length = str->length(); i <= length; ++count) {
|
2011-05-13 08:56:26 -07:00
|
|
|
if (!re.execute(cx, res, str, &i, testGlobal, rval))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2011-05-13 08:56:26 -07:00
|
|
|
if (!Matched(testGlobal, *rval))
|
2009-08-12 13:01:34 -07:00
|
|
|
break;
|
2010-09-13 15:53:50 -07:00
|
|
|
if (!callback(cx, res, count, data))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2010-09-13 15:53:50 -07:00
|
|
|
if (!res->matched())
|
2009-08-12 13:01:34 -07:00
|
|
|
++i;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* single match */
|
2009-09-23 11:16:30 -07:00
|
|
|
bool testSingle = !!(flags & TEST_SINGLE_BIT),
|
|
|
|
callbackOnSingle = !!(flags & CALLBACK_ON_SINGLE_BIT);
|
2009-08-12 13:01:34 -07:00
|
|
|
size_t i = 0;
|
2011-05-13 08:56:26 -07:00
|
|
|
if (!re.execute(cx, res, str, &i, testSingle, rval))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2011-05-13 08:56:26 -07:00
|
|
|
if (callbackOnSingle && Matched(testSingle, *rval) && !callback(cx, res, 0, data))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-08-11 13:30:07 -07:00
|
|
|
static bool
|
2010-08-24 14:46:19 -07:00
|
|
|
BuildFlatMatchArray(JSContext *cx, JSString *textstr, const FlatMatch &fm, Value *vp)
|
2010-08-11 13:30:07 -07:00
|
|
|
{
|
2010-08-24 14:46:19 -07:00
|
|
|
if (fm.match() < 0) {
|
2010-08-11 13:30:07 -07:00
|
|
|
vp->setNull();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For this non-global match, produce a RegExp.exec-style array. */
|
2010-12-13 16:22:59 -08:00
|
|
|
JSObject *obj = NewSlowEmptyArray(cx);
|
2010-08-11 13:30:07 -07:00
|
|
|
if (!obj)
|
|
|
|
return false;
|
|
|
|
vp->setObject(*obj);
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
return obj->defineProperty(cx, INT_TO_JSID(0), StringValue(fm.pattern())) &&
|
2010-08-11 13:30:07 -07:00
|
|
|
obj->defineProperty(cx, ATOM_TO_JSID(cx->runtime->atomState.indexAtom),
|
2010-08-24 14:46:19 -07:00
|
|
|
Int32Value(fm.match())) &&
|
2010-08-11 13:30:07 -07:00
|
|
|
obj->defineProperty(cx, ATOM_TO_JSID(cx->runtime->atomState.inputAtom),
|
|
|
|
StringValue(textstr));
|
|
|
|
}
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
typedef JSObject **MatchArgType;
|
|
|
|
|
2009-08-12 13:01:34 -07:00
|
|
|
/*
|
|
|
|
* DoMatch will only callback on global matches, hence this function builds
|
|
|
|
* only the "array of matches" returned by match on global regexps.
|
|
|
|
*/
|
|
|
|
static bool
|
2010-09-13 15:53:50 -07:00
|
|
|
MatchCallback(JSContext *cx, RegExpStatics *res, size_t count, void *p)
|
2009-08-12 13:01:34 -07:00
|
|
|
{
|
2010-07-14 23:19:36 -07:00
|
|
|
JS_ASSERT(count <= JSID_INT_MAX); /* by max string length */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
JSObject *&arrayobj = *static_cast<MatchArgType>(p);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!arrayobj) {
|
2010-12-13 16:22:59 -08:00
|
|
|
arrayobj = NewDenseEmptyArray(cx);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!arrayobj)
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-08-18 14:13:28 -07:00
|
|
|
|
2010-08-11 13:30:07 -07:00
|
|
|
Value v;
|
2010-09-13 15:53:50 -07:00
|
|
|
if (!res->createLastMatch(cx, &v))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2009-08-18 14:13:28 -07:00
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
JSAutoResolveFlags rf(cx, JSRESOLVE_QUALIFIED | JSRESOLVE_ASSIGNING);
|
2010-09-15 13:43:55 -07:00
|
|
|
return !!arrayobj->setProperty(cx, INT_TO_JSID(count), &v, false);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_match(JSContext *cx, uintN argc, Value *vp)
|
2008-08-15 09:29:03 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2011-07-22 07:46:07 -07:00
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
RegExpGuard g(cx);
|
2011-07-22 07:46:07 -07:00
|
|
|
if (!g.init(argc, vp, true))
|
2009-09-23 08:54:50 -07:00
|
|
|
return false;
|
2010-11-30 18:41:32 -08:00
|
|
|
if (const FlatMatch *fm = g.tryFlatMatch(cx, str, 1, argc))
|
2010-08-24 14:46:19 -07:00
|
|
|
return BuildFlatMatchArray(cx, str, *fm, vp);
|
2011-01-07 02:03:14 -08:00
|
|
|
if (cx->isExceptionPending()) /* from tryFlatMatch */
|
2010-11-30 18:41:32 -08:00
|
|
|
return false;
|
2010-08-24 14:46:19 -07:00
|
|
|
|
|
|
|
const RegExpPair *rep = g.normalizeRegExp(false, 1, argc, vp);
|
|
|
|
if (!rep)
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
AutoObjectRooter array(cx);
|
|
|
|
MatchArgType arg = array.addr();
|
2011-02-19 20:46:44 -08:00
|
|
|
RegExpStatics *res = cx->regExpStatics();
|
2011-05-13 08:56:26 -07:00
|
|
|
Value rval;
|
|
|
|
if (!DoMatch(cx, res, str, *rep, MatchCallback, arg, MATCH_ARGS, &rval))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
if (rep->re().global())
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setObjectOrNull(array.object());
|
2011-05-13 08:56:26 -07:00
|
|
|
else
|
|
|
|
*vp = rval;
|
2009-08-12 13:01:34 -07:00
|
|
|
return true;
|
2008-08-15 09:29:03 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_search(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2009-08-12 13:01:34 -07:00
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
RegExpGuard g(cx);
|
2011-07-22 07:46:07 -07:00
|
|
|
if (!g.init(argc, vp, true))
|
2009-09-23 08:54:50 -07:00
|
|
|
return false;
|
2010-11-30 18:41:32 -08:00
|
|
|
if (const FlatMatch *fm = g.tryFlatMatch(cx, str, 1, argc)) {
|
2010-08-24 14:46:19 -07:00
|
|
|
vp->setInt32(fm->match());
|
2009-09-23 08:54:50 -07:00
|
|
|
return true;
|
|
|
|
}
|
2011-01-07 02:03:14 -08:00
|
|
|
if (cx->isExceptionPending()) /* from tryFlatMatch */
|
2010-11-30 18:41:32 -08:00
|
|
|
return false;
|
2011-07-22 07:46:07 -07:00
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
const RegExpPair *rep = g.normalizeRegExp(false, 1, argc, vp);
|
|
|
|
if (!rep)
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-02-19 20:46:44 -08:00
|
|
|
RegExpStatics *res = cx->regExpStatics();
|
2009-08-12 13:01:34 -07:00
|
|
|
size_t i = 0;
|
2010-09-13 15:53:50 -07:00
|
|
|
if (!rep->re().execute(cx, res, str, &i, true, vp))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
if (vp->isTrue())
|
2010-11-10 17:02:08 -08:00
|
|
|
vp->setInt32(res->matchStart());
|
2009-09-23 08:54:50 -07:00
|
|
|
else
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setInt32(-1);
|
2009-08-12 13:01:34 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-10-28 09:53:33 -07:00
|
|
|
struct ReplaceData
|
|
|
|
{
|
|
|
|
ReplaceData(JSContext *cx)
|
2011-01-12 15:28:58 -08:00
|
|
|
: g(cx), sb(cx)
|
2009-10-28 09:53:33 -07:00
|
|
|
{}
|
|
|
|
|
2010-09-28 15:23:43 -07:00
|
|
|
JSString *str; /* 'this' parameter object as a string */
|
|
|
|
RegExpGuard g; /* regexp parameter object and private data */
|
|
|
|
JSObject *lambda; /* replacement function object or null */
|
2010-10-28 14:33:32 -07:00
|
|
|
JSObject *elembase; /* object for function(a){return b[a]} replace */
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *repstr; /* replacement string */
|
|
|
|
const jschar *dollar; /* null or pointer to first $ in repstr */
|
|
|
|
const jschar *dollarEnd; /* limit pointer for js_strchr_limit */
|
2010-09-28 15:23:43 -07:00
|
|
|
jsint leftIndex; /* left context index in str->chars */
|
|
|
|
JSSubString dollarStr; /* for "$$" InterpretDollar result */
|
|
|
|
bool calledBack; /* record whether callback has been called */
|
|
|
|
InvokeSessionGuard session; /* arguments for repeated lambda Invoke call */
|
|
|
|
InvokeArgsGuard singleShot; /* arguments for single lambda Invoke call */
|
2011-01-12 15:28:58 -08:00
|
|
|
StringBuffer sb; /* buffer built during DoMatch */
|
2009-08-18 14:13:28 -07:00
|
|
|
};
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-11 13:30:07 -07:00
|
|
|
static bool
|
2010-12-06 10:26:58 -08:00
|
|
|
InterpretDollar(JSContext *cx, RegExpStatics *res, const jschar *dp, const jschar *ep,
|
2010-12-29 16:25:04 -08:00
|
|
|
ReplaceData &rdata, JSSubString *out, size_t *skip)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
JS_ASSERT(*dp == '$');
|
|
|
|
|
|
|
|
/* If there is only a dollar, bail now */
|
|
|
|
if (dp + 1 >= ep)
|
2010-08-11 13:30:07 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/* Interpret all Perl match-induced dollar variables. */
|
2010-08-11 13:30:07 -07:00
|
|
|
jschar dc = dp[1];
|
2007-03-22 10:30:00 -07:00
|
|
|
if (JS7_ISDEC(dc)) {
|
|
|
|
/* ECMA-262 Edition 3: 1-9 or 01-99 */
|
2010-08-11 13:30:07 -07:00
|
|
|
uintN num = JS7_UNDEC(dc);
|
2010-12-01 16:34:10 -08:00
|
|
|
if (num > res->parenCount())
|
2010-08-11 13:30:07 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *cp = dp + 2;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (cp < ep && (dc = *cp, JS7_ISDEC(dc))) {
|
2010-08-11 13:30:07 -07:00
|
|
|
uintN tmp = 10 * num + JS7_UNDEC(dc);
|
2010-12-01 16:34:10 -08:00
|
|
|
if (tmp <= res->parenCount()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
cp++;
|
|
|
|
num = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (num == 0)
|
2010-08-11 13:30:07 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
*skip = cp - dp;
|
2010-12-01 16:34:10 -08:00
|
|
|
|
2010-12-06 13:12:58 -08:00
|
|
|
JS_ASSERT(num <= res->parenCount());
|
2010-12-01 16:34:42 -08:00
|
|
|
|
2010-12-01 16:34:10 -08:00
|
|
|
/*
|
|
|
|
* Note: we index to get the paren with the (1-indexed) pair
|
|
|
|
* number, as opposed to a (0-indexed) paren number.
|
|
|
|
*/
|
|
|
|
res->getParen(num, out);
|
2010-08-11 13:30:07 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
*skip = 2;
|
|
|
|
switch (dc) {
|
|
|
|
case '$':
|
2009-08-12 13:01:34 -07:00
|
|
|
rdata.dollarStr.chars = dp;
|
|
|
|
rdata.dollarStr.length = 1;
|
2010-08-11 13:30:07 -07:00
|
|
|
*out = rdata.dollarStr;
|
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
case '&':
|
2010-09-13 15:53:50 -07:00
|
|
|
res->getLastMatch(out);
|
2010-08-11 13:30:07 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
case '+':
|
2010-09-13 15:53:50 -07:00
|
|
|
res->getLastParen(out);
|
2010-08-11 13:30:07 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
case '`':
|
2010-09-13 15:53:50 -07:00
|
|
|
res->getLeftContext(out);
|
2010-08-11 13:30:07 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
case '\'':
|
2010-09-13 15:53:50 -07:00
|
|
|
res->getRightContext(out);
|
2010-08-11 13:30:07 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-11 13:30:07 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-08-18 14:13:28 -07:00
|
|
|
static bool
|
2010-09-13 15:53:50 -07:00
|
|
|
FindReplaceLength(JSContext *cx, RegExpStatics *res, ReplaceData &rdata, size_t *sizep)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-10-28 14:33:32 -07:00
|
|
|
JSObject *base = rdata.elembase;
|
|
|
|
if (base) {
|
|
|
|
/*
|
|
|
|
* The base object is used when replace was passed a lambda which looks like
|
|
|
|
* 'function(a) { return b[a]; }' for the base object b. b will not change
|
|
|
|
* in the course of the replace unless we end up making a scripted call due
|
|
|
|
* to accessing a scripted getter or a value with a scripted toString.
|
|
|
|
*/
|
|
|
|
JS_ASSERT(rdata.lambda);
|
|
|
|
JS_ASSERT(!base->getOps()->lookupProperty);
|
|
|
|
JS_ASSERT(!base->getOps()->getProperty);
|
|
|
|
|
|
|
|
Value match;
|
|
|
|
if (!res->createLastMatch(cx, &match))
|
|
|
|
return false;
|
|
|
|
JSString *str = match.toString();
|
|
|
|
|
|
|
|
JSAtom *atom;
|
2011-03-14 13:55:55 -07:00
|
|
|
if (str->isAtom()) {
|
2011-03-14 13:59:53 -07:00
|
|
|
atom = &str->asAtom();
|
2010-10-28 14:33:32 -07:00
|
|
|
} else {
|
2011-05-17 12:15:12 -07:00
|
|
|
atom = js_AtomizeString(cx, str);
|
2010-10-28 14:33:32 -07:00
|
|
|
if (!atom)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
jsid id = ATOM_TO_JSID(atom);
|
|
|
|
|
|
|
|
JSObject *holder;
|
|
|
|
JSProperty *prop = NULL;
|
2011-01-21 05:10:16 -08:00
|
|
|
if (!LookupPropertyWithFlags(cx, base, id, JSRESOLVE_QUALIFIED, &holder, &prop))
|
2010-10-28 14:33:32 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Only handle the case where the property exists and is on this object. */
|
|
|
|
if (prop && holder == base) {
|
|
|
|
Shape *shape = (Shape *) prop;
|
|
|
|
if (shape->slot != SHAPE_INVALID_SLOT && shape->hasDefaultGetter()) {
|
|
|
|
Value value = base->getSlot(shape->slot);
|
|
|
|
if (value.isString()) {
|
2010-12-06 10:26:58 -08:00
|
|
|
rdata.repstr = value.toString()->ensureLinear(cx);
|
|
|
|
if (!rdata.repstr)
|
|
|
|
return false;
|
2010-10-28 14:33:32 -07:00
|
|
|
*sizep = rdata.repstr->length();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Couldn't handle this property, fall through and despecialize to the
|
|
|
|
* general lambda case.
|
|
|
|
*/
|
|
|
|
rdata.elembase = NULL;
|
|
|
|
}
|
|
|
|
|
2010-08-11 13:30:07 -07:00
|
|
|
JSObject *lambda = rdata.lambda;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (lambda) {
|
|
|
|
/*
|
|
|
|
* 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
|
2009-08-12 13:01:34 -07:00
|
|
|
* DoReplace. The lambda is called with arguments ($&, $1, $2, ...,
|
2007-03-22 10:30:00 -07:00
|
|
|
* 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.
|
|
|
|
*/
|
2010-12-01 16:34:10 -08:00
|
|
|
uintN p = res->parenCount();
|
2009-08-18 14:13:28 -07:00
|
|
|
uintN argc = 1 + p + 2;
|
2009-10-28 09:53:33 -07:00
|
|
|
|
2010-10-04 11:45:34 -07:00
|
|
|
InvokeSessionGuard &session = rdata.session;
|
|
|
|
if (!session.started()) {
|
2010-09-28 15:23:43 -07:00
|
|
|
Value lambdav = ObjectValue(*lambda);
|
2010-10-12 11:50:03 -07:00
|
|
|
if (!session.start(cx, lambdav, UndefinedValue(), argc))
|
2010-09-28 15:23:43 -07:00
|
|
|
return false;
|
|
|
|
}
|
2009-08-18 14:13:28 -07:00
|
|
|
|
2010-10-04 11:45:34 -07:00
|
|
|
PreserveRegExpStatics staticsGuard(res);
|
|
|
|
if (!staticsGuard.init(cx))
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/* Push $&, $1, $2, ... */
|
2010-10-04 11:45:34 -07:00
|
|
|
uintN argi = 0;
|
2010-09-28 15:23:43 -07:00
|
|
|
if (!res->createLastMatch(cx, &session[argi++]))
|
2010-04-28 16:35:12 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-01 16:34:10 -08:00
|
|
|
for (size_t i = 0; i < res->parenCount(); ++i) {
|
|
|
|
if (!res->createParen(cx, i + 1, &session[argi++]))
|
2010-04-28 16:35:12 -07:00
|
|
|
return false;
|
|
|
|
}
|
2010-04-28 15:18:38 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/* Push match index and input string. */
|
2010-11-10 17:02:08 -08:00
|
|
|
session[argi++].setInt32(res->matchStart());
|
2010-09-28 15:23:43 -07:00
|
|
|
session[argi].setString(rdata.str);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-09-28 15:23:43 -07:00
|
|
|
if (!session.invoke(cx))
|
2010-04-28 16:35:12 -07:00
|
|
|
return false;
|
2009-08-18 14:13:28 -07:00
|
|
|
|
2010-10-04 11:45:34 -07:00
|
|
|
/* root repstr: rdata is on the stack, so scanned by conservative gc. */
|
2010-12-06 10:26:58 -08:00
|
|
|
JSString *repstr = ValueToString_TestForStringInline(cx, session.rval());
|
|
|
|
if (!repstr)
|
|
|
|
return false;
|
|
|
|
rdata.repstr = repstr->ensureLinear(cx);
|
2010-10-04 11:45:34 -07:00
|
|
|
if (!rdata.repstr)
|
2010-04-28 16:35:12 -07:00
|
|
|
return false;
|
2010-10-04 11:45:34 -07:00
|
|
|
*sizep = rdata.repstr->length();
|
2010-04-28 16:35:12 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-08-11 13:30:07 -07:00
|
|
|
JSString *repstr = rdata.repstr;
|
|
|
|
size_t replen = repstr->length();
|
2010-12-29 16:25:04 -08:00
|
|
|
for (const jschar *dp = rdata.dollar, *ep = rdata.dollarEnd; dp;
|
|
|
|
dp = js_strchr_limit(dp, '$', ep)) {
|
2010-08-11 13:30:07 -07:00
|
|
|
JSSubString sub;
|
|
|
|
size_t skip;
|
2010-12-06 13:12:58 -08:00
|
|
|
if (InterpretDollar(cx, res, dp, ep, rdata, &sub, &skip)) {
|
2010-08-11 13:30:07 -07:00
|
|
|
replen += sub.length - skip;
|
2007-03-22 10:30:00 -07:00
|
|
|
dp += skip;
|
2010-08-11 13:30:07 -07:00
|
|
|
} else {
|
2007-03-22 10:30:00 -07:00
|
|
|
dp++;
|
2010-08-11 13:30:07 -07:00
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
*sizep = replen;
|
2009-08-12 13:01:34 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
/*
|
|
|
|
* Precondition: |rdata.sb| already has necessary growth space reserved (as
|
|
|
|
* derived from FindReplaceLength).
|
|
|
|
*/
|
2007-03-22 10:30:00 -07:00
|
|
|
static void
|
2011-01-12 15:28:58 -08:00
|
|
|
DoReplace(JSContext *cx, RegExpStatics *res, ReplaceData &rdata)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *repstr = rdata.repstr;
|
|
|
|
const jschar *cp;
|
|
|
|
const jschar *bp = cp = repstr->chars();
|
2010-11-15 19:40:26 -08:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *dp = rdata.dollar;
|
|
|
|
const jschar *ep = rdata.dollarEnd;
|
|
|
|
for (; dp; dp = js_strchr_limit(dp, '$', ep)) {
|
2011-01-12 15:28:58 -08:00
|
|
|
/* Move one of the constant portions of the replacement value. */
|
2010-08-11 13:30:07 -07:00
|
|
|
size_t len = dp - cp;
|
2011-02-18 18:32:29 -08:00
|
|
|
rdata.sb.infallibleAppend(cp, len);
|
2007-03-22 10:30:00 -07:00
|
|
|
cp = dp;
|
2010-08-11 13:30:07 -07:00
|
|
|
|
|
|
|
JSSubString sub;
|
|
|
|
size_t skip;
|
2010-12-06 13:12:58 -08:00
|
|
|
if (InterpretDollar(cx, res, dp, ep, rdata, &sub, &skip)) {
|
2010-08-11 13:30:07 -07:00
|
|
|
len = sub.length;
|
2011-02-18 18:32:29 -08:00
|
|
|
rdata.sb.infallibleAppend(sub.chars, len);
|
2007-03-22 10:30:00 -07:00
|
|
|
cp += skip;
|
|
|
|
dp += skip;
|
|
|
|
} else {
|
|
|
|
dp++;
|
|
|
|
}
|
|
|
|
}
|
2011-01-12 15:28:58 -08:00
|
|
|
JS_ALWAYS_TRUE(rdata.sb.append(cp, repstr->length() - (cp - bp)));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-08-12 13:01:34 -07:00
|
|
|
static bool
|
2010-12-06 10:26:58 -08:00
|
|
|
ReplaceRegExpCallback(JSContext *cx, RegExpStatics *res, size_t count, void *p)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-08-12 13:01:34 -07:00
|
|
|
ReplaceData &rdata = *static_cast<ReplaceData *>(p);
|
|
|
|
|
|
|
|
rdata.calledBack = true;
|
2011-03-14 13:59:53 -07:00
|
|
|
JSLinearString &str = rdata.str->asLinear(); /* flattened for regexp */
|
2009-08-12 13:01:34 -07:00
|
|
|
size_t leftoff = rdata.leftIndex;
|
2011-03-14 13:59:53 -07:00
|
|
|
const jschar *left = str.chars() + leftoff;
|
2010-11-10 17:02:08 -08:00
|
|
|
size_t leftlen = res->matchStart() - leftoff;
|
|
|
|
rdata.leftIndex = res->matchLimit();
|
2009-08-12 13:01:34 -07:00
|
|
|
|
|
|
|
size_t replen = 0; /* silence 'unused' warning */
|
2010-09-13 15:53:50 -07:00
|
|
|
if (!FindReplaceLength(cx, res, rdata, &replen))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-08-12 13:01:34 -07:00
|
|
|
size_t growth = leftlen + replen;
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!rdata.sb.reserve(rdata.sb.length() + growth))
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2011-02-18 18:32:29 -08:00
|
|
|
rdata.sb.infallibleAppend(left, leftlen); /* skipped-over portion of the search value */
|
2011-01-12 15:28:58 -08:00
|
|
|
DoReplace(cx, res, rdata);
|
2009-08-12 13:01:34 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
static bool
|
|
|
|
BuildFlatReplacement(JSContext *cx, JSString *textstr, JSString *repstr,
|
2010-08-24 14:46:19 -07:00
|
|
|
const FlatMatch &fm, Value *vp)
|
2009-09-23 08:54:50 -07:00
|
|
|
{
|
2010-11-30 18:41:32 -08:00
|
|
|
RopeBuilder builder(cx);
|
|
|
|
size_t match = fm.match();
|
2010-08-24 14:46:19 -07:00
|
|
|
size_t matchEnd = match + fm.patternLength();
|
2009-09-23 08:54:50 -07:00
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
if (textstr->isRope()) {
|
2010-07-22 17:54:39 -07:00
|
|
|
/*
|
|
|
|
* If we are replacing over a rope, avoid flattening it by iterating
|
|
|
|
* through it, building a new rope.
|
|
|
|
*/
|
2010-11-30 18:41:32 -08:00
|
|
|
StringSegmentRange r(cx);
|
|
|
|
if (!r.init(textstr))
|
|
|
|
return false;
|
2010-07-22 17:54:39 -07:00
|
|
|
size_t pos = 0;
|
2010-11-30 18:41:32 -08:00
|
|
|
while (!r.empty()) {
|
|
|
|
JSString *str = r.front();
|
2010-07-22 17:54:39 -07:00
|
|
|
size_t len = str->length();
|
|
|
|
size_t strEnd = pos + len;
|
|
|
|
if (pos < matchEnd && strEnd > match) {
|
|
|
|
/*
|
|
|
|
* We need to special-case any part of the rope that overlaps
|
|
|
|
* with the replacement string.
|
|
|
|
*/
|
|
|
|
if (match >= pos) {
|
|
|
|
/*
|
|
|
|
* If this part of the rope overlaps with the left side of
|
|
|
|
* the pattern, then it must be the only one to overlap with
|
|
|
|
* the first character in the pattern, so we include the
|
|
|
|
* replacement string here.
|
|
|
|
*/
|
|
|
|
JSString *leftSide = js_NewDependentString(cx, str, 0, match - pos);
|
|
|
|
if (!leftSide ||
|
2010-08-24 14:46:19 -07:00
|
|
|
!builder.append(leftSide) ||
|
|
|
|
!builder.append(repstr)) {
|
2010-07-22 17:54:39 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If str runs off the end of the matched string, append the
|
|
|
|
* last part of str.
|
|
|
|
*/
|
|
|
|
if (strEnd > matchEnd) {
|
|
|
|
JSString *rightSide = js_NewDependentString(cx, str, matchEnd - pos,
|
|
|
|
strEnd - matchEnd);
|
2010-08-24 14:46:19 -07:00
|
|
|
if (!rightSide || !builder.append(rightSide))
|
2010-07-22 17:54:39 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2010-08-24 14:46:19 -07:00
|
|
|
if (!builder.append(str))
|
2010-07-22 17:54:39 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
pos += str->length();
|
2010-11-30 18:41:32 -08:00
|
|
|
if (!r.popFront())
|
|
|
|
return false;
|
2010-07-22 17:54:39 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
JSString *leftSide = js_NewDependentString(cx, textstr, 0, match);
|
|
|
|
if (!leftSide)
|
|
|
|
return false;
|
2010-08-24 14:46:19 -07:00
|
|
|
JSString *rightSide = js_NewDependentString(cx, textstr, match + fm.patternLength(),
|
|
|
|
textstr->length() - match - fm.patternLength());
|
2010-07-22 17:54:39 -07:00
|
|
|
if (!rightSide ||
|
2010-08-24 14:46:19 -07:00
|
|
|
!builder.append(leftSide) ||
|
|
|
|
!builder.append(repstr) ||
|
|
|
|
!builder.append(rightSide)) {
|
2010-07-22 17:54:39 -07:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-23 08:54:50 -07:00
|
|
|
}
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
vp->setString(builder.result());
|
2009-09-23 08:54:50 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
/*
|
|
|
|
* Perform a linear-scan dollar substitution on the replacement text,
|
|
|
|
* constructing a result string that looks like:
|
|
|
|
*
|
|
|
|
* newstring = string[:matchStart] + dollarSub(replaceValue) + string[matchLimit:]
|
|
|
|
*/
|
|
|
|
static inline bool
|
2010-12-06 10:26:58 -08:00
|
|
|
BuildDollarReplacement(JSContext *cx, JSString *textstrArg, JSLinearString *repstr,
|
2010-08-24 14:46:19 -07:00
|
|
|
const jschar *firstDollar, const FlatMatch &fm, Value *vp)
|
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *textstr = textstrArg->ensureLinear(cx);
|
|
|
|
if (!textstr)
|
|
|
|
return NULL;
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
JS_ASSERT(repstr->chars() <= firstDollar && firstDollar < repstr->chars() + repstr->length());
|
|
|
|
size_t matchStart = fm.match();
|
|
|
|
size_t matchLimit = matchStart + fm.patternLength();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Most probably:
|
|
|
|
*
|
|
|
|
* len(newstr) >= len(orig) - len(match) + len(replacement)
|
|
|
|
*
|
|
|
|
* Note that dollar vars _could_ make the resulting text smaller than this.
|
|
|
|
*/
|
2011-01-12 15:28:58 -08:00
|
|
|
StringBuffer newReplaceChars(cx);
|
2010-08-24 14:46:19 -07:00
|
|
|
if (!newReplaceChars.reserve(textstr->length() - fm.patternLength() + repstr->length()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Move the pre-dollar chunk in bulk. */
|
2011-02-18 18:32:29 -08:00
|
|
|
newReplaceChars.infallibleAppend(repstr->chars(), firstDollar);
|
2010-08-24 14:46:19 -07:00
|
|
|
|
|
|
|
/* Move the rest char-by-char, interpreting dollars as we encounter them. */
|
|
|
|
#define ENSURE(__cond) if (!(__cond)) return false;
|
|
|
|
const jschar *repstrLimit = repstr->chars() + repstr->length();
|
|
|
|
for (const jschar *it = firstDollar; it < repstrLimit; ++it) {
|
|
|
|
if (*it != '$' || it == repstrLimit - 1) {
|
|
|
|
ENSURE(newReplaceChars.append(*it));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (*(it + 1)) {
|
|
|
|
case '$': /* Eat one of the dollars. */
|
|
|
|
ENSURE(newReplaceChars.append(*it));
|
|
|
|
break;
|
|
|
|
case '&':
|
|
|
|
ENSURE(newReplaceChars.append(textstr->chars() + matchStart,
|
|
|
|
textstr->chars() + matchLimit));
|
|
|
|
break;
|
|
|
|
case '`':
|
|
|
|
ENSURE(newReplaceChars.append(textstr->chars(), textstr->chars() + matchStart));
|
|
|
|
break;
|
|
|
|
case '\'':
|
|
|
|
ENSURE(newReplaceChars.append(textstr->chars() + matchLimit,
|
|
|
|
textstr->chars() + textstr->length()));
|
|
|
|
break;
|
|
|
|
default: /* The dollar we saw was not special (no matter what its mother told it). */
|
|
|
|
ENSURE(newReplaceChars.append(*it));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
++it; /* We always eat an extra char in the above switch. */
|
|
|
|
}
|
|
|
|
|
|
|
|
JSString *leftSide = js_NewDependentString(cx, textstr, 0, matchStart);
|
|
|
|
ENSURE(leftSide);
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
JSString *newReplace = newReplaceChars.finishString();
|
2010-08-24 14:46:19 -07:00
|
|
|
ENSURE(newReplace);
|
|
|
|
|
|
|
|
JS_ASSERT(textstr->length() >= matchLimit);
|
|
|
|
JSString *rightSide = js_NewDependentString(cx, textstr, matchLimit,
|
|
|
|
textstr->length() - matchLimit);
|
|
|
|
ENSURE(rightSide);
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
RopeBuilder builder(cx);
|
2010-08-24 14:46:19 -07:00
|
|
|
ENSURE(builder.append(leftSide) &&
|
|
|
|
builder.append(newReplace) &&
|
|
|
|
builder.append(rightSide));
|
|
|
|
#undef ENSURE
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
vp->setString(builder.result());
|
2010-08-24 14:46:19 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
str_replace_regexp(JSContext *cx, uintN argc, Value *vp, ReplaceData &rdata)
|
|
|
|
{
|
|
|
|
const RegExpPair *rep = rdata.g.normalizeRegExp(true, 2, argc, vp);
|
|
|
|
if (!rep)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
rdata.leftIndex = 0;
|
|
|
|
rdata.calledBack = false;
|
|
|
|
|
2011-02-19 20:46:44 -08:00
|
|
|
RegExpStatics *res = cx->regExpStatics();
|
2011-05-13 08:56:26 -07:00
|
|
|
Value tmp;
|
|
|
|
if (!DoMatch(cx, res, rdata.str, *rep, ReplaceRegExpCallback, &rdata, REPLACE_ARGS, &tmp))
|
2010-08-24 14:46:19 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!rdata.calledBack) {
|
|
|
|
/* Didn't match, so the string is unmodified. */
|
|
|
|
vp->setString(rdata.str);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSSubString sub;
|
2010-09-13 15:53:50 -07:00
|
|
|
res->getRightContext(&sub);
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!rdata.sb.append(sub.chars, sub.length))
|
2010-08-24 14:46:19 -07:00
|
|
|
return false;
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
JSString *retstr = rdata.sb.finishString();
|
2010-08-24 14:46:19 -07:00
|
|
|
if (!retstr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
vp->setString(retstr);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
str_replace_flat_lambda(JSContext *cx, uintN argc, Value *vp, ReplaceData &rdata,
|
|
|
|
const FlatMatch &fm)
|
|
|
|
{
|
|
|
|
JS_ASSERT(fm.match() >= 0);
|
|
|
|
LeaveTrace(cx);
|
|
|
|
|
|
|
|
JSString *matchStr = js_NewDependentString(cx, rdata.str, fm.match(), fm.patternLength());
|
|
|
|
if (!matchStr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* lambda(matchStr, matchStart, textstr) */
|
|
|
|
static const uint32 lambdaArgc = 3;
|
2011-04-13 09:27:37 -07:00
|
|
|
if (!cx->stack.pushInvokeArgs(cx, lambdaArgc, &rdata.singleShot))
|
2010-08-24 14:46:19 -07:00
|
|
|
return false;
|
|
|
|
|
2010-09-28 15:23:43 -07:00
|
|
|
CallArgs &args = rdata.singleShot;
|
2011-04-08 10:52:48 -07:00
|
|
|
args.calleev().setObject(*rdata.lambda);
|
2010-10-12 11:50:03 -07:00
|
|
|
args.thisv().setUndefined();
|
2010-08-24 14:46:19 -07:00
|
|
|
|
|
|
|
Value *sp = args.argv();
|
|
|
|
sp[0].setString(matchStr);
|
|
|
|
sp[1].setInt32(fm.match());
|
|
|
|
sp[2].setString(rdata.str);
|
|
|
|
|
2011-04-13 09:27:37 -07:00
|
|
|
if (!Invoke(cx, rdata.singleShot))
|
2010-08-24 14:46:19 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
JSString *repstr = js_ValueToString(cx, args.rval());
|
|
|
|
if (!repstr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
JSString *leftSide = js_NewDependentString(cx, rdata.str, 0, fm.match());
|
|
|
|
if (!leftSide)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
size_t matchLimit = fm.match() + fm.patternLength();
|
|
|
|
JSString *rightSide = js_NewDependentString(cx, rdata.str, matchLimit,
|
|
|
|
rdata.str->length() - matchLimit);
|
|
|
|
if (!rightSide)
|
|
|
|
return false;
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
RopeBuilder builder(cx);
|
2010-08-24 14:46:19 -07:00
|
|
|
if (!(builder.append(leftSide) &&
|
|
|
|
builder.append(repstr) &&
|
|
|
|
builder.append(rightSide))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
vp->setString(builder.result());
|
2010-08-24 14:46:19 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-07-23 14:41:56 -07:00
|
|
|
JSBool
|
|
|
|
js::str_replace(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-08-12 13:01:34 -07:00
|
|
|
ReplaceData rdata(cx);
|
2011-01-26 13:37:45 -08:00
|
|
|
rdata.str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!rdata.str)
|
|
|
|
return false;
|
2010-09-22 10:27:38 -07:00
|
|
|
static const uint32 optarg = 2;
|
2008-08-12 18:52:28 -07:00
|
|
|
|
2011-08-05 19:28:31 -07:00
|
|
|
if (!rdata.g.init(argc, vp))
|
|
|
|
return false;
|
|
|
|
|
2009-09-23 08:54:50 -07:00
|
|
|
/* Extract replacement string/function. */
|
2010-09-22 10:27:38 -07:00
|
|
|
if (argc >= optarg && js_IsCallable(vp[3])) {
|
2010-07-14 23:19:36 -07:00
|
|
|
rdata.lambda = &vp[3].toObject();
|
2010-10-28 14:33:32 -07:00
|
|
|
rdata.elembase = NULL;
|
2009-08-12 13:01:34 -07:00
|
|
|
rdata.repstr = NULL;
|
|
|
|
rdata.dollar = rdata.dollarEnd = NULL;
|
2010-10-28 14:33:32 -07:00
|
|
|
|
|
|
|
if (rdata.lambda->isFunction()) {
|
|
|
|
JSFunction *fun = rdata.lambda->getFunctionPrivate();
|
|
|
|
if (fun->isInterpreted()) {
|
|
|
|
/*
|
|
|
|
* Pattern match the script to check if it is is indexing into a
|
|
|
|
* particular object, e.g. 'function(a) { return b[a]; }'. Avoid
|
|
|
|
* calling the script in such cases, which are used by javascript
|
|
|
|
* packers (particularly the popular Dean Edwards packer) to efficiently
|
|
|
|
* encode large scripts. We only handle the code patterns generated
|
|
|
|
* by such packers here.
|
|
|
|
*/
|
2011-05-20 11:27:27 -07:00
|
|
|
JSScript *script = fun->script();
|
2010-10-28 14:33:32 -07:00
|
|
|
jsbytecode *pc = script->code;
|
|
|
|
|
|
|
|
Value table = UndefinedValue();
|
|
|
|
if (JSOp(*pc) == JSOP_GETFCSLOT) {
|
|
|
|
table = rdata.lambda->getFlatClosureUpvar(GET_UINT16(pc));
|
|
|
|
pc += JSOP_GETFCSLOT_LENGTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (table.isObject() &&
|
|
|
|
JSOp(*pc) == JSOP_GETARG && GET_SLOTNO(pc) == 0 &&
|
|
|
|
JSOp(*(pc + JSOP_GETARG_LENGTH)) == JSOP_GETELEM &&
|
|
|
|
JSOp(*(pc + JSOP_GETARG_LENGTH + JSOP_GETELEM_LENGTH)) == JSOP_RETURN) {
|
|
|
|
Class *clasp = table.toObject().getClass();
|
|
|
|
if (clasp->isNative() &&
|
|
|
|
!clasp->ops.lookupProperty &&
|
|
|
|
!clasp->ops.getProperty) {
|
|
|
|
rdata.elembase = &table.toObject();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-08-12 13:01:34 -07:00
|
|
|
} else {
|
|
|
|
rdata.lambda = NULL;
|
2010-10-28 14:33:32 -07:00
|
|
|
rdata.elembase = NULL;
|
2009-08-12 13:01:34 -07:00
|
|
|
rdata.repstr = ArgToRootedString(cx, argc, vp, 1);
|
|
|
|
if (!rdata.repstr)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-07-23 14:41:19 -07:00
|
|
|
/* We're about to store pointers into the middle of our string. */
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *fixed = rdata.repstr->ensureFixed(cx);
|
|
|
|
if (!fixed)
|
2009-08-12 13:01:34 -07:00
|
|
|
return false;
|
2011-03-14 13:59:53 -07:00
|
|
|
rdata.dollarEnd = fixed->chars() + fixed->length();
|
|
|
|
rdata.dollar = js_strchr_limit(fixed->chars(), '$', rdata.dollarEnd);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-08-12 13:01:34 -07:00
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
/*
|
|
|
|
* Unlike its |String.prototype| brethren, |replace| doesn't convert
|
|
|
|
* its input to a regular expression. (Even if it contains metachars.)
|
|
|
|
*
|
|
|
|
* However, if the user invokes our (non-standard) |flags| argument
|
|
|
|
* extension then we revert to creating a regular expression. Note that
|
|
|
|
* this is observable behavior through the side-effect mutation of the
|
|
|
|
* |RegExp| statics.
|
|
|
|
*/
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
const FlatMatch *fm = rdata.g.tryFlatMatch(cx, rdata.str, optarg, argc, false);
|
2010-08-24 14:46:19 -07:00
|
|
|
if (!fm) {
|
2011-01-07 02:03:14 -08:00
|
|
|
if (cx->isExceptionPending()) /* oom in RopeMatch in tryFlatMatch */
|
2010-11-30 18:41:32 -08:00
|
|
|
return false;
|
2010-09-22 10:27:38 -07:00
|
|
|
JS_ASSERT_IF(!rdata.g.hasRegExpPair(), argc > optarg);
|
2010-08-24 14:46:19 -07:00
|
|
|
return str_replace_regexp(cx, argc, vp, rdata);
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
if (fm->match() < 0) {
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(rdata.str);
|
2009-08-12 13:01:34 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
if (rdata.lambda)
|
|
|
|
return str_replace_flat_lambda(cx, argc, vp, rdata, *fm);
|
2009-08-11 15:45:32 -07:00
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
/*
|
|
|
|
* Note: we could optimize the text.length == pattern.length case if we wanted,
|
|
|
|
* even in the presence of dollar metachars.
|
|
|
|
*/
|
|
|
|
if (rdata.dollar)
|
|
|
|
return BuildDollarReplacement(cx, rdata.str, rdata.repstr, rdata.dollar, *fm, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-24 14:46:19 -07:00
|
|
|
return BuildFlatReplacement(cx, rdata.str, rdata.repstr, *fm, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
class SplitMatchResult {
|
|
|
|
size_t endIndex_;
|
|
|
|
size_t length_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
void setFailure() {
|
|
|
|
JS_STATIC_ASSERT(SIZE_MAX > JSString::MAX_LENGTH);
|
|
|
|
endIndex_ = SIZE_MAX;
|
|
|
|
}
|
|
|
|
bool isFailure() const {
|
|
|
|
return (endIndex_ == SIZE_MAX);
|
|
|
|
}
|
|
|
|
size_t endIndex() const {
|
|
|
|
JS_ASSERT(!isFailure());
|
|
|
|
return endIndex_;
|
|
|
|
}
|
|
|
|
size_t length() const {
|
|
|
|
JS_ASSERT(!isFailure());
|
|
|
|
return length_;
|
|
|
|
}
|
|
|
|
void setResult(size_t length, size_t endIndex) {
|
|
|
|
length_ = length;
|
|
|
|
endIndex_ = endIndex;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class Matcher>
|
|
|
|
static JSObject *
|
|
|
|
SplitHelper(JSContext *cx, JSLinearString *str, uint32 limit, Matcher splitMatch)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-03-09 21:04:47 -08:00
|
|
|
size_t strLength = str->length();
|
|
|
|
SplitMatchResult result;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Step 11. */
|
|
|
|
if (strLength == 0) {
|
|
|
|
if (!splitMatch(cx, str, 0, &result))
|
|
|
|
return NULL;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/*
|
|
|
|
* NB: Unlike in the non-empty string case, it's perfectly fine
|
|
|
|
* (indeed the spec requires it) if we match at the end of the
|
|
|
|
* string. Thus these cases should hold:
|
|
|
|
*
|
|
|
|
* var a = "".split("");
|
|
|
|
* assertEq(a.length, 0);
|
|
|
|
* var b = "".split(/.?/);
|
|
|
|
* assertEq(b.length, 0);
|
|
|
|
*/
|
|
|
|
if (!result.isFailure())
|
|
|
|
return NewDenseEmptyArray(cx);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
Value v = StringValue(str);
|
|
|
|
return NewDenseCopiedArray(cx, 1, &v);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Step 12. */
|
|
|
|
size_t lastEndIndex = 0;
|
|
|
|
size_t index = 0;
|
|
|
|
|
|
|
|
/* Step 13. */
|
|
|
|
AutoValueVector splits(cx);
|
|
|
|
|
|
|
|
while (index < strLength) {
|
|
|
|
/* Step 13(a). */
|
|
|
|
if (!splitMatch(cx, str, index, &result))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Step 13(b).
|
|
|
|
*
|
|
|
|
* Our match algorithm differs from the spec in that it returns the
|
|
|
|
* next index at which a match happens. If no match happens we're
|
|
|
|
* done.
|
|
|
|
*
|
|
|
|
* But what if the match is at the end of the string (and the string is
|
|
|
|
* not empty)? Per 13(c)(ii) this shouldn't be a match, so we have to
|
|
|
|
* specially exclude it. Thus this case should hold:
|
|
|
|
*
|
|
|
|
* var a = "abc".split(/\b/);
|
|
|
|
* assertEq(a.length, 1);
|
|
|
|
* assertEq(a[0], "abc");
|
|
|
|
*/
|
|
|
|
if (result.isFailure())
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Step 13(c)(i). */
|
|
|
|
size_t sepLength = result.length();
|
|
|
|
size_t endIndex = result.endIndex();
|
|
|
|
if (sepLength == 0 && endIndex == strLength)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Step 13(c)(ii). */
|
|
|
|
if (endIndex == lastEndIndex) {
|
|
|
|
index++;
|
|
|
|
continue;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2011-03-09 21:04:47 -08:00
|
|
|
|
|
|
|
/* Step 13(c)(iii). */
|
|
|
|
JS_ASSERT(lastEndIndex < endIndex);
|
|
|
|
JS_ASSERT(sepLength <= strLength);
|
|
|
|
JS_ASSERT(lastEndIndex + sepLength <= endIndex);
|
|
|
|
|
|
|
|
/* Steps 13(c)(iii)(1-3). */
|
|
|
|
size_t subLength = size_t(endIndex - sepLength - lastEndIndex);
|
|
|
|
JSString *sub = js_NewDependentString(cx, str, lastEndIndex, subLength);
|
|
|
|
if (!sub || !splits.append(StringValue(sub)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Step 13(c)(iii)(4). */
|
|
|
|
if (splits.length() == limit)
|
|
|
|
return NewDenseCopiedArray(cx, splits.length(), splits.begin());
|
|
|
|
|
|
|
|
/* Step 13(c)(iii)(5). */
|
|
|
|
lastEndIndex = endIndex;
|
|
|
|
|
|
|
|
/* Step 13(c)(iii)(6-7). */
|
|
|
|
if (Matcher::returnsCaptures) {
|
|
|
|
RegExpStatics *res = cx->regExpStatics();
|
|
|
|
for (size_t i = 0; i < res->parenCount(); i++) {
|
|
|
|
/* Steps 13(c)(iii)(7)(a-c). */
|
|
|
|
if (res->pairIsPresent(i + 1)) {
|
|
|
|
JSSubString parsub;
|
|
|
|
res->getParen(i + 1, &parsub);
|
|
|
|
sub = js_NewStringCopyN(cx, parsub.chars, parsub.length);
|
|
|
|
if (!sub || !splits.append(StringValue(sub)))
|
|
|
|
return NULL;
|
|
|
|
} else {
|
|
|
|
if (!splits.append(UndefinedValue()))
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Step 13(c)(iii)(7)(d). */
|
|
|
|
if (splits.length() == limit)
|
|
|
|
return NewDenseCopiedArray(cx, splits.length(), splits.begin());
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
2011-03-09 21:04:47 -08:00
|
|
|
|
|
|
|
/* Step 13(c)(iii)(8). */
|
|
|
|
index = lastEndIndex;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Steps 14-15. */
|
|
|
|
JSString *sub = js_NewDependentString(cx, str, lastEndIndex, strLength - lastEndIndex);
|
|
|
|
if (!sub || !splits.append(StringValue(sub)))
|
|
|
|
return NULL;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Step 16. */
|
|
|
|
return NewDenseCopiedArray(cx, splits.length(), splits.begin());
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/*
|
|
|
|
* The SplitMatch operation from ES5 15.5.4.14 is implemented using different
|
|
|
|
* matchers for regular expression and string separators.
|
|
|
|
*
|
|
|
|
* The algorithm differs from the spec in that the matchers return the next
|
|
|
|
* index at which a match happens.
|
|
|
|
*/
|
|
|
|
class SplitRegExpMatcher {
|
|
|
|
RegExpStatics *res;
|
|
|
|
RegExp *re;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
public:
|
|
|
|
static const bool returnsCaptures = true;
|
|
|
|
SplitRegExpMatcher(RegExp *re, RegExpStatics *res) : res(res), re(re) {
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool operator()(JSContext *cx, JSLinearString *str, size_t index,
|
|
|
|
SplitMatchResult *result) {
|
2011-03-22 10:52:56 -07:00
|
|
|
Value rval
|
|
|
|
#ifdef __GNUC__ /* quell GCC overwarning */
|
|
|
|
= UndefinedValue()
|
|
|
|
#endif
|
|
|
|
;
|
2011-03-09 21:04:47 -08:00
|
|
|
if (!re->execute(cx, res, str, &index, true, &rval))
|
2010-03-01 14:46:15 -08:00
|
|
|
return false;
|
2011-03-09 21:04:47 -08:00
|
|
|
if (!rval.isTrue()) {
|
|
|
|
result->setFailure();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
JSSubString sep;
|
|
|
|
res->getLastMatch(&sep);
|
|
|
|
|
|
|
|
result->setResult(sep.length, index);
|
2010-03-01 14:46:15 -08:00
|
|
|
return true;
|
|
|
|
}
|
2011-03-09 21:04:47 -08:00
|
|
|
};
|
2010-03-01 14:46:15 -08:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
class SplitStringMatcher {
|
|
|
|
const jschar *sepChars;
|
|
|
|
size_t sepLength;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
public:
|
|
|
|
static const bool returnsCaptures = false;
|
|
|
|
SplitStringMatcher(JSLinearString *sep) {
|
|
|
|
sepChars = sep->chars();
|
|
|
|
sepLength = sep->length();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool operator()(JSContext *cx, JSLinearString *str, size_t index,
|
|
|
|
SplitMatchResult *res) {
|
|
|
|
JS_ASSERT(index == 0 || index < str->length());
|
|
|
|
const jschar *chars = str->chars();
|
|
|
|
jsint match = StringMatch(chars + index, str->length() - index, sepChars, sepLength);
|
|
|
|
if (match == -1)
|
|
|
|
res->setFailure();
|
|
|
|
else
|
|
|
|
res->setResult(sepLength, index + match + sepLength);
|
|
|
|
return true;
|
2010-03-01 14:46:15 -08:00
|
|
|
}
|
2011-03-09 21:04:47 -08:00
|
|
|
};
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* ES5 15.5.4.14 */
|
|
|
|
static JSBool
|
|
|
|
str_split(JSContext *cx, uintN argc, Value *vp)
|
|
|
|
{
|
|
|
|
/* Steps 1-2. */
|
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Step 5: Use the second argument as the split limit, if given. */
|
|
|
|
uint32 limit;
|
|
|
|
if (argc > 1 && !vp[3].isUndefined()) {
|
2010-04-07 13:18:50 -07:00
|
|
|
jsdouble d;
|
2011-06-13 21:49:59 -07:00
|
|
|
if (!ToNumber(cx, vp[3], &d))
|
2010-03-01 14:46:15 -08:00
|
|
|
return false;
|
|
|
|
limit = js_DoubleToECMAUint32(d);
|
2011-03-09 21:04:47 -08:00
|
|
|
} else {
|
|
|
|
limit = UINT32_MAX;
|
2010-03-01 14:46:15 -08:00
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Step 8. */
|
|
|
|
RegExp *re = NULL;
|
|
|
|
JSLinearString *sepstr = NULL;
|
|
|
|
bool sepUndefined = (argc == 0 || vp[2].isUndefined());
|
|
|
|
if (!sepUndefined) {
|
|
|
|
if (VALUE_IS_REGEXP(cx, vp[2])) {
|
|
|
|
re = static_cast<RegExp *>(vp[2].toObject().getPrivate());
|
|
|
|
} else {
|
|
|
|
JSString *sep = js_ValueToString(cx, vp[2]);
|
|
|
|
if (!sep)
|
|
|
|
return false;
|
|
|
|
vp[2].setString(sep);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
sepstr = sep->ensureLinear(cx);
|
|
|
|
if (!sepstr)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Step 9. */
|
|
|
|
if (limit == 0) {
|
|
|
|
JSObject *aobj = NewDenseEmptyArray(cx);
|
|
|
|
if (!aobj)
|
2010-03-01 14:46:15 -08:00
|
|
|
return false;
|
2011-03-09 21:04:47 -08:00
|
|
|
vp->setObject(*aobj);
|
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-03-01 14:46:15 -08:00
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Step 10. */
|
|
|
|
if (sepUndefined) {
|
|
|
|
Value v = StringValue(str);
|
|
|
|
JSObject *aobj = NewDenseCopiedArray(cx, 1, &v);
|
|
|
|
if (!aobj)
|
|
|
|
return false;
|
|
|
|
vp->setObject(*aobj);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
JSLinearString *strlin = str->ensureLinear(cx);
|
|
|
|
if (!strlin)
|
2010-03-01 14:46:15 -08:00
|
|
|
return false;
|
|
|
|
|
2011-03-09 21:04:47 -08:00
|
|
|
/* Steps 11-15. */
|
|
|
|
JSObject *aobj;
|
|
|
|
if (re) {
|
|
|
|
aobj = SplitHelper(cx, strlin, limit, SplitRegExpMatcher(re, cx->regExpStatics()));
|
|
|
|
} else {
|
|
|
|
// NB: sepstr is anchored through its storage in vp[2].
|
|
|
|
aobj = SplitHelper(cx, strlin, limit, SplitStringMatcher(sepstr));
|
|
|
|
}
|
2010-03-01 14:46:15 -08:00
|
|
|
if (!aobj)
|
|
|
|
return false;
|
2011-03-09 21:04:47 -08:00
|
|
|
|
|
|
|
/* Step 16. */
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setObject(*aobj);
|
2010-03-01 14:46:15 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#if JS_HAS_PERL_SUBSTR
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_substr(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2010-12-29 07:22:45 -08:00
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
int32 length, len, begin;
|
2010-12-29 07:22:45 -08:00
|
|
|
if (argc > 0) {
|
|
|
|
length = int32(str->length());
|
|
|
|
if (!ValueToIntegerRange(cx, vp[2], &begin))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (begin >= length) {
|
|
|
|
str = cx->runtime->emptyString;
|
|
|
|
goto out;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
if (begin < 0) {
|
2010-12-29 07:22:45 -08:00
|
|
|
begin += length; /* length + INT_MIN will always be less then 0 */
|
2007-03-22 10:30:00 -07:00
|
|
|
if (begin < 0)
|
|
|
|
begin = 0;
|
|
|
|
}
|
|
|
|
|
2010-10-21 15:02:02 -07:00
|
|
|
if (argc == 1 || vp[3].isUndefined()) {
|
2010-12-29 07:22:45 -08:00
|
|
|
len = length - begin;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
2010-12-29 07:22:45 -08:00
|
|
|
if (!ValueToIntegerRange(cx, vp[3], &len))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (len <= 0) {
|
|
|
|
str = cx->runtime->emptyString;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uint32(length) < uint32(begin + len))
|
|
|
|
len = length - begin;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-12-29 07:22:45 -08:00
|
|
|
str = js_NewDependentString(cx, str, size_t(begin), size_t(len));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2010-12-29 07:22:45 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-12-29 07:22:45 -08:00
|
|
|
|
|
|
|
out:
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2010-12-29 07:22:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
#endif /* JS_HAS_PERL_SUBSTR */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Python-esque sequence operations.
|
|
|
|
*/
|
2008-10-08 15:08:33 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_concat(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-05-13 08:56:26 -07:00
|
|
|
Value *argv = JS_ARGV(cx, vp);
|
|
|
|
for (uintN i = 0; i < argc; i++) {
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str2 = js_ValueToString(cx, argv[i]);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str2)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
str = js_ConcatStrings(cx, str, str2);
|
|
|
|
if (!str)
|
2011-01-26 13:37:45 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-05-13 08:56:26 -07:00
|
|
|
JS_SET_RVAL(cx, vp, StringValue(str));
|
2011-01-26 13:37:45 -08:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_slice(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-07-14 23:19:36 -07:00
|
|
|
if (argc == 1 && vp[1].isString() && vp[2].isInt32()) {
|
2007-08-01 21:33:52 -07:00
|
|
|
size_t begin, end, length;
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
JSString *str = vp[1].toString();
|
|
|
|
begin = vp[2].toInt32();
|
2009-06-10 18:29:44 -07:00
|
|
|
end = str->length();
|
2007-08-01 21:33:52 -07:00
|
|
|
if (begin <= end) {
|
|
|
|
length = end - begin;
|
|
|
|
if (length == 0) {
|
|
|
|
str = cx->runtime->emptyString;
|
|
|
|
} else {
|
2008-02-26 09:01:52 -08:00
|
|
|
str = (length == 1)
|
2011-03-14 13:59:53 -07:00
|
|
|
? JSAtom::getUnitStringForElement(cx, str, begin)
|
2007-08-01 21:33:52 -07:00
|
|
|
: js_NewDependentString(cx, str, begin, length);
|
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2007-08-01 21:33:52 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *str = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (argc != 0) {
|
2007-08-01 21:33:52 -07:00
|
|
|
double begin, end, length;
|
|
|
|
|
2011-04-02 11:33:20 -07:00
|
|
|
if (!ToInteger(cx, vp[2], &begin))
|
|
|
|
return false;
|
2009-06-10 18:29:44 -07:00
|
|
|
length = str->length();
|
2007-03-22 10:30:00 -07:00
|
|
|
if (begin < 0) {
|
|
|
|
begin += length;
|
|
|
|
if (begin < 0)
|
|
|
|
begin = 0;
|
|
|
|
} else if (begin > length) {
|
|
|
|
begin = length;
|
|
|
|
}
|
|
|
|
|
2010-10-21 15:02:02 -07:00
|
|
|
if (argc == 1 || vp[3].isUndefined()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
end = length;
|
|
|
|
} else {
|
2011-04-02 11:33:20 -07:00
|
|
|
if (!ToInteger(cx, vp[3], &end))
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (end < 0) {
|
|
|
|
end += length;
|
|
|
|
if (end < 0)
|
|
|
|
end = 0;
|
|
|
|
} else if (end > length) {
|
|
|
|
end = length;
|
|
|
|
}
|
|
|
|
if (end < begin)
|
|
|
|
end = begin;
|
|
|
|
}
|
|
|
|
|
2007-08-01 21:33:52 -07:00
|
|
|
str = js_NewDependentString(cx, str,
|
|
|
|
(size_t)begin,
|
2007-07-28 09:57:30 -07:00
|
|
|
(size_t)(end - begin));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if JS_HAS_STR_HTML_HELPERS
|
|
|
|
/*
|
|
|
|
* HTML composition aids.
|
|
|
|
*/
|
2010-12-06 10:26:58 -08:00
|
|
|
static bool
|
|
|
|
tagify(JSContext *cx, const char *begin, JSLinearString *param, const char *end,
|
2010-07-14 23:19:36 -07:00
|
|
|
Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-26 13:37:45 -08:00
|
|
|
JSString *thisstr = ThisToStringForStringProto(cx, vp);
|
|
|
|
if (!thisstr)
|
|
|
|
return false;
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str = thisstr->ensureLinear(cx);
|
|
|
|
if (!str)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (!end)
|
|
|
|
end = begin;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t beglen = strlen(begin);
|
|
|
|
size_t taglen = 1 + beglen + 1; /* '<begin' + '>' */
|
|
|
|
size_t parlen = 0; /* Avoid warning. */
|
2007-03-22 10:30:00 -07:00
|
|
|
if (param) {
|
2009-06-10 18:29:44 -07:00
|
|
|
parlen = param->length();
|
2007-03-22 10:30:00 -07:00
|
|
|
taglen += 2 + parlen + 1; /* '="param"' */
|
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t endlen = strlen(end);
|
|
|
|
taglen += str->length() + 2 + endlen + 1; /* 'str</end>' */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (taglen >= ~(size_t)0 / sizeof(jschar)) {
|
2008-03-12 16:07:47 -07:00
|
|
|
js_ReportAllocationOverflow(cx);
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *tagbuf = (jschar *) cx->malloc_((taglen + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!tagbuf)
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t j = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
tagbuf[j++] = '<';
|
2010-12-06 10:26:58 -08:00
|
|
|
for (size_t i = 0; i < beglen; i++)
|
2007-03-22 10:30:00 -07:00
|
|
|
tagbuf[j++] = (jschar)begin[i];
|
|
|
|
if (param) {
|
|
|
|
tagbuf[j++] = '=';
|
|
|
|
tagbuf[j++] = '"';
|
2009-06-10 18:29:44 -07:00
|
|
|
js_strncpy(&tagbuf[j], param->chars(), parlen);
|
2007-03-22 10:30:00 -07:00
|
|
|
j += parlen;
|
|
|
|
tagbuf[j++] = '"';
|
|
|
|
}
|
|
|
|
tagbuf[j++] = '>';
|
2010-12-06 10:26:58 -08:00
|
|
|
|
2009-06-10 18:29:44 -07:00
|
|
|
js_strncpy(&tagbuf[j], str->chars(), str->length());
|
|
|
|
j += str->length();
|
2007-03-22 10:30:00 -07:00
|
|
|
tagbuf[j++] = '<';
|
|
|
|
tagbuf[j++] = '/';
|
2010-12-06 10:26:58 -08:00
|
|
|
for (size_t i = 0; i < endlen; i++)
|
2007-03-22 10:30:00 -07:00
|
|
|
tagbuf[j++] = (jschar)end[i];
|
|
|
|
tagbuf[j++] = '>';
|
|
|
|
JS_ASSERT(j == taglen);
|
|
|
|
tagbuf[j] = 0;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSString *retstr = js_NewString(cx, tagbuf, taglen);
|
|
|
|
if (!retstr) {
|
2011-03-31 01:14:12 -07:00
|
|
|
Foreground::free_((char *)tagbuf);
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
vp->setString(retstr);
|
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
tagify_value(JSContext *cx, uintN argc, Value *vp,
|
2008-08-08 09:02:50 -07:00
|
|
|
const char *begin, const char *end)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *param = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!param)
|
|
|
|
return JS_FALSE;
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, begin, param, end, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_bold(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "b", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_italics(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "i", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_fixed(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "tt", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_fontsize(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-08-08 09:02:50 -07:00
|
|
|
return tagify_value(cx, argc, vp, "font size", "font");
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_fontcolor(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-08-08 09:02:50 -07:00
|
|
|
return tagify_value(cx, argc, vp, "font color", "font");
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_link(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-08-08 09:02:50 -07:00
|
|
|
return tagify_value(cx, argc, vp, "a href", "a");
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_anchor(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-08-08 09:02:50 -07:00
|
|
|
return tagify_value(cx, argc, vp, "a name", "a");
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_strike(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "strike", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_small(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "small", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_big(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "big", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_blink(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "blink", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_sup(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "sup", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_sub(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-08-01 21:33:52 -07:00
|
|
|
return tagify(cx, "sub", NULL, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
#endif /* JS_HAS_STR_HTML_HELPERS */
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
#ifdef JS_TRACER
|
|
|
|
JSString* FASTCALL
|
|
|
|
js_String_getelem(JSContext* cx, JSString* str, int32 i)
|
|
|
|
{
|
2009-06-10 18:29:44 -07:00
|
|
|
if ((size_t)i >= str->length())
|
2008-10-08 15:08:33 -07:00
|
|
|
return NULL;
|
2011-03-14 13:59:53 -07:00
|
|
|
return JSAtom::getUnitStringForElement(cx, str, size_t(i));
|
2008-10-08 15:08:33 -07:00
|
|
|
}
|
2008-10-21 15:58:06 -07:00
|
|
|
#endif
|
2008-10-08 15:08:33 -07:00
|
|
|
|
2009-03-20 18:53:07 -07:00
|
|
|
JS_DEFINE_TRCINFO_1(str_concat,
|
2010-08-06 03:29:14 -07:00
|
|
|
(3, (extern, STRING_RETRY, js_ConcatStrings, CONTEXT, THIS_STRING, STRING,
|
2010-07-27 22:33:42 -07:00
|
|
|
1, nanojit::ACCSET_NONE)))
|
2008-10-08 15:08:33 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
static JSFunctionSpec string_methods[] = {
|
|
|
|
#if JS_HAS_TOSOURCE
|
2010-12-22 15:05:07 -08:00
|
|
|
JS_FN("quote", str_quote, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN(js_toSource_str, str_toSource, 0,0),
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Java-like methods. */
|
2010-12-22 15:05:07 -08:00
|
|
|
JS_FN(js_toString_str, js_str_toString, 0,0),
|
|
|
|
JS_FN(js_valueOf_str, js_str_toString, 0,0),
|
|
|
|
JS_FN("substring", str_substring, 2,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("toLowerCase", str_toLowerCase, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("toUpperCase", str_toUpperCase, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("charAt", js_str_charAt, 1,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("charCodeAt", js_str_charCodeAt, 1,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("indexOf", str_indexOf, 1,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("lastIndexOf", str_lastIndexOf, 1,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("trim", str_trim, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("trimLeft", str_trimLeft, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("trimRight", str_trimRight, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("toLocaleLowerCase", str_toLocaleLowerCase, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("toLocaleUpperCase", str_toLocaleUpperCase, 0,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("localeCompare", str_localeCompare, 1,JSFUN_GENERIC_NATIVE),
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/* Perl-ish methods (search is actually Python-esque). */
|
2010-12-22 15:05:07 -08:00
|
|
|
JS_FN("match", str_match, 1,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("search", str_search, 1,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("replace", str_replace, 2,JSFUN_GENERIC_NATIVE),
|
|
|
|
JS_FN("split", str_split, 2,JSFUN_GENERIC_NATIVE),
|
2007-03-22 10:30:00 -07:00
|
|
|
#if JS_HAS_PERL_SUBSTR
|
2010-12-22 15:05:07 -08:00
|
|
|
JS_FN("substr", str_substr, 2,JSFUN_GENERIC_NATIVE),
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Python-esque sequence methods. */
|
2010-12-22 15:05:07 -08:00
|
|
|
JS_TN("concat", str_concat, 1,JSFUN_GENERIC_NATIVE, &str_concat_trcinfo),
|
|
|
|
JS_FN("slice", str_slice, 2,JSFUN_GENERIC_NATIVE),
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/* HTML string methods. */
|
|
|
|
#if JS_HAS_STR_HTML_HELPERS
|
2010-12-22 15:05:07 -08:00
|
|
|
JS_FN("bold", str_bold, 0,0),
|
|
|
|
JS_FN("italics", str_italics, 0,0),
|
|
|
|
JS_FN("fixed", str_fixed, 0,0),
|
|
|
|
JS_FN("fontsize", str_fontsize, 1,0),
|
|
|
|
JS_FN("fontcolor", str_fontcolor, 1,0),
|
|
|
|
JS_FN("link", str_link, 1,0),
|
|
|
|
JS_FN("anchor", str_anchor, 1,0),
|
|
|
|
JS_FN("strike", str_strike, 0,0),
|
|
|
|
JS_FN("small", str_small, 0,0),
|
|
|
|
JS_FN("big", str_big, 0,0),
|
|
|
|
JS_FN("blink", str_blink, 0,0),
|
|
|
|
JS_FN("sup", str_sup, 0,0),
|
|
|
|
JS_FN("sub", str_sub, 0,0),
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
|
|
|
|
2007-08-01 21:33:52 -07:00
|
|
|
JS_FS_END
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
2009-09-04 16:43:28 -07:00
|
|
|
/*
|
2010-07-27 21:14:31 -07:00
|
|
|
* Set up some tools to make it easier to generate large tables. After constant
|
|
|
|
* folding, for each n, Rn(0) is the comma-separated list R(0), R(1), ..., R(2^n-1).
|
|
|
|
* Similary, Rn(k) (for any k and n) generates the list R(k), R(k+1), ..., R(k+2^n-1).
|
|
|
|
* To use this, define R appropriately, then use Rn(0) (for some value of n), then
|
|
|
|
* undefine R.
|
2009-09-04 16:43:28 -07:00
|
|
|
*/
|
2010-07-27 21:14:31 -07:00
|
|
|
#define R2(n) R(n), R((n) + (1 << 0)), R((n) + (2 << 0)), R((n) + (3 << 0))
|
|
|
|
#define R4(n) R2(n), R2((n) + (1 << 2)), R2((n) + (2 << 2)), R2((n) + (3 << 2))
|
|
|
|
#define R6(n) R4(n), R4((n) + (1 << 4)), R4((n) + (2 << 4)), R4((n) + (3 << 4))
|
|
|
|
#define R8(n) R6(n), R6((n) + (1 << 6)), R6((n) + (2 << 6)), R6((n) + (3 << 6))
|
|
|
|
#define R10(n) R8(n), R8((n) + (1 << 8)), R8((n) + (2 << 8)), R8((n) + (3 << 8))
|
|
|
|
#define R12(n) R10(n), R10((n) + (1 << 10)), R10((n) + (2 << 10)), R10((n) + (3 << 10))
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
#define R3(n) R2(n), R2((n) + (1 << 2))
|
|
|
|
#define R7(n) R6(n), R6((n) + (1 << 6))
|
|
|
|
|
2010-11-30 18:41:32 -08:00
|
|
|
#define BUILD_LENGTH_AND_FLAGS(length, flags) \
|
|
|
|
(((length) << JSString::LENGTH_SHIFT) | (flags))
|
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
/*
|
|
|
|
* Declare unit strings. Pack the string data itself into the mInlineChars
|
|
|
|
* place in the header.
|
|
|
|
*/
|
|
|
|
#define R(c) { \
|
2011-03-14 13:55:55 -07:00
|
|
|
BUILD_LENGTH_AND_FLAGS(1, JSString::STATIC_ATOM_FLAGS), \
|
2011-06-19 14:04:48 -07:00
|
|
|
{ (jschar *)(uintptr_t(unitStaticTable + (c)) + \
|
2011-03-14 13:59:53 -07:00
|
|
|
offsetof(JSString::Data, inlineStorage)) }, \
|
2010-07-27 21:14:31 -07:00
|
|
|
{ {(c), 0x00} } }
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2011-04-19 16:25:07 -07:00
|
|
|
/*
|
|
|
|
* For all the pragma pack usage in this file, the following logic applies:
|
|
|
|
* To apply: To reset:
|
|
|
|
* Sun CC: pack(#) / pack(0)
|
|
|
|
* IBM xlC: pack(#) / pack(pop)
|
|
|
|
* HP aCC: pack # / pack
|
|
|
|
* Others: pack(push, #) / pack(pop)
|
|
|
|
* The -Dlint case is explicitly excluded because GCC will error out when
|
|
|
|
* pack pragmas are used on unsupported platforms. If GCC is being used
|
|
|
|
* simply for error checking, these errors will be avoided.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(__SUNPRO_CC) || defined(__xlC__)
|
2009-09-10 00:29:20 -07:00
|
|
|
#pragma pack(8)
|
2011-04-19 16:25:07 -07:00
|
|
|
#elif defined(__HP_aCC)
|
|
|
|
#pragma pack 8
|
|
|
|
#elif !defined(lint)
|
2009-09-08 13:18:59 -07:00
|
|
|
#pragma pack(push, 8)
|
2009-09-10 00:29:20 -07:00
|
|
|
#endif
|
2009-09-08 13:18:59 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
const JSString::Data JSAtom::unitStaticTable[]
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__GNUC__) || defined(__xlC__)
|
2009-09-08 14:55:30 -07:00
|
|
|
__attribute__ ((aligned (8)))
|
|
|
|
#endif
|
2010-07-27 21:14:31 -07:00
|
|
|
= { R8(0) };
|
2009-09-08 13:18:59 -07:00
|
|
|
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__SUNPRO_CC)
|
2009-09-10 00:29:20 -07:00
|
|
|
#pragma pack(0)
|
2011-04-19 16:25:07 -07:00
|
|
|
#elif defined(__HP_aCC)
|
|
|
|
#pragma pack
|
|
|
|
#elif !defined(lint)
|
2009-09-08 13:18:59 -07:00
|
|
|
#pragma pack(pop)
|
2009-09-10 00:29:20 -07:00
|
|
|
#endif
|
2009-09-08 14:55:30 -07:00
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
#undef R
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
/*
|
|
|
|
* Declare length-2 strings. We only store strings where both characters are
|
|
|
|
* alphanumeric. The lower 10 short chars are the numerals, the next 26 are
|
|
|
|
* the lowercase letters, and the next 26 are the uppercase letters.
|
|
|
|
*/
|
|
|
|
#define TO_SMALL_CHAR(c) ((c) >= '0' && (c) <= '9' ? (c) - '0' : \
|
|
|
|
(c) >= 'a' && (c) <= 'z' ? (c) - 'a' + 10 : \
|
|
|
|
(c) >= 'A' && (c) <= 'Z' ? (c) - 'A' + 36 : \
|
2011-03-14 13:59:53 -07:00
|
|
|
JSAtom::INVALID_SMALL_CHAR)
|
2010-07-27 21:14:31 -07:00
|
|
|
|
|
|
|
#define R TO_SMALL_CHAR
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
const JSAtom::SmallChar JSAtom::toSmallChar[] = { R7(0) };
|
2010-07-27 21:14:31 -07:00
|
|
|
|
|
|
|
#undef R
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2010-03-13 09:46:15 -08:00
|
|
|
/*
|
2010-07-27 21:14:31 -07:00
|
|
|
* This is used when we generate our table of short strings, so the compiler is
|
|
|
|
* happier if we use |c| as few times as possible.
|
2009-09-04 16:43:28 -07:00
|
|
|
*/
|
2010-07-27 21:14:31 -07:00
|
|
|
#define FROM_SMALL_CHAR(c) ((c) + ((c) < 10 ? '0' : \
|
|
|
|
(c) < 36 ? 'a' - 10 : \
|
|
|
|
'A' - 36))
|
|
|
|
#define R FROM_SMALL_CHAR
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
const jschar JSAtom::fromSmallChar[] = { R6(0) };
|
2010-07-16 17:41:22 -07:00
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
#undef R
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For code-generation ease, length-2 strings are encoded as 12-bit int values,
|
|
|
|
* where the upper 6 bits is the first character and the lower 6 bits is the
|
|
|
|
* second character.
|
|
|
|
*/
|
|
|
|
#define R(c) { \
|
2011-03-14 13:55:55 -07:00
|
|
|
BUILD_LENGTH_AND_FLAGS(2, JSString::STATIC_ATOM_FLAGS), \
|
2011-06-19 14:04:48 -07:00
|
|
|
{ (jschar *)(uintptr_t(length2StaticTable + (c)) + \
|
2011-03-14 13:59:53 -07:00
|
|
|
offsetof(JSString::Data, inlineStorage)) }, \
|
2010-07-27 21:14:31 -07:00
|
|
|
{ {FROM_SMALL_CHAR((c) >> 6), FROM_SMALL_CHAR((c) & 0x3F), 0x00} } }
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__SUNPRO_CC) || defined(__xlC__)
|
2009-09-10 00:29:20 -07:00
|
|
|
#pragma pack(8)
|
2011-04-19 16:25:07 -07:00
|
|
|
#elif defined(__HP_aCC)
|
|
|
|
#pragma pack 8
|
|
|
|
#elif !defined(lint)
|
2009-09-08 13:18:59 -07:00
|
|
|
#pragma pack(push, 8)
|
2009-09-10 00:29:20 -07:00
|
|
|
#endif
|
2009-09-08 13:18:59 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
const JSString::Data JSAtom::length2StaticTable[]
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__GNUC__) || defined(__xlC__)
|
2009-09-08 14:55:30 -07:00
|
|
|
__attribute__ ((aligned (8)))
|
|
|
|
#endif
|
2010-07-27 21:14:31 -07:00
|
|
|
= { R12(0) };
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__SUNPRO_CC)
|
2009-09-10 00:29:20 -07:00
|
|
|
#pragma pack(0)
|
2011-04-19 16:25:07 -07:00
|
|
|
#elif defined(__HP_aCC)
|
|
|
|
#pragma pack
|
|
|
|
#elif !defined(lint)
|
2009-09-08 13:18:59 -07:00
|
|
|
#pragma pack(pop)
|
2009-09-10 00:29:20 -07:00
|
|
|
#endif
|
2009-09-08 13:18:59 -07:00
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
#undef R
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Declare int strings. Only int strings from 100 to 255 actually have to be
|
|
|
|
* generated, since the rest are either unit strings or length-2 strings. To
|
|
|
|
* avoid the runtime cost of figuring out where to look for the string for a
|
|
|
|
* particular integer, we precompute a table of JSString*s which refer to the
|
|
|
|
* correct location of the int string.
|
|
|
|
*/
|
|
|
|
#define R(c) { \
|
2011-03-14 13:55:55 -07:00
|
|
|
BUILD_LENGTH_AND_FLAGS(3, JSString::STATIC_ATOM_FLAGS), \
|
2011-06-19 14:04:48 -07:00
|
|
|
{ (jschar *)(uintptr_t(hundredStaticTable + ((c) - 100)) + \
|
2011-03-14 13:59:53 -07:00
|
|
|
offsetof(JSString::Data, inlineStorage)) }, \
|
2010-07-27 21:14:31 -07:00
|
|
|
{ {((c) / 100) + '0', ((c) / 10 % 10) + '0', ((c) % 10) + '0', 0x00} } }
|
|
|
|
|
|
|
|
|
|
|
|
JS_STATIC_ASSERT(100 + (1 << 7) + (1 << 4) + (1 << 3) + (1 << 2) == 256);
|
|
|
|
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__SUNPRO_CC) || defined(__xlC__)
|
2010-07-27 21:14:31 -07:00
|
|
|
#pragma pack(8)
|
2011-04-19 16:25:07 -07:00
|
|
|
#elif defined(__HP_aCC)
|
|
|
|
#pragma pack 8
|
|
|
|
#elif !defined(lint)
|
2010-07-27 21:14:31 -07:00
|
|
|
#pragma pack(push, 8)
|
|
|
|
#endif
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
const JSString::Data JSAtom::hundredStaticTable[]
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__GNUC__) || defined(__xlC__)
|
2010-07-27 21:14:31 -07:00
|
|
|
__attribute__ ((aligned (8)))
|
|
|
|
#endif
|
|
|
|
= { R7(100), /* 100 through 227 */
|
|
|
|
R4(100 + (1 << 7)), /* 228 through 243 */
|
|
|
|
R3(100 + (1 << 7) + (1 << 4)), /* 244 through 251 */
|
|
|
|
R2(100 + (1 << 7) + (1 << 4) + (1 << 3)) /* 252 through 255 */
|
2009-09-08 17:46:02 -07:00
|
|
|
};
|
|
|
|
|
2010-07-27 21:14:31 -07:00
|
|
|
#undef R
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
#define R(c) ((c) < 10 ? JSAtom::unitStaticTable + ((c) + '0') : \
|
|
|
|
(c) < 100 ? JSAtom::length2StaticTable + \
|
2010-07-27 21:14:31 -07:00
|
|
|
((size_t)TO_SMALL_CHAR(((c) / 10) + '0') << 6) + \
|
|
|
|
TO_SMALL_CHAR(((c) % 10) + '0') : \
|
2011-03-14 13:59:53 -07:00
|
|
|
JSAtom::hundredStaticTable + ((c) - 100))
|
2010-07-27 21:14:31 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
const JSString::Data *const JSAtom::intStaticTable[] = { R8(0) };
|
2010-07-27 21:14:31 -07:00
|
|
|
|
|
|
|
#undef R
|
|
|
|
|
2011-04-19 16:25:07 -07:00
|
|
|
#if defined(__SUNPRO_CC)
|
2010-07-27 21:14:31 -07:00
|
|
|
#pragma pack(0)
|
2011-04-19 16:25:07 -07:00
|
|
|
#elif defined(__HP_aCC)
|
|
|
|
#pragma pack
|
|
|
|
#elif !defined(lint)
|
2010-07-27 21:14:31 -07:00
|
|
|
#pragma pack(pop)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#undef R2
|
|
|
|
#undef R4
|
|
|
|
#undef R6
|
|
|
|
#undef R8
|
|
|
|
#undef R10
|
|
|
|
#undef R12
|
|
|
|
|
|
|
|
#undef R3
|
|
|
|
#undef R7
|
2009-09-04 16:43:28 -07:00
|
|
|
|
2009-03-09 17:29:42 -07:00
|
|
|
JSBool
|
2010-08-03 12:20:48 -07:00
|
|
|
js_String(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-08-16 12:35:04 -07:00
|
|
|
Value *argv = vp + 2;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-16 12:35:04 -07:00
|
|
|
JSString *str;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (argc > 0) {
|
2010-08-16 12:35:04 -07:00
|
|
|
str = js_ValueToString(cx, argv[0]);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2010-08-16 12:35:04 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
|
|
|
str = cx->runtime->emptyString;
|
|
|
|
}
|
2010-08-03 12:20:48 -07:00
|
|
|
|
2010-08-16 12:35:04 -07:00
|
|
|
if (IsConstructing(vp)) {
|
2011-04-20 13:33:05 -07:00
|
|
|
StringObject *strobj = StringObject::create(cx, str);
|
|
|
|
if (!strobj)
|
2010-08-16 12:35:04 -07:00
|
|
|
return false;
|
2011-04-20 13:33:05 -07:00
|
|
|
vp->setObject(*strobj);
|
2010-08-03 12:20:48 -07:00
|
|
|
} else {
|
|
|
|
vp->setString(str);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-16 12:35:04 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_fromCharCode(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-03-14 13:59:53 -07:00
|
|
|
Value *argv = JS_ARGV(cx, vp);
|
2011-06-30 09:26:56 -07:00
|
|
|
JS_ASSERT(argc <= StackSpace::ARGS_LENGTH_MAX);
|
2010-04-07 13:18:50 -07:00
|
|
|
if (argc == 1) {
|
|
|
|
uint16_t code;
|
|
|
|
if (!ValueToUint16(cx, argv[0], &code))
|
2008-07-31 12:21:59 -07:00
|
|
|
return JS_FALSE;
|
2011-03-14 13:59:53 -07:00
|
|
|
if (JSAtom::hasUnitStatic(code)) {
|
|
|
|
vp->setString(&JSAtom::unitStatic(code));
|
2010-04-07 13:18:50 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2010-07-14 23:19:36 -07:00
|
|
|
argv[0].setInt32(code);
|
2008-07-31 12:21:59 -07:00
|
|
|
}
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *chars = (jschar *) cx->malloc_((argc + 1) * sizeof(jschar));
|
2008-08-14 15:19:57 -07:00
|
|
|
if (!chars)
|
|
|
|
return JS_FALSE;
|
2011-03-14 13:59:53 -07:00
|
|
|
for (uintN i = 0; i < argc; i++) {
|
2010-04-07 13:18:50 -07:00
|
|
|
uint16_t code;
|
|
|
|
if (!ValueToUint16(cx, argv[i], &code)) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(chars);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
chars[i] = (jschar)code;
|
|
|
|
}
|
2011-03-14 13:59:53 -07:00
|
|
|
chars[argc] = 0;
|
|
|
|
JSString *str = js_NewString(cx, chars, argc);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str) {
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(chars);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2010-07-14 23:19:36 -07:00
|
|
|
vp->setString(str);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2008-10-08 15:08:33 -07:00
|
|
|
#ifdef JS_TRACER
|
2008-10-16 12:24:10 -07:00
|
|
|
static JSString* FASTCALL
|
|
|
|
String_fromCharCode(JSContext* cx, int32 i)
|
2008-10-08 15:08:33 -07:00
|
|
|
{
|
|
|
|
JS_ASSERT(JS_ON_TRACE(cx));
|
|
|
|
jschar c = (jschar)i;
|
2011-03-14 13:59:53 -07:00
|
|
|
if (JSAtom::hasUnitStatic(c))
|
|
|
|
return &JSAtom::unitStatic(c);
|
2008-10-08 15:08:33 -07:00
|
|
|
return js_NewStringCopyN(cx, &c, 1);
|
|
|
|
}
|
2008-10-21 15:58:06 -07:00
|
|
|
#endif
|
2008-10-08 15:08:33 -07:00
|
|
|
|
2008-10-21 15:58:06 -07:00
|
|
|
JS_DEFINE_TRCINFO_1(str_fromCharCode,
|
2010-07-27 22:33:42 -07:00
|
|
|
(2, (static, STRING_RETRY, String_fromCharCode, CONTEXT, INT32, 1, nanojit::ACCSET_NONE)))
|
2008-10-08 15:08:33 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
static JSFunctionSpec string_static_methods[] = {
|
2009-09-04 13:44:31 -07:00
|
|
|
JS_TN("fromCharCode", str_fromCharCode, 1, 0, &str_fromCharCode_trcinfo),
|
2007-08-01 21:33:52 -07:00
|
|
|
JS_FS_END
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
2011-03-13 20:38:34 -07:00
|
|
|
const Shape *
|
2011-04-20 13:33:05 -07:00
|
|
|
StringObject::assignInitialShape(JSContext *cx)
|
2011-03-13 20:38:34 -07:00
|
|
|
{
|
|
|
|
JS_ASSERT(!cx->compartment->initialStringShape);
|
|
|
|
JS_ASSERT(nativeEmpty());
|
|
|
|
|
|
|
|
return addDataProperty(cx, ATOM_TO_JSID(cx->runtime->atomState.lengthAtom),
|
2011-04-20 13:33:05 -07:00
|
|
|
LENGTH_SLOT, JSPROP_PERMANENT | JSPROP_READONLY);
|
2011-03-13 20:38:34 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
JSObject *
|
2011-06-13 11:52:45 -07:00
|
|
|
js_InitStringClass(JSContext *cx, JSObject *obj)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-06-13 11:52:45 -07:00
|
|
|
JS_ASSERT(obj->isNative());
|
2011-03-13 20:38:34 -07:00
|
|
|
|
2011-06-13 11:52:45 -07:00
|
|
|
GlobalObject *global = obj->asGlobal();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-06-13 11:52:45 -07:00
|
|
|
JSObject *proto = global->createBlankPrototype(cx, &js_StringClass);
|
2011-04-20 13:33:05 -07:00
|
|
|
if (!proto || !proto->asString()->init(cx, cx->runtime->emptyString))
|
2011-03-13 20:38:34 -07:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Now create the String function. */
|
2011-06-13 11:52:45 -07:00
|
|
|
JSFunction *ctor = global->createConstructor(cx, js_String, &js_StringClass,
|
|
|
|
CLASS_ATOM(cx, String), 1);
|
2011-03-13 20:38:34 -07:00
|
|
|
if (!ctor)
|
|
|
|
return NULL;
|
|
|
|
|
2011-06-15 10:15:37 -07:00
|
|
|
if (!LinkConstructorAndPrototype(cx, ctor, proto))
|
2007-03-22 10:30:00 -07:00
|
|
|
return NULL;
|
2009-03-20 15:31:01 -07:00
|
|
|
|
2011-06-15 10:15:37 -07:00
|
|
|
if (!DefinePropertiesAndBrand(cx, proto, NULL, string_methods) ||
|
|
|
|
!DefinePropertiesAndBrand(cx, ctor, NULL, string_static_methods))
|
2011-03-13 20:38:34 -07:00
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!DefineConstructorAndPrototype(cx, global, JSProto_String, ctor, proto))
|
|
|
|
return NULL;
|
|
|
|
|
2011-06-13 11:52:45 -07:00
|
|
|
/*
|
|
|
|
* Define escape/unescape, the URI encode/decode functions, and maybe
|
|
|
|
* uneval on the global object.
|
|
|
|
*/
|
|
|
|
if (!JS_DefineFunctions(cx, global, string_functions))
|
|
|
|
return NULL;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
return proto;
|
|
|
|
}
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *
|
2008-08-11 11:03:54 -07:00
|
|
|
js_NewString(JSContext *cx, jschar *chars, size_t length)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-03-14 13:59:53 -07:00
|
|
|
if (!CheckStringLength(cx, length))
|
2007-03-22 10:30:00 -07:00
|
|
|
return NULL;
|
|
|
|
|
2010-09-20 12:43:53 -07:00
|
|
|
JSFixedString *s = JSFixedString::new_(cx, chars, length);
|
|
|
|
Probes::createString(cx, s, length);
|
|
|
|
return s;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-31 16:07:21 -07:00
|
|
|
static JS_ALWAYS_INLINE JSFixedString *
|
2010-10-26 16:20:21 -07:00
|
|
|
NewShortString(JSContext *cx, const jschar *chars, size_t length)
|
|
|
|
{
|
2011-03-31 16:07:21 -07:00
|
|
|
/*
|
|
|
|
* Don't bother trying to find a static atom; measurement shows that not
|
|
|
|
* many get here (for one, Atomize is catching them).
|
|
|
|
*/
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JS_ASSERT(JSShortString::lengthFits(length));
|
2011-03-31 16:07:21 -07:00
|
|
|
JSInlineString *str = JSInlineString::lengthFits(length)
|
|
|
|
? JSInlineString::new_(cx)
|
|
|
|
: JSShortString::new_(cx);
|
2010-10-26 16:20:21 -07:00
|
|
|
if (!str)
|
|
|
|
return NULL;
|
2011-03-31 16:07:21 -07:00
|
|
|
|
2010-10-26 16:20:21 -07:00
|
|
|
jschar *storage = str->init(length);
|
2011-03-14 13:59:53 -07:00
|
|
|
PodCopy(storage, chars, length);
|
2010-10-26 16:20:21 -07:00
|
|
|
storage[length] = 0;
|
2010-09-20 12:43:53 -07:00
|
|
|
Probes::createString(cx, str, length);
|
2011-03-14 13:59:53 -07:00
|
|
|
return str;
|
2010-10-26 16:20:21 -07:00
|
|
|
}
|
|
|
|
|
2011-03-31 16:07:21 -07:00
|
|
|
static JSInlineString *
|
2010-10-26 16:20:21 -07:00
|
|
|
NewShortString(JSContext *cx, const char *chars, size_t length)
|
|
|
|
{
|
2011-03-14 13:59:53 -07:00
|
|
|
JS_ASSERT(JSShortString::lengthFits(length));
|
2011-03-31 16:07:21 -07:00
|
|
|
JSInlineString *str = JSInlineString::lengthFits(length)
|
|
|
|
? JSInlineString::new_(cx)
|
|
|
|
: JSShortString::new_(cx);
|
2010-10-26 16:20:21 -07:00
|
|
|
if (!str)
|
|
|
|
return NULL;
|
|
|
|
|
2011-03-31 16:07:21 -07:00
|
|
|
jschar *storage = str->init(length);
|
2010-10-26 16:20:21 -07:00
|
|
|
if (js_CStringsAreUTF8) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
size_t oldLength = length;
|
|
|
|
#endif
|
2011-05-26 15:50:05 -07:00
|
|
|
if (!InflateUTF8StringToBuffer(cx, chars, length, storage, &length))
|
2010-10-26 16:20:21 -07:00
|
|
|
return NULL;
|
|
|
|
JS_ASSERT(length <= oldLength);
|
|
|
|
storage[length] = 0;
|
|
|
|
str->resetLength(length);
|
|
|
|
} else {
|
|
|
|
size_t n = length;
|
|
|
|
jschar *p = storage;
|
|
|
|
while (n--)
|
2010-12-06 10:26:58 -08:00
|
|
|
*p++ = (unsigned char)*chars++;
|
2010-10-26 16:20:21 -07:00
|
|
|
*p = 0;
|
|
|
|
}
|
2010-09-20 12:43:53 -07:00
|
|
|
Probes::createString(cx, str, length);
|
2011-03-14 13:59:53 -07:00
|
|
|
return str;
|
2010-10-26 16:20:21 -07:00
|
|
|
}
|
|
|
|
|
2011-04-01 16:33:41 -07:00
|
|
|
jschar *
|
|
|
|
StringBuffer::extractWellSized()
|
|
|
|
{
|
|
|
|
size_t capacity = cb.capacity();
|
|
|
|
size_t length = cb.length();
|
|
|
|
|
|
|
|
jschar *buf = cb.extractRawBuffer();
|
|
|
|
if (!buf)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* For medium/big buffers, avoid wasting more than 1/4 of the memory. */
|
|
|
|
JS_ASSERT(capacity >= length);
|
|
|
|
if (length > CharBuffer::sMaxInlineStorage &&
|
|
|
|
capacity - length > (length >> 2)) {
|
|
|
|
size_t bytes = sizeof(jschar) * (length + 1);
|
|
|
|
JSContext *cx = context();
|
|
|
|
jschar *tmp = (jschar *)cx->realloc_(buf, bytes);
|
|
|
|
if (!tmp) {
|
|
|
|
cx->free_(buf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
buf = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
2009-08-11 15:45:32 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *
|
2011-01-12 15:28:58 -08:00
|
|
|
StringBuffer::finishString()
|
2009-08-07 20:09:11 -07:00
|
|
|
{
|
2011-01-12 15:28:58 -08:00
|
|
|
JSContext *cx = context();
|
2009-08-07 20:09:11 -07:00
|
|
|
if (cb.empty())
|
2011-03-14 13:59:53 -07:00
|
|
|
return cx->runtime->atomState.emptyAtom;
|
2009-08-07 20:09:11 -07:00
|
|
|
|
2009-08-14 16:10:59 -07:00
|
|
|
size_t length = cb.length();
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!checkLength(length))
|
|
|
|
return NULL;
|
2010-10-26 16:20:21 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JS_STATIC_ASSERT(JSShortString::MAX_SHORT_LENGTH < CharBuffer::InlineLength);
|
|
|
|
if (JSShortString::lengthFits(length))
|
2010-10-26 16:20:21 -07:00
|
|
|
return NewShortString(cx, cb.begin(), length);
|
|
|
|
|
2009-08-07 20:09:11 -07:00
|
|
|
if (!cb.append('\0'))
|
|
|
|
return NULL;
|
|
|
|
|
2011-04-01 16:33:41 -07:00
|
|
|
jschar *buf = extractWellSized();
|
2009-08-07 20:09:11 -07:00
|
|
|
if (!buf)
|
|
|
|
return NULL;
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *str = js_NewString(cx, buf, length);
|
2009-08-07 20:09:11 -07:00
|
|
|
if (!str)
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(buf);
|
2009-08-07 20:09:11 -07:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2011-03-25 12:04:05 -07:00
|
|
|
JSAtom *
|
|
|
|
StringBuffer::finishAtom()
|
|
|
|
{
|
|
|
|
JSContext *cx = context();
|
|
|
|
|
|
|
|
size_t length = cb.length();
|
|
|
|
if (length == 0)
|
|
|
|
return cx->runtime->atomState.emptyAtom;
|
|
|
|
|
2011-05-17 12:15:12 -07:00
|
|
|
JSAtom *atom = js_AtomizeChars(cx, cb.begin(), length);
|
2011-03-25 12:04:05 -07:00
|
|
|
cb.clear();
|
|
|
|
return atom;
|
|
|
|
}
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *
|
2011-03-14 13:59:53 -07:00
|
|
|
js_NewDependentString(JSContext *cx, JSString *baseArg, size_t start, size_t length)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
if (length == 0)
|
|
|
|
return cx->runtime->emptyString;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *base = baseArg->ensureLinear(cx);
|
|
|
|
if (!base)
|
|
|
|
return NULL;
|
|
|
|
|
2009-06-10 18:29:44 -07:00
|
|
|
if (start == 0 && length == base->length())
|
2007-03-22 10:30:00 -07:00
|
|
|
return base;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *chars = base->chars() + start;
|
2010-07-24 20:22:35 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
if (JSLinearString *staticStr = JSAtom::lookupStatic(chars, length))
|
2010-10-28 14:33:32 -07:00
|
|
|
return staticStr;
|
2010-09-14 20:07:11 -07:00
|
|
|
|
2010-09-20 12:43:53 -07:00
|
|
|
JSLinearString *s = JSDependentString::new_(cx, base, chars, length);
|
|
|
|
Probes::createString(cx, s, length);
|
|
|
|
return s;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *
|
2007-07-28 09:57:30 -07:00
|
|
|
js_NewStringCopyN(JSContext *cx, const jschar *s, size_t n)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-03-14 13:59:53 -07:00
|
|
|
if (JSShortString::lengthFits(n))
|
2010-07-24 20:26:34 -07:00
|
|
|
return NewShortString(cx, s, n);
|
|
|
|
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *news = (jschar *) cx->malloc_((n + 1) * sizeof(jschar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!news)
|
|
|
|
return NULL;
|
|
|
|
js_strncpy(news, s, n);
|
|
|
|
news[n] = 0;
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *str = js_NewString(cx, news, n);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(news);
|
2007-03-22 10:30:00 -07:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *
|
2010-07-24 20:26:34 -07:00
|
|
|
js_NewStringCopyN(JSContext *cx, const char *s, size_t n)
|
|
|
|
{
|
2011-03-14 13:59:53 -07:00
|
|
|
if (JSShortString::lengthFits(n))
|
2010-07-24 20:26:34 -07:00
|
|
|
return NewShortString(cx, s, n);
|
2010-12-06 10:26:58 -08:00
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
jschar *chars = InflateString(cx, s, &n);
|
2010-12-06 10:26:58 -08:00
|
|
|
if (!chars)
|
|
|
|
return NULL;
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *str = js_NewString(cx, chars, n);
|
2010-12-06 10:26:58 -08:00
|
|
|
if (!str)
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(chars);
|
2010-12-06 10:26:58 -08:00
|
|
|
return str;
|
2010-07-24 20:26:34 -07:00
|
|
|
}
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *
|
2007-07-28 09:57:30 -07:00
|
|
|
js_NewStringCopyZ(JSContext *cx, const jschar *s)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t n = js_strlen(s);
|
2011-03-14 13:59:53 -07:00
|
|
|
if (JSShortString::lengthFits(n))
|
2010-07-24 20:26:34 -07:00
|
|
|
return NewShortString(cx, s, n);
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t m = (n + 1) * sizeof(jschar);
|
2011-03-31 01:14:12 -07:00
|
|
|
jschar *news = (jschar *) cx->malloc_(m);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!news)
|
|
|
|
return NULL;
|
|
|
|
memcpy(news, s, m);
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *str = js_NewString(cx, news, n);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
2011-03-31 01:14:12 -07:00
|
|
|
cx->free_(news);
|
2007-03-22 10:30:00 -07:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
JSFixedString *
|
2010-07-24 20:26:34 -07:00
|
|
|
js_NewStringCopyZ(JSContext *cx, const char *s)
|
|
|
|
{
|
|
|
|
return js_NewStringCopyN(cx, s, strlen(s));
|
|
|
|
}
|
|
|
|
|
2010-11-11 12:40:29 -08:00
|
|
|
const char *
|
|
|
|
js_ValueToPrintable(JSContext *cx, const Value &v, JSAutoByteString *bytes, bool asSource)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
JSString *str;
|
|
|
|
|
2010-11-11 12:40:29 -08:00
|
|
|
str = (asSource ? js_ValueToSource : js_ValueToString)(cx, v);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return NULL;
|
|
|
|
str = js_QuoteString(cx, str, 0);
|
|
|
|
if (!str)
|
|
|
|
return NULL;
|
2010-11-11 12:40:29 -08:00
|
|
|
return bytes->encode(cx, str);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
JSString *
|
|
|
|
js_ValueToString(JSContext *cx, const Value &arg)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-07-14 23:19:36 -07:00
|
|
|
Value v = arg;
|
2011-04-01 15:24:21 -07:00
|
|
|
if (!ToPrimitive(cx, JSTYPE_STRING, &v))
|
2009-06-30 17:19:42 -07:00
|
|
|
return NULL;
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
JSString *str;
|
|
|
|
if (v.isString()) {
|
|
|
|
str = v.toString();
|
|
|
|
} else if (v.isInt32()) {
|
2010-11-10 11:43:22 -08:00
|
|
|
str = js_IntToString(cx, v.toInt32());
|
2010-07-14 23:19:36 -07:00
|
|
|
} else if (v.isDouble()) {
|
|
|
|
str = js_NumberToString(cx, v.toDouble());
|
|
|
|
} else if (v.isBoolean()) {
|
|
|
|
str = js_BooleanToString(cx, v.toBoolean());
|
|
|
|
} else if (v.isNull()) {
|
2011-03-14 13:59:53 -07:00
|
|
|
str = cx->runtime->atomState.nullAtom;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
2011-03-14 13:59:53 -07:00
|
|
|
str = cx->runtime->atomState.typeAtoms[JSTYPE_VOID];
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2009-06-30 17:19:42 -07:00
|
|
|
/* This function implements E-262-3 section 9.8, toString. */
|
2011-01-12 15:28:58 -08:00
|
|
|
bool
|
2011-03-29 15:46:51 -07:00
|
|
|
js::ValueToStringBufferSlow(JSContext *cx, const Value &arg, StringBuffer &sb)
|
2009-06-30 17:19:42 -07:00
|
|
|
{
|
2010-07-14 23:19:36 -07:00
|
|
|
Value v = arg;
|
2011-04-01 15:24:21 -07:00
|
|
|
if (!ToPrimitive(cx, JSTYPE_STRING, &v))
|
2010-12-06 10:26:58 -08:00
|
|
|
return false;
|
2009-06-30 17:19:42 -07:00
|
|
|
|
2011-03-29 15:46:51 -07:00
|
|
|
if (v.isString())
|
|
|
|
return sb.append(v.toString());
|
2010-07-14 23:19:36 -07:00
|
|
|
if (v.isNumber())
|
2011-01-12 15:28:58 -08:00
|
|
|
return NumberValueToStringBuffer(cx, v, sb);
|
2010-07-14 23:19:36 -07:00
|
|
|
if (v.isBoolean())
|
2011-01-12 15:28:58 -08:00
|
|
|
return BooleanToStringBuffer(cx, v.toBoolean(), sb);
|
2010-07-14 23:19:36 -07:00
|
|
|
if (v.isNull())
|
2011-01-12 15:28:58 -08:00
|
|
|
return sb.append(cx->runtime->atomState.nullAtom);
|
2010-07-14 23:19:36 -07:00
|
|
|
JS_ASSERT(v.isUndefined());
|
2011-01-12 15:28:58 -08:00
|
|
|
return sb.append(cx->runtime->atomState.typeAtoms[JSTYPE_VOID]);
|
2009-06-30 17:19:42 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
JS_FRIEND_API(JSString *)
|
2010-07-14 23:19:36 -07:00
|
|
|
js_ValueToSource(JSContext *cx, const Value &v)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-04-26 13:05:58 -07:00
|
|
|
JS_CHECK_RECURSION(cx, return NULL);
|
|
|
|
|
2010-07-14 23:19:36 -07:00
|
|
|
if (v.isUndefined())
|
2011-03-14 13:59:53 -07:00
|
|
|
return cx->runtime->atomState.void0Atom;
|
2010-07-14 23:19:36 -07:00
|
|
|
if (v.isString())
|
|
|
|
return js_QuoteString(cx, v.toString(), '"');
|
|
|
|
if (v.isPrimitive()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
/* Special case to preserve negative zero, _contra_ toString. */
|
2010-07-14 23:19:36 -07:00
|
|
|
if (v.isDouble() && JSDOUBLE_IS_NEGZERO(v.toDouble())) {
|
2007-03-22 10:30:00 -07:00
|
|
|
/* NB: _ucNstr rather than _ucstr to indicate non-terminated. */
|
|
|
|
static const jschar js_negzero_ucNstr[] = {'-', '0'};
|
|
|
|
|
2007-07-28 09:57:30 -07:00
|
|
|
return js_NewStringCopyN(cx, js_negzero_ucNstr, 2);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
return js_ValueToString(cx, v);
|
|
|
|
}
|
|
|
|
|
2011-03-28 20:01:53 -07:00
|
|
|
Value rval = NullValue();
|
|
|
|
Value fval;
|
|
|
|
jsid id = ATOM_TO_JSID(cx->runtime->atomState.toSourceAtom);
|
|
|
|
if (!js_GetMethod(cx, &v.toObject(), id, JSGET_NO_METHOD_BARRIER, &fval))
|
|
|
|
return false;
|
|
|
|
if (js_IsCallable(fval)) {
|
|
|
|
if (!ExternalInvoke(cx, v, fval, 0, NULL, &rval))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return js_ValueToString(cx, rval);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
namespace js {
|
|
|
|
|
|
|
|
bool
|
|
|
|
EqualStrings(JSContext *cx, JSString *str1, JSString *str2, JSBool *result)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
if (str1 == str2) {
|
|
|
|
*result = true;
|
|
|
|
return true;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length1 = str1->length();
|
|
|
|
if (length1 != str2->length()) {
|
|
|
|
*result = false;
|
|
|
|
return true;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *linear1 = str1->ensureLinear(cx);
|
|
|
|
if (!linear1)
|
|
|
|
return false;
|
|
|
|
JSLinearString *linear2 = str2->ensureLinear(cx);
|
|
|
|
if (!linear2)
|
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
*result = PodEqual(linear1->chars(), linear2->chars(), length1);
|
2010-12-06 10:26:58 -08:00
|
|
|
return true;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
bool
|
|
|
|
EqualStrings(JSLinearString *str1, JSLinearString *str2)
|
|
|
|
{
|
|
|
|
if (str1 == str2)
|
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length1 = str1->length();
|
|
|
|
if (length1 != str2->length())
|
|
|
|
return false;
|
|
|
|
|
2011-03-14 13:59:53 -07:00
|
|
|
return PodEqual(str1->chars(), str2->chars(), length1);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
} /* namespace js */
|
|
|
|
|
|
|
|
JSBool JS_FASTCALL
|
|
|
|
js_EqualStringsOnTrace(JSContext *cx, JSString *str1, JSString *str2)
|
2007-08-11 13:25:16 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSBool result;
|
|
|
|
return EqualStrings(cx, str1, str2, &result) ? result : JS_NEITHER;
|
|
|
|
}
|
|
|
|
JS_DEFINE_CALLINFO_3(extern, BOOL, js_EqualStringsOnTrace,
|
|
|
|
CONTEXT, STRING, STRING, 1, nanojit::ACCSET_NONE)
|
2007-08-11 13:25:16 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
namespace js {
|
2007-08-11 13:25:16 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
static bool
|
|
|
|
CompareStringsImpl(JSContext *cx, JSString *str1, JSString *str2, int32 *result)
|
|
|
|
{
|
2007-08-11 13:25:16 -07:00
|
|
|
JS_ASSERT(str1);
|
|
|
|
JS_ASSERT(str2);
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
if (str1 == str2) {
|
|
|
|
*result = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t l1 = str1->length();
|
|
|
|
const jschar *s1 = str1->getChars(cx);
|
|
|
|
if (!s1)
|
|
|
|
return false;
|
2007-08-11 13:25:16 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t l2 = str2->length();
|
|
|
|
const jschar *s2 = str2->getChars(cx);
|
|
|
|
if (!s2)
|
|
|
|
return false;
|
2007-08-11 13:25:16 -07:00
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t n = JS_MIN(l1, l2);
|
|
|
|
for (size_t i = 0; i < n; i++) {
|
|
|
|
if (int32 cmp = s1[i] - s2[i]) {
|
|
|
|
*result = cmp;
|
|
|
|
return true;
|
|
|
|
}
|
2007-08-11 13:25:16 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
*result = (int32)(l1 - l2);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
CompareStrings(JSContext *cx, JSString *str1, JSString *str2, int32 *result)
|
|
|
|
{
|
|
|
|
return CompareStringsImpl(cx, str1, str2, result);
|
2007-08-11 13:25:16 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
|
|
|
|
} /* namespace js */
|
|
|
|
|
|
|
|
int32 JS_FASTCALL
|
|
|
|
js_CompareStringsOnTrace(JSContext *cx, JSString *str1, JSString *str2)
|
|
|
|
{
|
|
|
|
int32 result;
|
|
|
|
if (!CompareStringsImpl(cx, str1, str2, &result))
|
|
|
|
return INT32_MIN;
|
|
|
|
JS_ASSERT(result != INT32_MIN);
|
|
|
|
return result;
|
2007-08-11 13:25:16 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
JS_DEFINE_CALLINFO_3(extern, INT32, js_CompareStringsOnTrace,
|
|
|
|
CONTEXT, STRING, STRING, 1, nanojit::ACCSET_NONE)
|
2007-08-11 13:25:16 -07:00
|
|
|
|
2010-10-28 08:15:53 -07:00
|
|
|
namespace js {
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
bool
|
|
|
|
StringEqualsAscii(JSLinearString *str, const char *asciiBytes)
|
2010-10-28 08:15:53 -07:00
|
|
|
{
|
|
|
|
size_t length = strlen(asciiBytes);
|
|
|
|
#ifdef DEBUG
|
|
|
|
for (size_t i = 0; i != length; ++i)
|
|
|
|
JS_ASSERT(unsigned(asciiBytes[i]) <= 127);
|
|
|
|
#endif
|
|
|
|
if (length != str->length())
|
|
|
|
return false;
|
|
|
|
const jschar *chars = str->chars();
|
|
|
|
for (size_t i = 0; i != length; ++i) {
|
|
|
|
if (unsigned(asciiBytes[i]) != unsigned(chars[i]))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} /* namespacejs */
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
size_t
|
|
|
|
js_strlen(const jschar *s)
|
|
|
|
{
|
|
|
|
const jschar *t;
|
|
|
|
|
|
|
|
for (t = s; *t != 0; t++)
|
|
|
|
continue;
|
|
|
|
return (size_t)(t - s);
|
|
|
|
}
|
|
|
|
|
|
|
|
jschar *
|
|
|
|
js_strchr(const jschar *s, jschar c)
|
|
|
|
{
|
|
|
|
while (*s != 0) {
|
|
|
|
if (*s == c)
|
|
|
|
return (jschar *)s;
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
jschar *
|
|
|
|
js_strchr_limit(const jschar *s, jschar c, const jschar *limit)
|
|
|
|
{
|
|
|
|
while (s < limit) {
|
|
|
|
if (*s == c)
|
|
|
|
return (jschar *)s;
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
namespace js {
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
jschar *
|
2011-05-26 15:50:05 -07:00
|
|
|
InflateString(JSContext *cx, const char *bytes, size_t *lengthp, FlationCoding fc)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-05-26 15:50:05 -07:00
|
|
|
size_t nchars;
|
2007-12-11 02:09:58 -08:00
|
|
|
jschar *chars;
|
2011-05-26 15:50:05 -07:00
|
|
|
size_t nbytes = *lengthp;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
if (js_CStringsAreUTF8 || fc == CESU8Encoding) {
|
|
|
|
if (!InflateUTF8StringToBuffer(cx, bytes, nbytes, NULL, &nchars, fc))
|
2007-12-11 02:09:58 -08:00
|
|
|
goto bad;
|
2011-03-31 01:14:12 -07:00
|
|
|
chars = (jschar *) cx->malloc_((nchars + 1) * sizeof (jschar));
|
2007-12-11 02:09:58 -08:00
|
|
|
if (!chars)
|
|
|
|
goto bad;
|
2011-05-26 15:50:05 -07:00
|
|
|
JS_ALWAYS_TRUE(InflateUTF8StringToBuffer(cx, bytes, nbytes, chars, &nchars, fc));
|
2007-12-11 02:09:58 -08:00
|
|
|
} else {
|
|
|
|
nchars = nbytes;
|
2011-03-31 01:14:12 -07:00
|
|
|
chars = (jschar *) cx->malloc_((nchars + 1) * sizeof(jschar));
|
2007-12-11 02:09:58 -08:00
|
|
|
if (!chars)
|
|
|
|
goto bad;
|
2011-05-26 15:50:05 -07:00
|
|
|
for (size_t i = 0; i < nchars; i++)
|
2007-12-11 02:09:58 -08:00
|
|
|
chars[i] = (unsigned char) bytes[i];
|
|
|
|
}
|
|
|
|
*lengthp = nchars;
|
|
|
|
chars[nchars] = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
return chars;
|
2007-12-11 02:09:58 -08:00
|
|
|
|
|
|
|
bad:
|
|
|
|
/*
|
|
|
|
* For compatibility with callers of JS_DecodeBytes we must zero lengthp
|
|
|
|
* on errors.
|
|
|
|
*/
|
|
|
|
*lengthp = 0;
|
|
|
|
return NULL;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-11-24 08:56:43 -08:00
|
|
|
* May be called with null cx.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
|
|
|
char *
|
2011-05-26 15:50:05 -07:00
|
|
|
DeflateString(JSContext *cx, const jschar *chars, size_t nchars)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-12-11 02:09:58 -08:00
|
|
|
size_t nbytes, i;
|
2007-03-22 10:30:00 -07:00
|
|
|
char *bytes;
|
|
|
|
|
2007-12-11 02:09:58 -08:00
|
|
|
if (js_CStringsAreUTF8) {
|
2011-05-26 15:50:05 -07:00
|
|
|
nbytes = GetDeflatedStringLength(cx, chars, nchars);
|
2007-12-11 02:09:58 -08:00
|
|
|
if (nbytes == (size_t) -1)
|
|
|
|
return NULL;
|
2011-03-31 01:14:12 -07:00
|
|
|
bytes = (char *) (cx ? cx->malloc_(nbytes + 1) : OffTheBooks::malloc_(nbytes + 1));
|
2007-12-11 02:09:58 -08:00
|
|
|
if (!bytes)
|
|
|
|
return NULL;
|
2011-05-26 15:50:05 -07:00
|
|
|
JS_ALWAYS_TRUE(DeflateStringToBuffer(cx, chars, nchars, bytes, &nbytes));
|
2007-12-11 02:09:58 -08:00
|
|
|
} else {
|
|
|
|
nbytes = nchars;
|
2011-03-31 01:14:12 -07:00
|
|
|
bytes = (char *) (cx ? cx->malloc_(nbytes + 1) : OffTheBooks::malloc_(nbytes + 1));
|
2007-12-11 02:09:58 -08:00
|
|
|
if (!bytes)
|
|
|
|
return NULL;
|
|
|
|
for (i = 0; i < nbytes; i++)
|
|
|
|
bytes[i] = (char) chars[i];
|
|
|
|
}
|
|
|
|
bytes[nbytes] = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
return bytes;
|
|
|
|
}
|
|
|
|
|
2010-04-02 13:05:30 -07:00
|
|
|
size_t
|
2011-05-26 15:50:05 -07:00
|
|
|
GetDeflatedStringLength(JSContext *cx, const jschar *chars, size_t nchars)
|
2010-04-02 13:05:30 -07:00
|
|
|
{
|
|
|
|
if (!js_CStringsAreUTF8)
|
|
|
|
return nchars;
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
return GetDeflatedUTF8StringLength(cx, chars, nchars);
|
2010-04-02 13:05:30 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/*
|
2010-11-24 08:56:43 -08:00
|
|
|
* May be called with null cx through public API, see below.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
|
|
|
size_t
|
2011-05-26 15:50:05 -07:00
|
|
|
GetDeflatedUTF8StringLength(JSContext *cx, const jschar *chars,
|
|
|
|
size_t nchars, FlationCoding fc)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-12-11 02:09:58 -08:00
|
|
|
size_t nbytes;
|
2007-03-22 10:30:00 -07:00
|
|
|
const jschar *end;
|
|
|
|
uintN c, c2;
|
|
|
|
char buffer[10];
|
2011-05-26 15:50:05 -07:00
|
|
|
bool useCESU8 = fc == CESU8Encoding;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-12-11 02:09:58 -08:00
|
|
|
nbytes = nchars;
|
|
|
|
for (end = chars + nchars; chars != end; chars++) {
|
2007-03-22 10:30:00 -07:00
|
|
|
c = *chars;
|
|
|
|
if (c < 0x80)
|
|
|
|
continue;
|
2011-03-21 11:06:34 -07:00
|
|
|
if (0xD800 <= c && c <= 0xDFFF && !useCESU8) {
|
2007-03-22 10:30:00 -07:00
|
|
|
/* Surrogate pair. */
|
|
|
|
chars++;
|
2010-10-19 01:06:10 -07:00
|
|
|
|
|
|
|
/* nbytes sets 1 length since this is surrogate pair. */
|
|
|
|
nbytes--;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (c >= 0xDC00 || chars == end)
|
|
|
|
goto bad_surrogate;
|
|
|
|
c2 = *chars;
|
|
|
|
if (c2 < 0xDC00 || c2 > 0xDFFF)
|
|
|
|
goto bad_surrogate;
|
|
|
|
c = ((c - 0xD800) << 10) + (c2 - 0xDC00) + 0x10000;
|
|
|
|
}
|
|
|
|
c >>= 11;
|
2007-12-11 02:09:58 -08:00
|
|
|
nbytes++;
|
2007-03-22 10:30:00 -07:00
|
|
|
while (c) {
|
|
|
|
c >>= 5;
|
2007-12-11 02:09:58 -08:00
|
|
|
nbytes++;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
2007-12-11 02:09:58 -08:00
|
|
|
return nbytes;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
bad_surrogate:
|
|
|
|
if (cx) {
|
|
|
|
JS_snprintf(buffer, 10, "0x%x", c);
|
|
|
|
JS_ReportErrorFlagsAndNumber(cx, JSREPORT_ERROR, js_GetErrorMessage,
|
|
|
|
NULL, JSMSG_BAD_SURROGATE_CHAR, buffer);
|
|
|
|
}
|
2007-12-11 02:09:58 -08:00
|
|
|
return (size_t) -1;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
bool
|
|
|
|
DeflateStringToBuffer(JSContext *cx, const jschar *src, size_t srclen,
|
|
|
|
char *dst, size_t *dstlenp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-04-02 13:05:30 -07:00
|
|
|
size_t dstlen, i;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-12-11 02:09:58 -08:00
|
|
|
dstlen = *dstlenp;
|
|
|
|
if (!js_CStringsAreUTF8) {
|
|
|
|
if (srclen > dstlen) {
|
|
|
|
for (i = 0; i < dstlen; i++)
|
|
|
|
dst[i] = (char) src[i];
|
|
|
|
if (cx) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BUFFER_TOO_SMALL);
|
|
|
|
}
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
for (i = 0; i < srclen; i++)
|
|
|
|
dst[i] = (char) src[i];
|
|
|
|
*dstlenp = srclen;
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
return DeflateStringToUTF8Buffer(cx, src, srclen, dst, dstlenp);
|
2010-04-02 13:05:30 -07:00
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
bool
|
|
|
|
DeflateStringToUTF8Buffer(JSContext *cx, const jschar *src, size_t srclen,
|
|
|
|
char *dst, size_t *dstlenp, FlationCoding fc)
|
2010-04-02 13:05:30 -07:00
|
|
|
{
|
2011-05-26 15:50:05 -07:00
|
|
|
size_t i, utf8Len;
|
2010-04-02 13:05:30 -07:00
|
|
|
jschar c, c2;
|
|
|
|
uint32 v;
|
|
|
|
uint8 utf8buf[6];
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
bool useCESU8 = fc == CESU8Encoding;
|
|
|
|
size_t dstlen = *dstlenp;
|
|
|
|
size_t origDstlen = dstlen;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
while (srclen) {
|
|
|
|
c = *src++;
|
|
|
|
srclen--;
|
2011-03-21 11:06:34 -07:00
|
|
|
if ((c >= 0xDC00) && (c <= 0xDFFF) && !useCESU8)
|
2007-03-22 10:30:00 -07:00
|
|
|
goto badSurrogate;
|
2011-03-21 11:06:34 -07:00
|
|
|
if (c < 0xD800 || c > 0xDBFF || useCESU8) {
|
2007-03-22 10:30:00 -07:00
|
|
|
v = c;
|
|
|
|
} else {
|
|
|
|
if (srclen < 1)
|
|
|
|
goto badSurrogate;
|
|
|
|
c2 = *src;
|
|
|
|
if ((c2 < 0xDC00) || (c2 > 0xDFFF))
|
|
|
|
goto badSurrogate;
|
|
|
|
src++;
|
|
|
|
srclen--;
|
|
|
|
v = ((c - 0xD800) << 10) + (c2 - 0xDC00) + 0x10000;
|
|
|
|
}
|
|
|
|
if (v < 0x0080) {
|
|
|
|
/* no encoding necessary - performance hack */
|
2007-06-18 14:38:31 -07:00
|
|
|
if (dstlen == 0)
|
2007-03-22 10:30:00 -07:00
|
|
|
goto bufferTooSmall;
|
|
|
|
*dst++ = (char) v;
|
|
|
|
utf8Len = 1;
|
|
|
|
} else {
|
|
|
|
utf8Len = js_OneUcs4ToUtf8Char(utf8buf, v);
|
|
|
|
if (utf8Len > dstlen)
|
|
|
|
goto bufferTooSmall;
|
|
|
|
for (i = 0; i < utf8Len; i++)
|
|
|
|
*dst++ = (char) utf8buf[i];
|
|
|
|
}
|
|
|
|
dstlen -= utf8Len;
|
|
|
|
}
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
badSurrogate:
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
/* Delegate error reporting to the measurement function. */
|
|
|
|
if (cx)
|
2011-05-26 15:50:05 -07:00
|
|
|
GetDeflatedStringLength(cx, src - 1, srclen + 1);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
|
|
|
|
bufferTooSmall:
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
if (cx) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BUFFER_TOO_SMALL);
|
|
|
|
}
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
bool
|
|
|
|
InflateStringToBuffer(JSContext *cx, const char *src, size_t srclen,
|
|
|
|
jschar *dst, size_t *dstlenp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-04-02 13:05:30 -07:00
|
|
|
size_t dstlen, i;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
if (js_CStringsAreUTF8)
|
|
|
|
return InflateUTF8StringToBuffer(cx, src, srclen, dst, dstlenp);
|
|
|
|
|
|
|
|
if (dst) {
|
|
|
|
dstlen = *dstlenp;
|
|
|
|
if (srclen > dstlen) {
|
|
|
|
for (i = 0; i < dstlen; i++)
|
2007-12-11 02:09:58 -08:00
|
|
|
dst[i] = (unsigned char) src[i];
|
2011-05-26 15:50:05 -07:00
|
|
|
if (cx) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BUFFER_TOO_SMALL);
|
|
|
|
}
|
|
|
|
return JS_FALSE;
|
2007-12-11 02:09:58 -08:00
|
|
|
}
|
2011-05-26 15:50:05 -07:00
|
|
|
for (i = 0; i < srclen; i++)
|
|
|
|
dst[i] = (unsigned char) src[i];
|
2007-12-11 02:09:58 -08:00
|
|
|
}
|
2011-05-26 15:50:05 -07:00
|
|
|
*dstlenp = srclen;
|
|
|
|
return JS_TRUE;
|
2010-04-02 13:05:30 -07:00
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
bool
|
|
|
|
InflateUTF8StringToBuffer(JSContext *cx, const char *src, size_t srclen,
|
|
|
|
jschar *dst, size_t *dstlenp, FlationCoding fc)
|
2010-04-02 13:05:30 -07:00
|
|
|
{
|
|
|
|
size_t dstlen, origDstlen, offset, j, n;
|
|
|
|
uint32 v;
|
|
|
|
|
2007-12-11 02:09:58 -08:00
|
|
|
dstlen = dst ? *dstlenp : (size_t) -1;
|
|
|
|
origDstlen = dstlen;
|
|
|
|
offset = 0;
|
2011-05-26 15:50:05 -07:00
|
|
|
bool useCESU8 = fc == CESU8Encoding;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
while (srclen) {
|
|
|
|
v = (uint8) *src;
|
|
|
|
n = 1;
|
|
|
|
if (v & 0x80) {
|
|
|
|
while (v & (0x80 >> n))
|
|
|
|
n++;
|
|
|
|
if (n > srclen)
|
|
|
|
goto bufferTooSmall;
|
2009-10-01 17:10:14 -07:00
|
|
|
if (n == 1 || n > 4)
|
2007-03-22 10:30:00 -07:00
|
|
|
goto badCharacter;
|
|
|
|
for (j = 1; j < n; j++) {
|
|
|
|
if ((src[j] & 0xC0) != 0x80)
|
|
|
|
goto badCharacter;
|
|
|
|
}
|
2007-04-05 03:52:35 -07:00
|
|
|
v = Utf8ToOneUcs4Char((uint8 *)src, n);
|
2011-03-21 11:06:34 -07:00
|
|
|
if (v >= 0x10000 && !useCESU8) {
|
2007-03-22 10:30:00 -07:00
|
|
|
v -= 0x10000;
|
|
|
|
if (v > 0xFFFFF || dstlen < 2) {
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
if (cx) {
|
|
|
|
char buffer[10];
|
|
|
|
JS_snprintf(buffer, 10, "0x%x", v + 0x10000);
|
|
|
|
JS_ReportErrorFlagsAndNumber(cx,
|
|
|
|
JSREPORT_ERROR,
|
|
|
|
js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_UTF8_CHAR_TOO_LARGE,
|
|
|
|
buffer);
|
|
|
|
}
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
if (dst) {
|
|
|
|
*dst++ = (jschar)((v >> 10) + 0xD800);
|
|
|
|
v = (jschar)((v & 0x3FF) + 0xDC00);
|
|
|
|
}
|
|
|
|
dstlen--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!dstlen)
|
|
|
|
goto bufferTooSmall;
|
|
|
|
if (dst)
|
|
|
|
*dst++ = (jschar) v;
|
|
|
|
dstlen--;
|
|
|
|
offset += n;
|
|
|
|
src += n;
|
|
|
|
srclen -= n;
|
|
|
|
}
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
badCharacter:
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
if (cx) {
|
|
|
|
char buffer[10];
|
|
|
|
JS_snprintf(buffer, 10, "%d", offset);
|
|
|
|
JS_ReportErrorFlagsAndNumber(cx, JSREPORT_ERROR,
|
|
|
|
js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_MALFORMED_UTF8_CHAR,
|
|
|
|
buffer);
|
|
|
|
}
|
|
|
|
return JS_FALSE;
|
|
|
|
|
|
|
|
bufferTooSmall:
|
|
|
|
*dstlenp = (origDstlen - dstlen);
|
|
|
|
if (cx) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BUFFER_TOO_SMALL);
|
|
|
|
}
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2011-05-26 15:50:05 -07:00
|
|
|
} /* namepsace js */
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
const jschar js_uriReservedPlusPound_ucstr[] =
|
|
|
|
{';', '/', '?', ':', '@', '&', '=', '+', '$', ',', '#', 0};
|
|
|
|
const jschar js_uriUnescaped_ucstr[] =
|
|
|
|
{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
|
|
|
|
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
|
|
|
|
'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
|
|
|
|
'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
|
|
|
|
'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
|
|
|
|
'-', '_', '.', '!', '~', '*', '\'', '(', ')', 0};
|
|
|
|
|
2011-03-16 15:17:55 -07:00
|
|
|
#define ____ false
|
|
|
|
|
2011-03-16 15:18:12 -07:00
|
|
|
/*
|
|
|
|
* Identifier start chars:
|
|
|
|
* - 36: $
|
|
|
|
* - 65..90: A..Z
|
|
|
|
* - 95: _
|
|
|
|
* - 97..122: a..z
|
|
|
|
*/
|
|
|
|
const bool js_isidstart[] = {
|
|
|
|
/* 0 1 2 3 4 5 6 7 8 9 */
|
|
|
|
/* 0 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 1 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 2 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 3 */ ____, ____, ____, ____, ____, ____, true, ____, ____, ____,
|
|
|
|
/* 4 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 5 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 6 */ ____, ____, ____, ____, ____, true, true, true, true, true,
|
|
|
|
/* 7 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 8 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 9 */ true, ____, ____, ____, ____, true, ____, true, true, true,
|
|
|
|
/* 10 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 11 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 12 */ true, true, true, ____, ____, ____, ____, ____
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Identifier chars:
|
|
|
|
* - 36: $
|
|
|
|
* - 48..57: 0..9
|
|
|
|
* - 65..90: A..Z
|
|
|
|
* - 95: _
|
|
|
|
* - 97..122: a..z
|
|
|
|
*/
|
|
|
|
const bool js_isident[] = {
|
|
|
|
/* 0 1 2 3 4 5 6 7 8 9 */
|
|
|
|
/* 0 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 1 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 2 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 3 */ ____, ____, ____, ____, ____, ____, true, ____, ____, ____,
|
|
|
|
/* 4 */ ____, ____, ____, ____, ____, ____, ____, ____, true, true,
|
|
|
|
/* 5 */ true, true, true, true, true, true, true, true, ____, ____,
|
|
|
|
/* 6 */ ____, ____, ____, ____, ____, true, true, true, true, true,
|
|
|
|
/* 7 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 8 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 9 */ true, ____, ____, ____, ____, true, ____, true, true, true,
|
|
|
|
/* 10 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 11 */ true, true, true, true, true, true, true, true, true, true,
|
|
|
|
/* 12 */ true, true, true, ____, ____, ____, ____, ____
|
|
|
|
};
|
|
|
|
|
2011-03-16 15:17:55 -07:00
|
|
|
/* Whitespace chars: '\t', '\n', '\v', '\f', '\r', ' '. */
|
|
|
|
const bool js_isspace[] = {
|
|
|
|
/* 0 1 2 3 4 5 6 7 8 9 */
|
|
|
|
/* 0 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, true,
|
|
|
|
/* 1 */ true, true, true, true, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 2 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 3 */ ____, ____, true, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 4 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 5 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 6 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 7 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 8 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 9 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 10 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 11 */ ____, ____, ____, ____, ____, ____, ____, ____, ____, ____,
|
|
|
|
/* 12 */ ____, ____, ____, ____, ____, ____, ____, ____
|
2009-07-16 17:17:35 -07:00
|
|
|
};
|
|
|
|
|
2011-03-16 15:17:55 -07:00
|
|
|
#undef ____
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
#define URI_CHUNK 64U
|
|
|
|
|
2009-08-14 16:10:59 -07:00
|
|
|
static inline bool
|
2011-01-12 15:28:58 -08:00
|
|
|
TransferBufferToString(JSContext *cx, StringBuffer &sb, Value *rval)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-01-12 15:28:58 -08:00
|
|
|
JSString *str = sb.finishString();
|
2007-08-15 23:23:06 -07:00
|
|
|
if (!str)
|
2009-08-14 16:10:59 -07:00
|
|
|
return false;
|
2010-07-14 23:19:36 -07:00
|
|
|
rval->setString(str);
|
2011-03-16 15:17:55 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ECMA 3, 15.1.3 URI Handling Function Properties
|
|
|
|
*
|
|
|
|
* The following are implementations of the algorithms
|
|
|
|
* given in the ECMA specification for the hidden functions
|
|
|
|
* 'Encode' and 'Decode'.
|
|
|
|
*/
|
|
|
|
static JSBool
|
|
|
|
Encode(JSContext *cx, JSString *str, const jschar *unescapedSet,
|
2010-07-14 23:19:36 -07:00
|
|
|
const jschar *unescapedSet2, Value *rval)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
static const char HexDigits[] = "0123456789ABCDEF"; /* NB: uppercase */
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length = str->length();
|
|
|
|
const jschar *chars = str->getChars(cx);
|
|
|
|
if (!chars)
|
|
|
|
return JS_FALSE;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
if (length == 0) {
|
2010-07-14 23:19:36 -07:00
|
|
|
rval->setString(cx->runtime->emptyString);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
StringBuffer sb(cx);
|
2010-12-06 10:26:58 -08:00
|
|
|
jschar hexBuf[4];
|
2007-03-22 10:30:00 -07:00
|
|
|
hexBuf[0] = '%';
|
|
|
|
hexBuf[3] = 0;
|
2010-12-06 10:26:58 -08:00
|
|
|
for (size_t k = 0; k < length; k++) {
|
|
|
|
jschar c = chars[k];
|
2007-03-22 10:30:00 -07:00
|
|
|
if (js_strchr(unescapedSet, c) ||
|
|
|
|
(unescapedSet2 && js_strchr(unescapedSet2, c))) {
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(c))
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
|
|
|
if ((c >= 0xDC00) && (c <= 0xDFFF)) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BAD_URI, NULL);
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
uint32 v;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (c < 0xD800 || c > 0xDBFF) {
|
|
|
|
v = c;
|
|
|
|
} else {
|
|
|
|
k++;
|
|
|
|
if (k == length) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BAD_URI, NULL);
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
jschar c2 = chars[k];
|
2007-03-22 10:30:00 -07:00
|
|
|
if ((c2 < 0xDC00) || (c2 > 0xDFFF)) {
|
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
|
|
|
JSMSG_BAD_URI, NULL);
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
v = ((c - 0xD800) << 10) + (c2 - 0xDC00) + 0x10000;
|
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
uint8 utf8buf[4];
|
|
|
|
size_t L = js_OneUcs4ToUtf8Char(utf8buf, v);
|
|
|
|
for (size_t j = 0; j < L; j++) {
|
2007-03-22 10:30:00 -07:00
|
|
|
hexBuf[1] = HexDigits[utf8buf[j] >> 4];
|
|
|
|
hexBuf[2] = HexDigits[utf8buf[j] & 0xf];
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(hexBuf, 3))
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
return TransferBufferToString(cx, sb, rval);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
Decode(JSContext *cx, JSString *str, const jschar *reservedSet, Value *rval)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t length = str->length();
|
|
|
|
const jschar *chars = str->getChars(cx);
|
|
|
|
if (!chars)
|
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (length == 0) {
|
2010-07-14 23:19:36 -07:00
|
|
|
rval->setString(cx->runtime->emptyString);
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
StringBuffer sb(cx);
|
2010-12-06 10:26:58 -08:00
|
|
|
for (size_t k = 0; k < length; k++) {
|
|
|
|
jschar c = chars[k];
|
2007-03-22 10:30:00 -07:00
|
|
|
if (c == '%') {
|
2010-12-06 10:26:58 -08:00
|
|
|
size_t start = k;
|
2007-03-22 10:30:00 -07:00
|
|
|
if ((k + 2) >= length)
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!JS7_ISHEX(chars[k+1]) || !JS7_ISHEX(chars[k+2]))
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2010-12-06 10:26:58 -08:00
|
|
|
jsuint B = JS7_UNHEX(chars[k+1]) * 16 + JS7_UNHEX(chars[k+2]);
|
2007-03-22 10:30:00 -07:00
|
|
|
k += 2;
|
|
|
|
if (!(B & 0x80)) {
|
|
|
|
c = (jschar)B;
|
|
|
|
} else {
|
2010-12-06 10:26:58 -08:00
|
|
|
intN n = 1;
|
2007-03-22 10:30:00 -07:00
|
|
|
while (B & (0x80 >> n))
|
|
|
|
n++;
|
2009-10-01 17:10:14 -07:00
|
|
|
if (n == 1 || n > 4)
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2010-12-06 10:26:58 -08:00
|
|
|
uint8 octets[4];
|
2007-03-22 10:30:00 -07:00
|
|
|
octets[0] = (uint8)B;
|
|
|
|
if (k + 3 * (n - 1) >= length)
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2010-12-06 10:26:58 -08:00
|
|
|
for (intN j = 1; j < n; j++) {
|
2007-03-22 10:30:00 -07:00
|
|
|
k++;
|
|
|
|
if (chars[k] != '%')
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!JS7_ISHEX(chars[k+1]) || !JS7_ISHEX(chars[k+2]))
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2007-03-22 10:30:00 -07:00
|
|
|
B = JS7_UNHEX(chars[k+1]) * 16 + JS7_UNHEX(chars[k+2]);
|
|
|
|
if ((B & 0xC0) != 0x80)
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2007-03-22 10:30:00 -07:00
|
|
|
k += 2;
|
|
|
|
octets[j] = (char)B;
|
|
|
|
}
|
2010-12-06 10:26:58 -08:00
|
|
|
uint32 v = Utf8ToOneUcs4Char(octets, n);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (v >= 0x10000) {
|
|
|
|
v -= 0x10000;
|
|
|
|
if (v > 0xFFFFF)
|
2007-08-15 23:23:06 -07:00
|
|
|
goto report_bad_uri;
|
2007-03-22 10:30:00 -07:00
|
|
|
c = (jschar)((v & 0x3FF) + 0xDC00);
|
2010-12-06 10:26:58 -08:00
|
|
|
jschar H = (jschar)((v >> 10) + 0xD800);
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(H))
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
|
|
|
c = (jschar)v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (js_strchr(reservedSet, c)) {
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(chars + start, k - start + 1))
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
} else {
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(c))
|
2009-08-14 16:10:59 -07:00
|
|
|
return JS_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
} else {
|
2011-01-12 15:28:58 -08:00
|
|
|
if (!sb.append(c))
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-12 15:28:58 -08:00
|
|
|
return TransferBufferToString(cx, sb, rval);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-08-15 23:23:06 -07:00
|
|
|
report_bad_uri:
|
2007-03-22 10:30:00 -07:00
|
|
|
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_URI);
|
2007-08-15 23:23:06 -07:00
|
|
|
/* FALL THROUGH */
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_decodeURI(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
2007-08-01 21:33:52 -07:00
|
|
|
return Decode(cx, str, js_uriReservedPlusPound_ucstr, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_decodeURI_Component(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
2007-08-01 21:33:52 -07:00
|
|
|
return Decode(cx, str, js_empty_ucstr, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_encodeURI(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
|
|
|
return Encode(cx, str, js_uriReservedPlusPound_ucstr, js_uriUnescaped_ucstr,
|
2007-08-01 21:33:52 -07:00
|
|
|
vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSBool
|
2010-07-14 23:19:36 -07:00
|
|
|
str_encodeURI_Component(JSContext *cx, uintN argc, Value *vp)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-12-06 10:26:58 -08:00
|
|
|
JSLinearString *str = ArgToRootedString(cx, argc, vp, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!str)
|
|
|
|
return JS_FALSE;
|
2007-08-01 21:33:52 -07:00
|
|
|
return Encode(cx, str, js_uriUnescaped_ucstr, NULL, vp);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert one UCS-4 char and write it into a UTF-8 buffer, which must be at
|
2009-10-01 17:10:14 -07:00
|
|
|
* least 4 bytes long. Return the number of UTF-8 bytes of data written.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
|
|
|
int
|
|
|
|
js_OneUcs4ToUtf8Char(uint8 *utf8Buffer, uint32 ucs4Char)
|
|
|
|
{
|
|
|
|
int utf8Length = 1;
|
|
|
|
|
2009-10-01 17:10:14 -07:00
|
|
|
JS_ASSERT(ucs4Char <= 0x10FFFF);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (ucs4Char < 0x80) {
|
|
|
|
*utf8Buffer = (uint8)ucs4Char;
|
|
|
|
} else {
|
|
|
|
int i;
|
|
|
|
uint32 a = ucs4Char >> 11;
|
|
|
|
utf8Length = 2;
|
|
|
|
while (a) {
|
|
|
|
a >>= 5;
|
|
|
|
utf8Length++;
|
|
|
|
}
|
|
|
|
i = utf8Length;
|
|
|
|
while (--i) {
|
|
|
|
utf8Buffer[i] = (uint8)((ucs4Char & 0x3F) | 0x80);
|
|
|
|
ucs4Char >>= 6;
|
|
|
|
}
|
|
|
|
*utf8Buffer = (uint8)(0x100 - (1 << (8-utf8Length)) + ucs4Char);
|
|
|
|
}
|
|
|
|
return utf8Length;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert a utf8 character sequence into a UCS-4 character and return that
|
|
|
|
* character. It is assumed that the caller already checked that the sequence
|
|
|
|
* is valid.
|
|
|
|
*/
|
|
|
|
static uint32
|
|
|
|
Utf8ToOneUcs4Char(const uint8 *utf8Buffer, int utf8Length)
|
|
|
|
{
|
2011-07-15 15:41:47 -07:00
|
|
|
JS_ASSERT(1 <= utf8Length && utf8Length <= 4);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (utf8Length == 1) {
|
2011-07-15 15:41:47 -07:00
|
|
|
JS_ASSERT(!(*utf8Buffer & 0x80));
|
|
|
|
return *utf8Buffer;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2011-07-15 15:41:47 -07:00
|
|
|
|
|
|
|
/* from Unicode 3.1, non-shortest form is illegal */
|
|
|
|
static const uint32 minucs4Table[] = { 0x80, 0x800, 0x10000 };
|
|
|
|
|
|
|
|
JS_ASSERT((*utf8Buffer & (0x100 - (1 << (7 - utf8Length)))) ==
|
|
|
|
(0x100 - (1 << (8 - utf8Length))));
|
|
|
|
uint32 ucs4Char = *utf8Buffer++ & ((1 << (7 - utf8Length)) - 1);
|
|
|
|
uint32 minucs4Char = minucs4Table[utf8Length - 2];
|
|
|
|
while (--utf8Length) {
|
|
|
|
JS_ASSERT((*utf8Buffer & 0xC0) == 0x80);
|
|
|
|
ucs4Char = (ucs4Char << 6) | (*utf8Buffer++ & 0x3F);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (JS_UNLIKELY(ucs4Char < minucs4Char || (ucs4Char >= 0xD800 && ucs4Char <= 0xDFFF)))
|
2011-07-15 14:15:55 -07:00
|
|
|
return INVALID_UTF8;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
return ucs4Char;
|
|
|
|
}
|
|
|
|
|
2010-10-28 08:15:53 -07:00
|
|
|
namespace js {
|
2007-08-11 22:10:02 -07:00
|
|
|
|
2010-10-28 08:15:53 -07:00
|
|
|
size_t
|
2010-12-06 10:26:58 -08:00
|
|
|
PutEscapedStringImpl(char *buffer, size_t bufferSize, FILE *fp, JSLinearString *str, uint32 quote)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
enum {
|
|
|
|
STOP, FIRST_QUOTE, LAST_QUOTE, CHARS, ESCAPE_START, ESCAPE_MORE
|
|
|
|
} state;
|
|
|
|
|
|
|
|
JS_ASSERT(quote == 0 || quote == '\'' || quote == '"');
|
|
|
|
JS_ASSERT_IF(!buffer, bufferSize == 0);
|
|
|
|
JS_ASSERT_IF(fp, !buffer);
|
|
|
|
|
2010-10-28 08:15:53 -07:00
|
|
|
if (bufferSize == 0)
|
|
|
|
buffer = NULL;
|
|
|
|
else
|
|
|
|
bufferSize--;
|
|
|
|
|
2010-12-06 10:26:58 -08:00
|
|
|
const jschar *chars = str->chars();
|
|
|
|
const jschar *charsEnd = chars + str->length();
|
|
|
|
size_t n = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
state = FIRST_QUOTE;
|
2010-12-06 10:26:58 -08:00
|
|
|
uintN shift = 0;
|
|
|
|
uintN hex = 0;
|
|
|
|
uintN u = 0;
|
|
|
|
char c = 0; /* to quell GCC warnings */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
switch (state) {
|
|
|
|
case STOP:
|
|
|
|
goto stop;
|
|
|
|
case FIRST_QUOTE:
|
|
|
|
state = CHARS;
|
|
|
|
goto do_quote;
|
|
|
|
case LAST_QUOTE:
|
|
|
|
state = STOP;
|
|
|
|
do_quote:
|
|
|
|
if (quote == 0)
|
|
|
|
continue;
|
|
|
|
c = (char)quote;
|
|
|
|
break;
|
|
|
|
case CHARS:
|
|
|
|
if (chars == charsEnd) {
|
|
|
|
state = LAST_QUOTE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
u = *chars++;
|
|
|
|
if (u < ' ') {
|
2007-04-25 15:30:35 -07:00
|
|
|
if (u != 0) {
|
2010-12-06 10:26:58 -08:00
|
|
|
const char *escape = strchr(js_EscapeMap, (int)u);
|
2007-04-25 15:30:35 -07:00
|
|
|
if (escape) {
|
|
|
|
u = escape[1];
|
|
|
|
goto do_escape;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
goto do_hex_escape;
|
|
|
|
}
|
|
|
|
if (u < 127) {
|
|
|
|
if (u == quote || u == '\\')
|
|
|
|
goto do_escape;
|
|
|
|
c = (char)u;
|
|
|
|
} else if (u < 0x100) {
|
|
|
|
goto do_hex_escape;
|
|
|
|
} else {
|
|
|
|
shift = 16;
|
|
|
|
hex = u;
|
|
|
|
u = 'u';
|
|
|
|
goto do_escape;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
do_hex_escape:
|
|
|
|
shift = 8;
|
|
|
|
hex = u;
|
|
|
|
u = 'x';
|
|
|
|
do_escape:
|
|
|
|
c = '\\';
|
|
|
|
state = ESCAPE_START;
|
|
|
|
break;
|
|
|
|
case ESCAPE_START:
|
|
|
|
JS_ASSERT(' ' <= u && u < 127);
|
|
|
|
c = (char)u;
|
|
|
|
state = ESCAPE_MORE;
|
|
|
|
break;
|
|
|
|
case ESCAPE_MORE:
|
|
|
|
if (shift == 0) {
|
|
|
|
state = CHARS;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
shift -= 4;
|
|
|
|
u = 0xF & (hex >> shift);
|
|
|
|
c = (char)(u + (u < 10 ? '0' : 'A' - 10));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (buffer) {
|
2010-10-28 08:15:53 -07:00
|
|
|
JS_ASSERT(n <= bufferSize);
|
|
|
|
if (n != bufferSize) {
|
|
|
|
buffer[n] = c;
|
|
|
|
} else {
|
|
|
|
buffer[n] = '\0';
|
|
|
|
buffer = NULL;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
} else if (fp) {
|
2010-10-28 08:15:53 -07:00
|
|
|
if (fputc(c, fp) < 0)
|
|
|
|
return size_t(-1);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
stop:
|
|
|
|
if (buffer)
|
|
|
|
buffer[n] = '\0';
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2010-10-28 08:15:53 -07:00
|
|
|
} /* namespace js */
|