jbarnette-johnson 1.0.0.200806240111 → 1.0.0.200807291507
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/MANIFEST +1 -0
- data/Rakefile +3 -10
- data/bin/johnson +2 -1
- data/ext/spidermonkey/context.c +3 -4
- data/ext/spidermonkey/context.h +1 -1
- data/ext/spidermonkey/conversions.c +39 -33
- data/ext/spidermonkey/debugger.c +5 -5
- data/ext/spidermonkey/immutable_node.c.erb +11 -11
- data/ext/spidermonkey/jroot.h +4 -4
- data/ext/spidermonkey/js_land_proxy.c +9 -8
- data/ext/spidermonkey/ruby_land_proxy.c +5 -4
- data/ext/spidermonkey/runtime.c +1 -1
- data/johnson.gemspec +36 -0
- data/lib/hoe.rb +0 -7
- data/lib/johnson/cli/options.rb +10 -4
- data/lib/johnson/spidermonkey/runtime.rb +2 -2
- data/lib/johnson/version.rb +4 -2
- data/lib/johnson.rb +1 -0
- data/test/johnson/runtime_test.rb +11 -0
- data/test/johnson/spidermonkey/ruby_land_proxy_test.rb +6 -0
- data/vendor/spidermonkey/.cvsignore +9 -0
- data/vendor/spidermonkey/Makefile.in +462 -0
- data/vendor/spidermonkey/Makefile.ref +364 -0
- data/vendor/spidermonkey/README.html +820 -0
- data/vendor/spidermonkey/SpiderMonkey.rsp +12 -0
- data/vendor/spidermonkey/Y.js +19 -0
- data/vendor/spidermonkey/build.mk +43 -0
- data/vendor/spidermonkey/config/AIX4.1.mk +65 -0
- data/vendor/spidermonkey/config/AIX4.2.mk +64 -0
- data/vendor/spidermonkey/config/AIX4.3.mk +65 -0
- data/vendor/spidermonkey/config/Darwin.mk +83 -0
- data/vendor/spidermonkey/config/Darwin1.3.mk +81 -0
- data/vendor/spidermonkey/config/Darwin1.4.mk +41 -0
- data/vendor/spidermonkey/config/Darwin5.2.mk +81 -0
- data/vendor/spidermonkey/config/Darwin5.3.mk +81 -0
- data/vendor/spidermonkey/config/HP-UXB.10.10.mk +77 -0
- data/vendor/spidermonkey/config/HP-UXB.10.20.mk +77 -0
- data/vendor/spidermonkey/config/HP-UXB.11.00.mk +80 -0
- data/vendor/spidermonkey/config/IRIX.mk +87 -0
- data/vendor/spidermonkey/config/IRIX5.3.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.1.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.2.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.3.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.5.mk +44 -0
- data/vendor/spidermonkey/config/Linux_All.mk +103 -0
- data/vendor/spidermonkey/config/Mac_OS10.0.mk +82 -0
- data/vendor/spidermonkey/config/OSF1V4.0.mk +72 -0
- data/vendor/spidermonkey/config/OSF1V5.0.mk +69 -0
- data/vendor/spidermonkey/config/SunOS4.1.4.mk +101 -0
- data/vendor/spidermonkey/config/SunOS5.10.mk +50 -0
- data/vendor/spidermonkey/config/SunOS5.3.mk +91 -0
- data/vendor/spidermonkey/config/SunOS5.4.mk +92 -0
- data/vendor/spidermonkey/config/SunOS5.5.1.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.5.mk +87 -0
- data/vendor/spidermonkey/config/SunOS5.6.mk +89 -0
- data/vendor/spidermonkey/config/SunOS5.7.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.8.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.9.mk +44 -0
- data/vendor/spidermonkey/config/WINNT4.0.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.0.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.1.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.2.mk +117 -0
- data/vendor/spidermonkey/config/WINNT6.0.mk +117 -0
- data/vendor/spidermonkey/config/dgux.mk +64 -0
- data/vendor/spidermonkey/config.mk +192 -0
- data/vendor/spidermonkey/editline/Makefile.ref +144 -0
- data/vendor/spidermonkey/editline/README +83 -0
- data/vendor/spidermonkey/editline/editline.3 +175 -0
- data/vendor/spidermonkey/editline/editline.c +1369 -0
- data/vendor/spidermonkey/editline/editline.h +135 -0
- data/vendor/spidermonkey/editline/sysunix.c +182 -0
- data/vendor/spidermonkey/editline/unix.h +82 -0
- data/vendor/spidermonkey/fdlibm/.cvsignore +7 -0
- data/vendor/spidermonkey/fdlibm/Makefile.in +127 -0
- data/vendor/spidermonkey/fdlibm/Makefile.ref +192 -0
- data/vendor/spidermonkey/fdlibm/e_acos.c +147 -0
- data/vendor/spidermonkey/fdlibm/e_acosh.c +105 -0
- data/vendor/spidermonkey/fdlibm/e_asin.c +156 -0
- data/vendor/spidermonkey/fdlibm/e_atan2.c +165 -0
- data/vendor/spidermonkey/fdlibm/e_atanh.c +110 -0
- data/vendor/spidermonkey/fdlibm/e_cosh.c +133 -0
- data/vendor/spidermonkey/fdlibm/e_exp.c +202 -0
- data/vendor/spidermonkey/fdlibm/e_fmod.c +184 -0
- data/vendor/spidermonkey/fdlibm/e_gamma.c +71 -0
- data/vendor/spidermonkey/fdlibm/e_gamma_r.c +70 -0
- data/vendor/spidermonkey/fdlibm/e_hypot.c +173 -0
- data/vendor/spidermonkey/fdlibm/e_j0.c +524 -0
- data/vendor/spidermonkey/fdlibm/e_j1.c +523 -0
- data/vendor/spidermonkey/fdlibm/e_jn.c +315 -0
- data/vendor/spidermonkey/fdlibm/e_lgamma.c +71 -0
- data/vendor/spidermonkey/fdlibm/e_lgamma_r.c +347 -0
- data/vendor/spidermonkey/fdlibm/e_log.c +184 -0
- data/vendor/spidermonkey/fdlibm/e_log10.c +134 -0
- data/vendor/spidermonkey/fdlibm/e_pow.c +386 -0
- data/vendor/spidermonkey/fdlibm/e_rem_pio2.c +222 -0
- data/vendor/spidermonkey/fdlibm/e_remainder.c +120 -0
- data/vendor/spidermonkey/fdlibm/e_scalb.c +89 -0
- data/vendor/spidermonkey/fdlibm/e_sinh.c +122 -0
- data/vendor/spidermonkey/fdlibm/e_sqrt.c +497 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.h +273 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.mak +1453 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.mdp +0 -0
- data/vendor/spidermonkey/fdlibm/k_cos.c +135 -0
- data/vendor/spidermonkey/fdlibm/k_rem_pio2.c +354 -0
- data/vendor/spidermonkey/fdlibm/k_sin.c +114 -0
- data/vendor/spidermonkey/fdlibm/k_standard.c +785 -0
- data/vendor/spidermonkey/fdlibm/k_tan.c +170 -0
- data/vendor/spidermonkey/fdlibm/s_asinh.c +101 -0
- data/vendor/spidermonkey/fdlibm/s_atan.c +175 -0
- data/vendor/spidermonkey/fdlibm/s_cbrt.c +133 -0
- data/vendor/spidermonkey/fdlibm/s_ceil.c +120 -0
- data/vendor/spidermonkey/fdlibm/s_copysign.c +72 -0
- data/vendor/spidermonkey/fdlibm/s_cos.c +118 -0
- data/vendor/spidermonkey/fdlibm/s_erf.c +356 -0
- data/vendor/spidermonkey/fdlibm/s_expm1.c +267 -0
- data/vendor/spidermonkey/fdlibm/s_fabs.c +70 -0
- data/vendor/spidermonkey/fdlibm/s_finite.c +71 -0
- data/vendor/spidermonkey/fdlibm/s_floor.c +121 -0
- data/vendor/spidermonkey/fdlibm/s_frexp.c +99 -0
- data/vendor/spidermonkey/fdlibm/s_ilogb.c +85 -0
- data/vendor/spidermonkey/fdlibm/s_isnan.c +74 -0
- data/vendor/spidermonkey/fdlibm/s_ldexp.c +66 -0
- data/vendor/spidermonkey/fdlibm/s_lib_version.c +73 -0
- data/vendor/spidermonkey/fdlibm/s_log1p.c +211 -0
- data/vendor/spidermonkey/fdlibm/s_logb.c +79 -0
- data/vendor/spidermonkey/fdlibm/s_matherr.c +64 -0
- data/vendor/spidermonkey/fdlibm/s_modf.c +132 -0
- data/vendor/spidermonkey/fdlibm/s_nextafter.c +124 -0
- data/vendor/spidermonkey/fdlibm/s_rint.c +131 -0
- data/vendor/spidermonkey/fdlibm/s_scalbn.c +107 -0
- data/vendor/spidermonkey/fdlibm/s_signgam.c +40 -0
- data/vendor/spidermonkey/fdlibm/s_significand.c +68 -0
- data/vendor/spidermonkey/fdlibm/s_sin.c +118 -0
- data/vendor/spidermonkey/fdlibm/s_tan.c +112 -0
- data/vendor/spidermonkey/fdlibm/s_tanh.c +122 -0
- data/vendor/spidermonkey/fdlibm/w_acos.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_acosh.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_asin.c +80 -0
- data/vendor/spidermonkey/fdlibm/w_atan2.c +79 -0
- data/vendor/spidermonkey/fdlibm/w_atanh.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_cosh.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_exp.c +88 -0
- data/vendor/spidermonkey/fdlibm/w_fmod.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_gamma.c +85 -0
- data/vendor/spidermonkey/fdlibm/w_gamma_r.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_hypot.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_j0.c +105 -0
- data/vendor/spidermonkey/fdlibm/w_j1.c +106 -0
- data/vendor/spidermonkey/fdlibm/w_jn.c +128 -0
- data/vendor/spidermonkey/fdlibm/w_lgamma.c +85 -0
- data/vendor/spidermonkey/fdlibm/w_lgamma_r.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_log.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_log10.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_pow.c +99 -0
- data/vendor/spidermonkey/fdlibm/w_remainder.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_scalb.c +95 -0
- data/vendor/spidermonkey/fdlibm/w_sinh.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_sqrt.c +77 -0
- data/vendor/spidermonkey/javascript-trace.d +73 -0
- data/vendor/spidermonkey/js.c +3951 -0
- data/vendor/spidermonkey/js.mak +4438 -0
- data/vendor/spidermonkey/js.mdp +0 -0
- data/vendor/spidermonkey/js.msg +307 -0
- data/vendor/spidermonkey/js.pkg +2 -0
- data/vendor/spidermonkey/js3240.rc +79 -0
- data/vendor/spidermonkey/jsOS240.def +654 -0
- data/vendor/spidermonkey/jsapi.c +5836 -0
- data/vendor/spidermonkey/jsapi.h +2624 -0
- data/vendor/spidermonkey/jsarena.c +450 -0
- data/vendor/spidermonkey/jsarena.h +318 -0
- data/vendor/spidermonkey/jsarray.c +2988 -0
- data/vendor/spidermonkey/jsarray.h +124 -0
- data/vendor/spidermonkey/jsatom.c +1045 -0
- data/vendor/spidermonkey/jsatom.h +442 -0
- data/vendor/spidermonkey/jsbit.h +253 -0
- data/vendor/spidermonkey/jsbool.c +176 -0
- data/vendor/spidermonkey/jsbool.h +73 -0
- data/vendor/spidermonkey/jsclist.h +139 -0
- data/vendor/spidermonkey/jscntxt.c +1348 -0
- data/vendor/spidermonkey/jscntxt.h +1120 -0
- data/vendor/spidermonkey/jscompat.h +57 -0
- data/vendor/spidermonkey/jsconfig.h +248 -0
- data/vendor/spidermonkey/jsconfig.mk +181 -0
- data/vendor/spidermonkey/jscpucfg.c +383 -0
- data/vendor/spidermonkey/jscpucfg.h +212 -0
- data/vendor/spidermonkey/jsdate.c +2398 -0
- data/vendor/spidermonkey/jsdate.h +124 -0
- data/vendor/spidermonkey/jsdbgapi.c +1799 -0
- data/vendor/spidermonkey/jsdbgapi.h +464 -0
- data/vendor/spidermonkey/jsdhash.c +868 -0
- data/vendor/spidermonkey/jsdhash.h +592 -0
- data/vendor/spidermonkey/jsdtoa.c +3167 -0
- data/vendor/spidermonkey/jsdtoa.h +130 -0
- data/vendor/spidermonkey/jsdtracef.c +317 -0
- data/vendor/spidermonkey/jsdtracef.h +77 -0
- data/vendor/spidermonkey/jsemit.c +6909 -0
- data/vendor/spidermonkey/jsemit.h +741 -0
- data/vendor/spidermonkey/jsexn.c +1371 -0
- data/vendor/spidermonkey/jsexn.h +96 -0
- data/vendor/spidermonkey/jsfile.c +2736 -0
- data/vendor/spidermonkey/jsfile.h +56 -0
- data/vendor/spidermonkey/jsfile.msg +90 -0
- data/vendor/spidermonkey/jsfun.c +2634 -0
- data/vendor/spidermonkey/jsfun.h +254 -0
- data/vendor/spidermonkey/jsgc.c +3554 -0
- data/vendor/spidermonkey/jsgc.h +403 -0
- data/vendor/spidermonkey/jshash.c +476 -0
- data/vendor/spidermonkey/jshash.h +151 -0
- data/vendor/spidermonkey/jsify.pl +485 -0
- data/vendor/spidermonkey/jsinterp.c +6981 -0
- data/vendor/spidermonkey/jsinterp.h +521 -0
- data/vendor/spidermonkey/jsinvoke.c +43 -0
- data/vendor/spidermonkey/jsiter.c +1067 -0
- data/vendor/spidermonkey/jsiter.h +122 -0
- data/vendor/spidermonkey/jskeyword.tbl +124 -0
- data/vendor/spidermonkey/jskwgen.c +460 -0
- data/vendor/spidermonkey/jslibmath.h +266 -0
- data/vendor/spidermonkey/jslock.c +1309 -0
- data/vendor/spidermonkey/jslock.h +313 -0
- data/vendor/spidermonkey/jslocko.asm +60 -0
- data/vendor/spidermonkey/jslog2.c +94 -0
- data/vendor/spidermonkey/jslong.c +264 -0
- data/vendor/spidermonkey/jslong.h +412 -0
- data/vendor/spidermonkey/jsmath.c +568 -0
- data/vendor/spidermonkey/jsmath.h +57 -0
- data/vendor/spidermonkey/jsnum.c +1228 -0
- data/vendor/spidermonkey/jsnum.h +283 -0
- data/vendor/spidermonkey/jsobj.c +5266 -0
- data/vendor/spidermonkey/jsobj.h +709 -0
- data/vendor/spidermonkey/jsopcode.c +5245 -0
- data/vendor/spidermonkey/jsopcode.h +394 -0
- data/vendor/spidermonkey/jsopcode.tbl +523 -0
- data/vendor/spidermonkey/jsotypes.h +202 -0
- data/vendor/spidermonkey/jsparse.c +6680 -0
- data/vendor/spidermonkey/jsparse.h +511 -0
- data/vendor/spidermonkey/jsprf.c +1262 -0
- data/vendor/spidermonkey/jsprf.h +150 -0
- data/vendor/spidermonkey/jsproto.tbl +128 -0
- data/vendor/spidermonkey/jsprvtd.h +267 -0
- data/vendor/spidermonkey/jspubtd.h +744 -0
- data/vendor/spidermonkey/jsregexp.c +4352 -0
- data/vendor/spidermonkey/jsregexp.h +183 -0
- data/vendor/spidermonkey/jsreops.tbl +145 -0
- data/vendor/spidermonkey/jsscan.c +2003 -0
- data/vendor/spidermonkey/jsscan.h +387 -0
- data/vendor/spidermonkey/jsscope.c +1948 -0
- data/vendor/spidermonkey/jsscope.h +418 -0
- data/vendor/spidermonkey/jsscript.c +1832 -0
- data/vendor/spidermonkey/jsscript.h +287 -0
- data/vendor/spidermonkey/jsshell.msg +50 -0
- data/vendor/spidermonkey/jsstddef.h +83 -0
- data/vendor/spidermonkey/jsstr.c +5004 -0
- data/vendor/spidermonkey/jsstr.h +641 -0
- data/vendor/spidermonkey/jstypes.h +475 -0
- data/vendor/spidermonkey/jsutil.c +345 -0
- data/vendor/spidermonkey/jsutil.h +157 -0
- data/vendor/spidermonkey/jsxdrapi.c +800 -0
- data/vendor/spidermonkey/jsxdrapi.h +218 -0
- data/vendor/spidermonkey/jsxml.c +8471 -0
- data/vendor/spidermonkey/jsxml.h +349 -0
- data/vendor/spidermonkey/lock_SunOS.s +119 -0
- data/vendor/spidermonkey/perfect.js +39 -0
- data/vendor/spidermonkey/plify_jsdhash.sed +36 -0
- data/vendor/spidermonkey/prmjtime.c +846 -0
- data/vendor/spidermonkey/prmjtime.h +103 -0
- data/vendor/spidermonkey/resource.h +15 -0
- data/vendor/spidermonkey/rules.mk +197 -0
- data/vendor/spidermonkey/win32.order +384 -0
- metadata +4 -3
@@ -0,0 +1,176 @@
|
|
1
|
+
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2
|
+
*
|
3
|
+
* ***** BEGIN LICENSE BLOCK *****
|
4
|
+
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
5
|
+
*
|
6
|
+
* The contents of this file are subject to the Mozilla Public License Version
|
7
|
+
* 1.1 (the "License"); you may not use this file except in compliance with
|
8
|
+
* the License. You may obtain a copy of the License at
|
9
|
+
* http://www.mozilla.org/MPL/
|
10
|
+
*
|
11
|
+
* Software distributed under the License is distributed on an "AS IS" basis,
|
12
|
+
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
13
|
+
* for the specific language governing rights and limitations under the
|
14
|
+
* License.
|
15
|
+
*
|
16
|
+
* The Original Code is Mozilla Communicator client code, released
|
17
|
+
* March 31, 1998.
|
18
|
+
*
|
19
|
+
* The Initial Developer of the Original Code is
|
20
|
+
* Netscape Communications Corporation.
|
21
|
+
* Portions created by the Initial Developer are Copyright (C) 1998
|
22
|
+
* the Initial Developer. All Rights Reserved.
|
23
|
+
*
|
24
|
+
* Contributor(s):
|
25
|
+
*
|
26
|
+
* Alternatively, the contents of this file may be used under the terms of
|
27
|
+
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
28
|
+
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
29
|
+
* in which case the provisions of the GPL or the LGPL are applicable instead
|
30
|
+
* of those above. If you wish to allow use of your version of this file only
|
31
|
+
* under the terms of either the GPL or the LGPL, and not to allow others to
|
32
|
+
* use your version of this file under the terms of the MPL, indicate your
|
33
|
+
* decision by deleting the provisions above and replace them with the notice
|
34
|
+
* and other provisions required by the GPL or the LGPL. If you do not delete
|
35
|
+
* the provisions above, a recipient may use your version of this file under
|
36
|
+
* the terms of any one of the MPL, the GPL or the LGPL.
|
37
|
+
*
|
38
|
+
* ***** END LICENSE BLOCK ***** */
|
39
|
+
|
40
|
+
/*
|
41
|
+
* JS boolean implementation.
|
42
|
+
*/
|
43
|
+
#include "jsstddef.h"
|
44
|
+
#include "jstypes.h"
|
45
|
+
#include "jsutil.h" /* Added by JSIFY */
|
46
|
+
#include "jsapi.h"
|
47
|
+
#include "jsatom.h"
|
48
|
+
#include "jsbool.h"
|
49
|
+
#include "jscntxt.h"
|
50
|
+
#include "jsconfig.h"
|
51
|
+
#include "jsinterp.h"
|
52
|
+
#include "jslock.h"
|
53
|
+
#include "jsnum.h"
|
54
|
+
#include "jsobj.h"
|
55
|
+
#include "jsstr.h"
|
56
|
+
|
57
|
+
JSClass js_BooleanClass = {
|
58
|
+
"Boolean",
|
59
|
+
JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_Boolean),
|
60
|
+
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
|
61
|
+
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, JS_FinalizeStub,
|
62
|
+
JSCLASS_NO_OPTIONAL_MEMBERS
|
63
|
+
};
|
64
|
+
|
65
|
+
#if JS_HAS_TOSOURCE
|
66
|
+
#include "jsprf.h"
|
67
|
+
|
68
|
+
static JSBool
|
69
|
+
bool_toSource(JSContext *cx, uintN argc, jsval *vp)
|
70
|
+
{
|
71
|
+
jsval v;
|
72
|
+
char buf[32];
|
73
|
+
JSString *str;
|
74
|
+
|
75
|
+
if (!js_GetPrimitiveThis(cx, vp, &js_BooleanClass, &v))
|
76
|
+
return JS_FALSE;
|
77
|
+
JS_ASSERT(JSVAL_IS_BOOLEAN(v));
|
78
|
+
JS_snprintf(buf, sizeof buf, "(new %s(%s))",
|
79
|
+
js_BooleanClass.name,
|
80
|
+
JS_BOOLEAN_STR(JSVAL_TO_BOOLEAN(v)));
|
81
|
+
str = JS_NewStringCopyZ(cx, buf);
|
82
|
+
if (!str)
|
83
|
+
return JS_FALSE;
|
84
|
+
*vp = STRING_TO_JSVAL(str);
|
85
|
+
return JS_TRUE;
|
86
|
+
}
|
87
|
+
#endif
|
88
|
+
|
89
|
+
static JSBool
|
90
|
+
bool_toString(JSContext *cx, uintN argc, jsval *vp)
|
91
|
+
{
|
92
|
+
jsval v;
|
93
|
+
JSAtom *atom;
|
94
|
+
JSString *str;
|
95
|
+
|
96
|
+
if (!js_GetPrimitiveThis(cx, vp, &js_BooleanClass, &v))
|
97
|
+
return JS_FALSE;
|
98
|
+
JS_ASSERT(JSVAL_IS_BOOLEAN(v));
|
99
|
+
atom = cx->runtime->atomState.booleanAtoms[JSVAL_TO_BOOLEAN(v) ? 1 : 0];
|
100
|
+
str = ATOM_TO_STRING(atom);
|
101
|
+
if (!str)
|
102
|
+
return JS_FALSE;
|
103
|
+
*vp = STRING_TO_JSVAL(str);
|
104
|
+
return JS_TRUE;
|
105
|
+
}
|
106
|
+
|
107
|
+
static JSBool
|
108
|
+
bool_valueOf(JSContext *cx, uintN argc, jsval *vp)
|
109
|
+
{
|
110
|
+
return js_GetPrimitiveThis(cx, vp, &js_BooleanClass, vp);
|
111
|
+
}
|
112
|
+
|
113
|
+
static JSFunctionSpec boolean_methods[] = {
|
114
|
+
#if JS_HAS_TOSOURCE
|
115
|
+
JS_FN(js_toSource_str, bool_toSource, 0, 0, JSFUN_THISP_BOOLEAN),
|
116
|
+
#endif
|
117
|
+
JS_FN(js_toString_str, bool_toString, 0, 0, JSFUN_THISP_BOOLEAN),
|
118
|
+
JS_FN(js_valueOf_str, bool_valueOf, 0, 0, JSFUN_THISP_BOOLEAN),
|
119
|
+
JS_FS_END
|
120
|
+
};
|
121
|
+
|
122
|
+
static JSBool
|
123
|
+
Boolean(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
124
|
+
{
|
125
|
+
jsval bval;
|
126
|
+
|
127
|
+
bval = (argc != 0)
|
128
|
+
? BOOLEAN_TO_JSVAL(js_ValueToBoolean(argv[0]))
|
129
|
+
: JSVAL_FALSE;
|
130
|
+
if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) {
|
131
|
+
*rval = bval;
|
132
|
+
return JS_TRUE;
|
133
|
+
}
|
134
|
+
STOBJ_SET_SLOT(obj, JSSLOT_PRIVATE, bval);
|
135
|
+
return JS_TRUE;
|
136
|
+
}
|
137
|
+
|
138
|
+
JSObject *
|
139
|
+
js_InitBooleanClass(JSContext *cx, JSObject *obj)
|
140
|
+
{
|
141
|
+
JSObject *proto;
|
142
|
+
|
143
|
+
proto = JS_InitClass(cx, obj, NULL, &js_BooleanClass, Boolean, 1,
|
144
|
+
NULL, boolean_methods, NULL, NULL);
|
145
|
+
if (!proto)
|
146
|
+
return NULL;
|
147
|
+
STOBJ_SET_SLOT(proto, JSSLOT_PRIVATE, JSVAL_FALSE);
|
148
|
+
return proto;
|
149
|
+
}
|
150
|
+
|
151
|
+
JSString *
|
152
|
+
js_BooleanToString(JSContext *cx, JSBool b)
|
153
|
+
{
|
154
|
+
return ATOM_TO_STRING(cx->runtime->atomState.booleanAtoms[b ? 1 : 0]);
|
155
|
+
}
|
156
|
+
|
157
|
+
JSBool
|
158
|
+
js_ValueToBoolean(jsval v)
|
159
|
+
{
|
160
|
+
if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v))
|
161
|
+
return JS_FALSE;
|
162
|
+
if (JSVAL_IS_OBJECT(v))
|
163
|
+
return JS_TRUE;
|
164
|
+
if (JSVAL_IS_STRING(v))
|
165
|
+
return JSSTRING_LENGTH(JSVAL_TO_STRING(v)) != 0;
|
166
|
+
if (JSVAL_IS_INT(v))
|
167
|
+
return JSVAL_TO_INT(v) != 0;
|
168
|
+
if (JSVAL_IS_DOUBLE(v)) {
|
169
|
+
jsdouble d;
|
170
|
+
|
171
|
+
d = *JSVAL_TO_DOUBLE(v);
|
172
|
+
return !JSDOUBLE_IS_NaN(d) && d != 0;
|
173
|
+
}
|
174
|
+
JS_ASSERT(JSVAL_IS_BOOLEAN(v));
|
175
|
+
return JSVAL_TO_BOOLEAN(v);
|
176
|
+
}
|
@@ -0,0 +1,73 @@
|
|
1
|
+
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2
|
+
*
|
3
|
+
* ***** BEGIN LICENSE BLOCK *****
|
4
|
+
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
5
|
+
*
|
6
|
+
* The contents of this file are subject to the Mozilla Public License Version
|
7
|
+
* 1.1 (the "License"); you may not use this file except in compliance with
|
8
|
+
* the License. You may obtain a copy of the License at
|
9
|
+
* http://www.mozilla.org/MPL/
|
10
|
+
*
|
11
|
+
* Software distributed under the License is distributed on an "AS IS" basis,
|
12
|
+
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
13
|
+
* for the specific language governing rights and limitations under the
|
14
|
+
* License.
|
15
|
+
*
|
16
|
+
* The Original Code is Mozilla Communicator client code, released
|
17
|
+
* March 31, 1998.
|
18
|
+
*
|
19
|
+
* The Initial Developer of the Original Code is
|
20
|
+
* Netscape Communications Corporation.
|
21
|
+
* Portions created by the Initial Developer are Copyright (C) 1998
|
22
|
+
* the Initial Developer. All Rights Reserved.
|
23
|
+
*
|
24
|
+
* Contributor(s):
|
25
|
+
*
|
26
|
+
* Alternatively, the contents of this file may be used under the terms of
|
27
|
+
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
28
|
+
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
29
|
+
* in which case the provisions of the GPL or the LGPL are applicable instead
|
30
|
+
* of those above. If you wish to allow use of your version of this file only
|
31
|
+
* under the terms of either the GPL or the LGPL, and not to allow others to
|
32
|
+
* use your version of this file under the terms of the MPL, indicate your
|
33
|
+
* decision by deleting the provisions above and replace them with the notice
|
34
|
+
* and other provisions required by the GPL or the LGPL. If you do not delete
|
35
|
+
* the provisions above, a recipient may use your version of this file under
|
36
|
+
* the terms of any one of the MPL, the GPL or the LGPL.
|
37
|
+
*
|
38
|
+
* ***** END LICENSE BLOCK ***** */
|
39
|
+
|
40
|
+
#ifndef jsbool_h___
|
41
|
+
#define jsbool_h___
|
42
|
+
/*
|
43
|
+
* JS boolean interface.
|
44
|
+
*/
|
45
|
+
|
46
|
+
JS_BEGIN_EXTERN_C
|
47
|
+
|
48
|
+
/*
|
49
|
+
* Crypto-booleans, not visible to script but used internally by the engine.
|
50
|
+
*
|
51
|
+
* JSVAL_HOLE is a useful value for identifying a hole in an array. It's also
|
52
|
+
* used in the interpreter to represent "no exception pending". In general it
|
53
|
+
* can be used to represent "no value".
|
54
|
+
*
|
55
|
+
* JSVAL_ARETURN is used to throw asynchronous return for generator.close().
|
56
|
+
*/
|
57
|
+
#define JSVAL_HOLE BOOLEAN_TO_JSVAL(2)
|
58
|
+
#define JSVAL_ARETURN BOOLEAN_TO_JSVAL(3)
|
59
|
+
|
60
|
+
extern JSClass js_BooleanClass;
|
61
|
+
|
62
|
+
extern JSObject *
|
63
|
+
js_InitBooleanClass(JSContext *cx, JSObject *obj);
|
64
|
+
|
65
|
+
extern JSString *
|
66
|
+
js_BooleanToString(JSContext *cx, JSBool b);
|
67
|
+
|
68
|
+
extern JSBool
|
69
|
+
js_ValueToBoolean(jsval v);
|
70
|
+
|
71
|
+
JS_END_EXTERN_C
|
72
|
+
|
73
|
+
#endif /* jsbool_h___ */
|
@@ -0,0 +1,139 @@
|
|
1
|
+
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2
|
+
/* ***** BEGIN LICENSE BLOCK *****
|
3
|
+
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
4
|
+
*
|
5
|
+
* The contents of this file are subject to the Mozilla Public License Version
|
6
|
+
* 1.1 (the "License"); you may not use this file except in compliance with
|
7
|
+
* the License. You may obtain a copy of the License at
|
8
|
+
* http://www.mozilla.org/MPL/
|
9
|
+
*
|
10
|
+
* Software distributed under the License is distributed on an "AS IS" basis,
|
11
|
+
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
12
|
+
* for the specific language governing rights and limitations under the
|
13
|
+
* License.
|
14
|
+
*
|
15
|
+
* The Original Code is Mozilla Communicator client code, released
|
16
|
+
* March 31, 1998.
|
17
|
+
*
|
18
|
+
* The Initial Developer of the Original Code is
|
19
|
+
* Netscape Communications Corporation.
|
20
|
+
* Portions created by the Initial Developer are Copyright (C) 1998
|
21
|
+
* the Initial Developer. All Rights Reserved.
|
22
|
+
*
|
23
|
+
* Contributor(s):
|
24
|
+
*
|
25
|
+
* Alternatively, the contents of this file may be used under the terms of
|
26
|
+
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
27
|
+
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
28
|
+
* in which case the provisions of the GPL or the LGPL are applicable instead
|
29
|
+
* of those above. If you wish to allow use of your version of this file only
|
30
|
+
* under the terms of either the GPL or the LGPL, and not to allow others to
|
31
|
+
* use your version of this file under the terms of the MPL, indicate your
|
32
|
+
* decision by deleting the provisions above and replace them with the notice
|
33
|
+
* and other provisions required by the GPL or the LGPL. If you do not delete
|
34
|
+
* the provisions above, a recipient may use your version of this file under
|
35
|
+
* the terms of any one of the MPL, the GPL or the LGPL.
|
36
|
+
*
|
37
|
+
* ***** END LICENSE BLOCK ***** */
|
38
|
+
|
39
|
+
#ifndef jsclist_h___
|
40
|
+
#define jsclist_h___
|
41
|
+
|
42
|
+
#include "jstypes.h"
|
43
|
+
|
44
|
+
/*
|
45
|
+
** Circular linked list
|
46
|
+
*/
|
47
|
+
typedef struct JSCListStr {
|
48
|
+
struct JSCListStr *next;
|
49
|
+
struct JSCListStr *prev;
|
50
|
+
} JSCList;
|
51
|
+
|
52
|
+
/*
|
53
|
+
** Insert element "_e" into the list, before "_l".
|
54
|
+
*/
|
55
|
+
#define JS_INSERT_BEFORE(_e,_l) \
|
56
|
+
JS_BEGIN_MACRO \
|
57
|
+
(_e)->next = (_l); \
|
58
|
+
(_e)->prev = (_l)->prev; \
|
59
|
+
(_l)->prev->next = (_e); \
|
60
|
+
(_l)->prev = (_e); \
|
61
|
+
JS_END_MACRO
|
62
|
+
|
63
|
+
/*
|
64
|
+
** Insert element "_e" into the list, after "_l".
|
65
|
+
*/
|
66
|
+
#define JS_INSERT_AFTER(_e,_l) \
|
67
|
+
JS_BEGIN_MACRO \
|
68
|
+
(_e)->next = (_l)->next; \
|
69
|
+
(_e)->prev = (_l); \
|
70
|
+
(_l)->next->prev = (_e); \
|
71
|
+
(_l)->next = (_e); \
|
72
|
+
JS_END_MACRO
|
73
|
+
|
74
|
+
/*
|
75
|
+
** Return the element following element "_e"
|
76
|
+
*/
|
77
|
+
#define JS_NEXT_LINK(_e) \
|
78
|
+
((_e)->next)
|
79
|
+
/*
|
80
|
+
** Return the element preceding element "_e"
|
81
|
+
*/
|
82
|
+
#define JS_PREV_LINK(_e) \
|
83
|
+
((_e)->prev)
|
84
|
+
|
85
|
+
/*
|
86
|
+
** Append an element "_e" to the end of the list "_l"
|
87
|
+
*/
|
88
|
+
#define JS_APPEND_LINK(_e,_l) JS_INSERT_BEFORE(_e,_l)
|
89
|
+
|
90
|
+
/*
|
91
|
+
** Insert an element "_e" at the head of the list "_l"
|
92
|
+
*/
|
93
|
+
#define JS_INSERT_LINK(_e,_l) JS_INSERT_AFTER(_e,_l)
|
94
|
+
|
95
|
+
/* Return the head/tail of the list */
|
96
|
+
#define JS_LIST_HEAD(_l) (_l)->next
|
97
|
+
#define JS_LIST_TAIL(_l) (_l)->prev
|
98
|
+
|
99
|
+
/*
|
100
|
+
** Remove the element "_e" from it's circular list.
|
101
|
+
*/
|
102
|
+
#define JS_REMOVE_LINK(_e) \
|
103
|
+
JS_BEGIN_MACRO \
|
104
|
+
(_e)->prev->next = (_e)->next; \
|
105
|
+
(_e)->next->prev = (_e)->prev; \
|
106
|
+
JS_END_MACRO
|
107
|
+
|
108
|
+
/*
|
109
|
+
** Remove the element "_e" from it's circular list. Also initializes the
|
110
|
+
** linkage.
|
111
|
+
*/
|
112
|
+
#define JS_REMOVE_AND_INIT_LINK(_e) \
|
113
|
+
JS_BEGIN_MACRO \
|
114
|
+
(_e)->prev->next = (_e)->next; \
|
115
|
+
(_e)->next->prev = (_e)->prev; \
|
116
|
+
(_e)->next = (_e); \
|
117
|
+
(_e)->prev = (_e); \
|
118
|
+
JS_END_MACRO
|
119
|
+
|
120
|
+
/*
|
121
|
+
** Return non-zero if the given circular list "_l" is empty, zero if the
|
122
|
+
** circular list is not empty
|
123
|
+
*/
|
124
|
+
#define JS_CLIST_IS_EMPTY(_l) \
|
125
|
+
((_l)->next == (_l))
|
126
|
+
|
127
|
+
/*
|
128
|
+
** Initialize a circular list
|
129
|
+
*/
|
130
|
+
#define JS_INIT_CLIST(_l) \
|
131
|
+
JS_BEGIN_MACRO \
|
132
|
+
(_l)->next = (_l); \
|
133
|
+
(_l)->prev = (_l); \
|
134
|
+
JS_END_MACRO
|
135
|
+
|
136
|
+
#define JS_INIT_STATIC_CLIST(_l) \
|
137
|
+
{(_l), (_l)}
|
138
|
+
|
139
|
+
#endif /* jsclist_h___ */
|
@@ -0,0 +1,1348 @@
|
|
1
|
+
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2
|
+
* vim: set ts=8 sw=4 et tw=80:
|
3
|
+
*
|
4
|
+
* ***** BEGIN LICENSE BLOCK *****
|
5
|
+
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
6
|
+
*
|
7
|
+
* The contents of this file are subject to the Mozilla Public License Version
|
8
|
+
* 1.1 (the "License"); you may not use this file except in compliance with
|
9
|
+
* the License. You may obtain a copy of the License at
|
10
|
+
* http://www.mozilla.org/MPL/
|
11
|
+
*
|
12
|
+
* Software distributed under the License is distributed on an "AS IS" basis,
|
13
|
+
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
14
|
+
* for the specific language governing rights and limitations under the
|
15
|
+
* License.
|
16
|
+
*
|
17
|
+
* The Original Code is Mozilla Communicator client code, released
|
18
|
+
* March 31, 1998.
|
19
|
+
*
|
20
|
+
* The Initial Developer of the Original Code is
|
21
|
+
* Netscape Communications Corporation.
|
22
|
+
* Portions created by the Initial Developer are Copyright (C) 1998
|
23
|
+
* the Initial Developer. All Rights Reserved.
|
24
|
+
*
|
25
|
+
* Contributor(s):
|
26
|
+
*
|
27
|
+
* Alternatively, the contents of this file may be used under the terms of
|
28
|
+
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
29
|
+
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
30
|
+
* in which case the provisions of the GPL or the LGPL are applicable instead
|
31
|
+
* of those above. If you wish to allow use of your version of this file only
|
32
|
+
* under the terms of either the GPL or the LGPL, and not to allow others to
|
33
|
+
* use your version of this file under the terms of the MPL, indicate your
|
34
|
+
* decision by deleting the provisions above and replace them with the notice
|
35
|
+
* and other provisions required by the GPL or the LGPL. If you do not delete
|
36
|
+
* the provisions above, a recipient may use your version of this file under
|
37
|
+
* the terms of any one of the MPL, the GPL or the LGPL.
|
38
|
+
*
|
39
|
+
* ***** END LICENSE BLOCK ***** */
|
40
|
+
|
41
|
+
/*
|
42
|
+
* JS execution context.
|
43
|
+
*/
|
44
|
+
#include "jsstddef.h"
|
45
|
+
#include <stdarg.h>
|
46
|
+
#include <stdlib.h>
|
47
|
+
#include <string.h>
|
48
|
+
#include "jstypes.h"
|
49
|
+
#include "jsarena.h" /* Added by JSIFY */
|
50
|
+
#include "jsutil.h" /* Added by JSIFY */
|
51
|
+
#include "jsclist.h"
|
52
|
+
#include "jsprf.h"
|
53
|
+
#include "jsatom.h"
|
54
|
+
#include "jscntxt.h"
|
55
|
+
#include "jsconfig.h"
|
56
|
+
#include "jsdbgapi.h"
|
57
|
+
#include "jsexn.h"
|
58
|
+
#include "jsgc.h"
|
59
|
+
#include "jslock.h"
|
60
|
+
#include "jsnum.h"
|
61
|
+
#include "jsobj.h"
|
62
|
+
#include "jsopcode.h"
|
63
|
+
#include "jsscan.h"
|
64
|
+
#include "jsscope.h"
|
65
|
+
#include "jsscript.h"
|
66
|
+
#include "jsstr.h"
|
67
|
+
|
68
|
+
#ifdef JS_THREADSAFE
|
69
|
+
#include "prtypes.h"
|
70
|
+
|
71
|
+
/*
|
72
|
+
* The index for JSThread info, returned by PR_NewThreadPrivateIndex. The
|
73
|
+
* index value is visible and shared by all threads, but the data associated
|
74
|
+
* with it is private to each thread.
|
75
|
+
*/
|
76
|
+
static PRUintn threadTPIndex;
|
77
|
+
static JSBool tpIndexInited = JS_FALSE;
|
78
|
+
|
79
|
+
JSBool
|
80
|
+
js_InitThreadPrivateIndex(void (JS_DLL_CALLBACK *ptr)(void *))
|
81
|
+
{
|
82
|
+
PRStatus status;
|
83
|
+
|
84
|
+
if (tpIndexInited)
|
85
|
+
return JS_TRUE;
|
86
|
+
|
87
|
+
status = PR_NewThreadPrivateIndex(&threadTPIndex, ptr);
|
88
|
+
|
89
|
+
if (status == PR_SUCCESS)
|
90
|
+
tpIndexInited = JS_TRUE;
|
91
|
+
return status == PR_SUCCESS;
|
92
|
+
}
|
93
|
+
|
94
|
+
/*
|
95
|
+
* Callback function to delete a JSThread info when the thread that owns it
|
96
|
+
* is destroyed.
|
97
|
+
*/
|
98
|
+
void JS_DLL_CALLBACK
|
99
|
+
js_ThreadDestructorCB(void *ptr)
|
100
|
+
{
|
101
|
+
JSThread *thread = (JSThread *)ptr;
|
102
|
+
|
103
|
+
if (!thread)
|
104
|
+
return;
|
105
|
+
|
106
|
+
/*
|
107
|
+
* Check that this thread properly called either JS_DestroyContext or
|
108
|
+
* JS_ClearContextThread on each JSContext it created or used.
|
109
|
+
*/
|
110
|
+
JS_ASSERT(JS_CLIST_IS_EMPTY(&thread->contextList));
|
111
|
+
GSN_CACHE_CLEAR(&thread->gsnCache);
|
112
|
+
free(thread);
|
113
|
+
}
|
114
|
+
|
115
|
+
/*
|
116
|
+
* Get current thread-local JSThread info, creating one if it doesn't exist.
|
117
|
+
* Each thread has a unique JSThread pointer.
|
118
|
+
*
|
119
|
+
* Since we are dealing with thread-local data, no lock is needed.
|
120
|
+
*
|
121
|
+
* Return a pointer to the thread local info, NULL if the system runs out
|
122
|
+
* of memory, or it failed to set thread private data (neither case is very
|
123
|
+
* likely; both are probably due to out-of-memory). It is up to the caller
|
124
|
+
* to report an error, if possible.
|
125
|
+
*/
|
126
|
+
JSThread *
|
127
|
+
js_GetCurrentThread(JSRuntime *rt)
|
128
|
+
{
|
129
|
+
JSThread *thread;
|
130
|
+
|
131
|
+
thread = (JSThread *)PR_GetThreadPrivate(threadTPIndex);
|
132
|
+
if (!thread) {
|
133
|
+
thread = (JSThread *) malloc(sizeof(JSThread));
|
134
|
+
if (!thread)
|
135
|
+
return NULL;
|
136
|
+
#ifdef DEBUG
|
137
|
+
memset(thread, JS_FREE_PATTERN, sizeof(JSThread));
|
138
|
+
#endif
|
139
|
+
if (PR_FAILURE == PR_SetThreadPrivate(threadTPIndex, thread)) {
|
140
|
+
free(thread);
|
141
|
+
return NULL;
|
142
|
+
}
|
143
|
+
|
144
|
+
JS_INIT_CLIST(&thread->contextList);
|
145
|
+
thread->id = js_CurrentThreadId();
|
146
|
+
thread->gcMallocBytes = 0;
|
147
|
+
|
148
|
+
/*
|
149
|
+
* js_SetContextThread initializes the remaining fields as necessary.
|
150
|
+
*/
|
151
|
+
}
|
152
|
+
return thread;
|
153
|
+
}
|
154
|
+
|
155
|
+
/*
|
156
|
+
* Sets current thread as owning thread of a context by assigning the
|
157
|
+
* thread-private info to the context. If the current thread doesn't have
|
158
|
+
* private JSThread info, create one.
|
159
|
+
*/
|
160
|
+
JSBool
|
161
|
+
js_SetContextThread(JSContext *cx)
|
162
|
+
{
|
163
|
+
JSThread *thread = js_GetCurrentThread(cx->runtime);
|
164
|
+
|
165
|
+
if (!thread) {
|
166
|
+
JS_ReportOutOfMemory(cx);
|
167
|
+
return JS_FALSE;
|
168
|
+
}
|
169
|
+
|
170
|
+
/*
|
171
|
+
* Clear gcFreeLists and caches on each transition from 0 to 1 context
|
172
|
+
* active on the current thread. See bug 351602 and bug 425828.
|
173
|
+
*/
|
174
|
+
if (JS_CLIST_IS_EMPTY(&thread->contextList)) {
|
175
|
+
memset(thread->gcFreeLists, 0, sizeof(thread->gcFreeLists));
|
176
|
+
memset(&thread->gsnCache, 0, sizeof(thread->gsnCache));
|
177
|
+
memset(&thread->propertyCache, 0, sizeof(thread->propertyCache));
|
178
|
+
}
|
179
|
+
|
180
|
+
/* Assert that the previous cx->thread called JS_ClearContextThread(). */
|
181
|
+
JS_ASSERT(!cx->thread || cx->thread == thread);
|
182
|
+
if (!cx->thread)
|
183
|
+
JS_APPEND_LINK(&cx->threadLinks, &thread->contextList);
|
184
|
+
cx->thread = thread;
|
185
|
+
return JS_TRUE;
|
186
|
+
}
|
187
|
+
|
188
|
+
/* Remove the owning thread info of a context. */
|
189
|
+
void
|
190
|
+
js_ClearContextThread(JSContext *cx)
|
191
|
+
{
|
192
|
+
/*
|
193
|
+
* If cx is associated with a thread, this must be called only from that
|
194
|
+
* thread. If not, this is a harmless no-op.
|
195
|
+
*/
|
196
|
+
JS_ASSERT(cx->thread == js_GetCurrentThread(cx->runtime) || !cx->thread);
|
197
|
+
JS_REMOVE_AND_INIT_LINK(&cx->threadLinks);
|
198
|
+
#ifdef DEBUG
|
199
|
+
if (JS_CLIST_IS_EMPTY(&cx->thread->contextList)) {
|
200
|
+
memset(cx->thread->gcFreeLists, JS_FREE_PATTERN,
|
201
|
+
sizeof(cx->thread->gcFreeLists));
|
202
|
+
}
|
203
|
+
#endif
|
204
|
+
cx->thread = NULL;
|
205
|
+
}
|
206
|
+
|
207
|
+
#endif /* JS_THREADSAFE */
|
208
|
+
|
209
|
+
void
|
210
|
+
js_OnVersionChange(JSContext *cx)
|
211
|
+
{
|
212
|
+
#ifdef DEBUG
|
213
|
+
JSVersion version = JSVERSION_NUMBER(cx);
|
214
|
+
|
215
|
+
JS_ASSERT(version == JSVERSION_DEFAULT || version >= JSVERSION_ECMA_3);
|
216
|
+
#endif
|
217
|
+
}
|
218
|
+
|
219
|
+
void
|
220
|
+
js_SetVersion(JSContext *cx, JSVersion version)
|
221
|
+
{
|
222
|
+
cx->version = version;
|
223
|
+
js_OnVersionChange(cx);
|
224
|
+
}
|
225
|
+
|
226
|
+
JSContext *
|
227
|
+
js_NewContext(JSRuntime *rt, size_t stackChunkSize)
|
228
|
+
{
|
229
|
+
JSContext *cx;
|
230
|
+
JSBool ok, first;
|
231
|
+
JSContextCallback cxCallback;
|
232
|
+
|
233
|
+
cx = (JSContext *) malloc(sizeof *cx);
|
234
|
+
if (!cx)
|
235
|
+
return NULL;
|
236
|
+
memset(cx, 0, sizeof *cx);
|
237
|
+
|
238
|
+
cx->runtime = rt;
|
239
|
+
JS_ClearOperationCallback(cx);
|
240
|
+
cx->debugHooks = &rt->globalDebugHooks;
|
241
|
+
#if JS_STACK_GROWTH_DIRECTION > 0
|
242
|
+
cx->stackLimit = (jsuword)-1;
|
243
|
+
#endif
|
244
|
+
cx->scriptStackQuota = JS_DEFAULT_SCRIPT_STACK_QUOTA;
|
245
|
+
#ifdef JS_THREADSAFE
|
246
|
+
JS_INIT_CLIST(&cx->threadLinks);
|
247
|
+
js_SetContextThread(cx);
|
248
|
+
#endif
|
249
|
+
|
250
|
+
JS_LOCK_GC(rt);
|
251
|
+
for (;;) {
|
252
|
+
first = (rt->contextList.next == &rt->contextList);
|
253
|
+
if (rt->state == JSRTS_UP) {
|
254
|
+
JS_ASSERT(!first);
|
255
|
+
break;
|
256
|
+
}
|
257
|
+
if (rt->state == JSRTS_DOWN) {
|
258
|
+
JS_ASSERT(first);
|
259
|
+
rt->state = JSRTS_LAUNCHING;
|
260
|
+
break;
|
261
|
+
}
|
262
|
+
JS_WAIT_CONDVAR(rt->stateChange, JS_NO_TIMEOUT);
|
263
|
+
}
|
264
|
+
JS_APPEND_LINK(&cx->links, &rt->contextList);
|
265
|
+
JS_UNLOCK_GC(rt);
|
266
|
+
|
267
|
+
/*
|
268
|
+
* First we do the infallible, every-time per-context initializations.
|
269
|
+
* Should a later, fallible initialization (js_InitRegExpStatics, e.g.,
|
270
|
+
* or the stuff under 'if (first)' below) fail, at least the version
|
271
|
+
* and arena-pools will be valid and safe to use (say, from the last GC
|
272
|
+
* done by js_DestroyContext).
|
273
|
+
*/
|
274
|
+
cx->version = JSVERSION_DEFAULT;
|
275
|
+
JS_INIT_ARENA_POOL(&cx->stackPool, "stack", stackChunkSize, sizeof(jsval),
|
276
|
+
&cx->scriptStackQuota);
|
277
|
+
JS_INIT_ARENA_POOL(&cx->tempPool, "temp", 1024, sizeof(jsdouble),
|
278
|
+
&cx->scriptStackQuota);
|
279
|
+
|
280
|
+
if (!js_InitRegExpStatics(cx, &cx->regExpStatics)) {
|
281
|
+
js_DestroyContext(cx, JSDCM_NEW_FAILED);
|
282
|
+
return NULL;
|
283
|
+
}
|
284
|
+
|
285
|
+
/*
|
286
|
+
* If cx is the first context on this runtime, initialize well-known atoms,
|
287
|
+
* keywords, numbers, and strings. If one of these steps should fail, the
|
288
|
+
* runtime will be left in a partially initialized state, with zeroes and
|
289
|
+
* nulls stored in the default-initialized remainder of the struct. We'll
|
290
|
+
* clean the runtime up under js_DestroyContext, because cx will be "last"
|
291
|
+
* as well as "first".
|
292
|
+
*/
|
293
|
+
if (first) {
|
294
|
+
#ifdef JS_THREADSAFE
|
295
|
+
JS_BeginRequest(cx);
|
296
|
+
#endif
|
297
|
+
ok = js_InitCommonAtoms(cx);
|
298
|
+
|
299
|
+
/*
|
300
|
+
* scriptFilenameTable may be left over from a previous episode of
|
301
|
+
* non-zero contexts alive in rt, so don't re-init the table if it's
|
302
|
+
* not necessary.
|
303
|
+
*/
|
304
|
+
if (ok && !rt->scriptFilenameTable)
|
305
|
+
ok = js_InitRuntimeScriptState(rt);
|
306
|
+
if (ok)
|
307
|
+
ok = js_InitRuntimeNumberState(cx);
|
308
|
+
if (ok)
|
309
|
+
ok = js_InitRuntimeStringState(cx);
|
310
|
+
#ifdef JS_THREADSAFE
|
311
|
+
JS_EndRequest(cx);
|
312
|
+
#endif
|
313
|
+
if (!ok) {
|
314
|
+
js_DestroyContext(cx, JSDCM_NEW_FAILED);
|
315
|
+
return NULL;
|
316
|
+
}
|
317
|
+
|
318
|
+
JS_LOCK_GC(rt);
|
319
|
+
rt->state = JSRTS_UP;
|
320
|
+
JS_NOTIFY_ALL_CONDVAR(rt->stateChange);
|
321
|
+
JS_UNLOCK_GC(rt);
|
322
|
+
}
|
323
|
+
|
324
|
+
cxCallback = rt->cxCallback;
|
325
|
+
if (cxCallback && !cxCallback(cx, JSCONTEXT_NEW)) {
|
326
|
+
js_DestroyContext(cx, JSDCM_NEW_FAILED);
|
327
|
+
return NULL;
|
328
|
+
}
|
329
|
+
return cx;
|
330
|
+
}
|
331
|
+
|
332
|
+
void
|
333
|
+
js_DestroyContext(JSContext *cx, JSDestroyContextMode mode)
|
334
|
+
{
|
335
|
+
JSRuntime *rt;
|
336
|
+
JSContextCallback cxCallback;
|
337
|
+
JSBool last;
|
338
|
+
JSArgumentFormatMap *map;
|
339
|
+
JSLocalRootStack *lrs;
|
340
|
+
JSLocalRootChunk *lrc;
|
341
|
+
|
342
|
+
rt = cx->runtime;
|
343
|
+
|
344
|
+
if (mode != JSDCM_NEW_FAILED) {
|
345
|
+
cxCallback = rt->cxCallback;
|
346
|
+
if (cxCallback) {
|
347
|
+
/*
|
348
|
+
* JSCONTEXT_DESTROY callback is not allowed to fail and must
|
349
|
+
* return true.
|
350
|
+
*/
|
351
|
+
#ifdef DEBUG
|
352
|
+
JSBool callbackStatus =
|
353
|
+
#endif
|
354
|
+
cxCallback(cx, JSCONTEXT_DESTROY);
|
355
|
+
JS_ASSERT(callbackStatus);
|
356
|
+
}
|
357
|
+
}
|
358
|
+
|
359
|
+
/* Remove cx from context list first. */
|
360
|
+
JS_LOCK_GC(rt);
|
361
|
+
JS_ASSERT(rt->state == JSRTS_UP || rt->state == JSRTS_LAUNCHING);
|
362
|
+
JS_REMOVE_LINK(&cx->links);
|
363
|
+
last = (rt->contextList.next == &rt->contextList);
|
364
|
+
if (last)
|
365
|
+
rt->state = JSRTS_LANDING;
|
366
|
+
JS_UNLOCK_GC(rt);
|
367
|
+
|
368
|
+
if (last) {
|
369
|
+
#ifdef JS_THREADSAFE
|
370
|
+
/*
|
371
|
+
* If cx is not in a request already, begin one now so that we wait
|
372
|
+
* for any racing GC started on a not-last context to finish, before
|
373
|
+
* we plow ahead and unpin atoms. Note that even though we begin a
|
374
|
+
* request here if necessary, we end all requests on cx below before
|
375
|
+
* forcing a final GC. This lets any not-last context destruction
|
376
|
+
* racing in another thread try to force or maybe run the GC, but by
|
377
|
+
* that point, rt->state will not be JSRTS_UP, and that GC attempt
|
378
|
+
* will return early.
|
379
|
+
*/
|
380
|
+
if (cx->requestDepth == 0)
|
381
|
+
JS_BeginRequest(cx);
|
382
|
+
#endif
|
383
|
+
|
384
|
+
/* Unlock and clear GC things held by runtime pointers. */
|
385
|
+
js_FinishRuntimeNumberState(cx);
|
386
|
+
js_FinishRuntimeStringState(cx);
|
387
|
+
|
388
|
+
/* Unpin all common atoms before final GC. */
|
389
|
+
js_FinishCommonAtoms(cx);
|
390
|
+
|
391
|
+
/* Clear debugging state to remove GC roots. */
|
392
|
+
JS_ClearAllTraps(cx);
|
393
|
+
JS_ClearAllWatchPoints(cx);
|
394
|
+
}
|
395
|
+
|
396
|
+
/*
|
397
|
+
* Remove more GC roots in regExpStatics, then collect garbage.
|
398
|
+
* XXX anti-modularity alert: we rely on the call to js_RemoveRoot within
|
399
|
+
* XXX this function call to wait for any racing GC to complete, in the
|
400
|
+
* XXX case where JS_DestroyContext is called outside of a request on cx
|
401
|
+
*/
|
402
|
+
js_FreeRegExpStatics(cx, &cx->regExpStatics);
|
403
|
+
|
404
|
+
#ifdef JS_THREADSAFE
|
405
|
+
/*
|
406
|
+
* Destroying a context implicitly calls JS_EndRequest(). Also, we must
|
407
|
+
* end our request here in case we are "last" -- in that event, another
|
408
|
+
* js_DestroyContext that was not last might be waiting in the GC for our
|
409
|
+
* request to end. We'll let it run below, just before we do the truly
|
410
|
+
* final GC and then free atom state.
|
411
|
+
*
|
412
|
+
* At this point, cx must be inaccessible to other threads. It's off the
|
413
|
+
* rt->contextList, and it should not be reachable via any object private
|
414
|
+
* data structure.
|
415
|
+
*/
|
416
|
+
while (cx->requestDepth != 0)
|
417
|
+
JS_EndRequest(cx);
|
418
|
+
#endif
|
419
|
+
|
420
|
+
if (last) {
|
421
|
+
js_GC(cx, GC_LAST_CONTEXT);
|
422
|
+
|
423
|
+
/*
|
424
|
+
* Free the script filename table if it exists and is empty. Do this
|
425
|
+
* after the last GC to avoid finalizers tripping on free memory.
|
426
|
+
*/
|
427
|
+
if (rt->scriptFilenameTable && rt->scriptFilenameTable->nentries == 0)
|
428
|
+
js_FinishRuntimeScriptState(rt);
|
429
|
+
|
430
|
+
/* Take the runtime down, now that it has no contexts or atoms. */
|
431
|
+
JS_LOCK_GC(rt);
|
432
|
+
rt->state = JSRTS_DOWN;
|
433
|
+
JS_NOTIFY_ALL_CONDVAR(rt->stateChange);
|
434
|
+
JS_UNLOCK_GC(rt);
|
435
|
+
} else {
|
436
|
+
if (mode == JSDCM_FORCE_GC)
|
437
|
+
js_GC(cx, GC_NORMAL);
|
438
|
+
else if (mode == JSDCM_MAYBE_GC)
|
439
|
+
JS_MaybeGC(cx);
|
440
|
+
}
|
441
|
+
|
442
|
+
/* Free the stuff hanging off of cx. */
|
443
|
+
JS_FinishArenaPool(&cx->stackPool);
|
444
|
+
JS_FinishArenaPool(&cx->tempPool);
|
445
|
+
|
446
|
+
if (cx->lastMessage)
|
447
|
+
free(cx->lastMessage);
|
448
|
+
|
449
|
+
/* Remove any argument formatters. */
|
450
|
+
map = cx->argumentFormatMap;
|
451
|
+
while (map) {
|
452
|
+
JSArgumentFormatMap *temp = map;
|
453
|
+
map = map->next;
|
454
|
+
JS_free(cx, temp);
|
455
|
+
}
|
456
|
+
|
457
|
+
/* Destroy the resolve recursion damper. */
|
458
|
+
if (cx->resolvingTable) {
|
459
|
+
JS_DHashTableDestroy(cx->resolvingTable);
|
460
|
+
cx->resolvingTable = NULL;
|
461
|
+
}
|
462
|
+
|
463
|
+
lrs = cx->localRootStack;
|
464
|
+
if (lrs) {
|
465
|
+
while ((lrc = lrs->topChunk) != &lrs->firstChunk) {
|
466
|
+
lrs->topChunk = lrc->down;
|
467
|
+
JS_free(cx, lrc);
|
468
|
+
}
|
469
|
+
JS_free(cx, lrs);
|
470
|
+
}
|
471
|
+
|
472
|
+
#ifdef JS_THREADSAFE
|
473
|
+
js_ClearContextThread(cx);
|
474
|
+
#endif
|
475
|
+
|
476
|
+
/* Finally, free cx itself. */
|
477
|
+
free(cx);
|
478
|
+
}
|
479
|
+
|
480
|
+
JSBool
|
481
|
+
js_ValidContextPointer(JSRuntime *rt, JSContext *cx)
|
482
|
+
{
|
483
|
+
JSCList *cl;
|
484
|
+
|
485
|
+
for (cl = rt->contextList.next; cl != &rt->contextList; cl = cl->next) {
|
486
|
+
if (cl == &cx->links)
|
487
|
+
return JS_TRUE;
|
488
|
+
}
|
489
|
+
JS_RUNTIME_METER(rt, deadContexts);
|
490
|
+
return JS_FALSE;
|
491
|
+
}
|
492
|
+
|
493
|
+
JSContext *
|
494
|
+
js_ContextIterator(JSRuntime *rt, JSBool unlocked, JSContext **iterp)
|
495
|
+
{
|
496
|
+
JSContext *cx = *iterp;
|
497
|
+
|
498
|
+
if (unlocked)
|
499
|
+
JS_LOCK_GC(rt);
|
500
|
+
if (!cx)
|
501
|
+
cx = (JSContext *)&rt->contextList;
|
502
|
+
cx = (JSContext *)cx->links.next;
|
503
|
+
if (&cx->links == &rt->contextList)
|
504
|
+
cx = NULL;
|
505
|
+
*iterp = cx;
|
506
|
+
if (unlocked)
|
507
|
+
JS_UNLOCK_GC(rt);
|
508
|
+
return cx;
|
509
|
+
}
|
510
|
+
|
511
|
+
JS_STATIC_DLL_CALLBACK(JSDHashNumber)
|
512
|
+
resolving_HashKey(JSDHashTable *table, const void *ptr)
|
513
|
+
{
|
514
|
+
const JSResolvingKey *key = (const JSResolvingKey *)ptr;
|
515
|
+
|
516
|
+
return ((JSDHashNumber)JS_PTR_TO_UINT32(key->obj) >> JSVAL_TAGBITS) ^ key->id;
|
517
|
+
}
|
518
|
+
|
519
|
+
JS_PUBLIC_API(JSBool)
|
520
|
+
resolving_MatchEntry(JSDHashTable *table,
|
521
|
+
const JSDHashEntryHdr *hdr,
|
522
|
+
const void *ptr)
|
523
|
+
{
|
524
|
+
const JSResolvingEntry *entry = (const JSResolvingEntry *)hdr;
|
525
|
+
const JSResolvingKey *key = (const JSResolvingKey *)ptr;
|
526
|
+
|
527
|
+
return entry->key.obj == key->obj && entry->key.id == key->id;
|
528
|
+
}
|
529
|
+
|
530
|
+
static const JSDHashTableOps resolving_dhash_ops = {
|
531
|
+
JS_DHashAllocTable,
|
532
|
+
JS_DHashFreeTable,
|
533
|
+
resolving_HashKey,
|
534
|
+
resolving_MatchEntry,
|
535
|
+
JS_DHashMoveEntryStub,
|
536
|
+
JS_DHashClearEntryStub,
|
537
|
+
JS_DHashFinalizeStub,
|
538
|
+
NULL
|
539
|
+
};
|
540
|
+
|
541
|
+
JSBool
|
542
|
+
js_StartResolving(JSContext *cx, JSResolvingKey *key, uint32 flag,
|
543
|
+
JSResolvingEntry **entryp)
|
544
|
+
{
|
545
|
+
JSDHashTable *table;
|
546
|
+
JSResolvingEntry *entry;
|
547
|
+
|
548
|
+
table = cx->resolvingTable;
|
549
|
+
if (!table) {
|
550
|
+
table = JS_NewDHashTable(&resolving_dhash_ops, NULL,
|
551
|
+
sizeof(JSResolvingEntry),
|
552
|
+
JS_DHASH_MIN_SIZE);
|
553
|
+
if (!table)
|
554
|
+
goto outofmem;
|
555
|
+
cx->resolvingTable = table;
|
556
|
+
}
|
557
|
+
|
558
|
+
entry = (JSResolvingEntry *)
|
559
|
+
JS_DHashTableOperate(table, key, JS_DHASH_ADD);
|
560
|
+
if (!entry)
|
561
|
+
goto outofmem;
|
562
|
+
|
563
|
+
if (entry->flags & flag) {
|
564
|
+
/* An entry for (key, flag) exists already -- dampen recursion. */
|
565
|
+
entry = NULL;
|
566
|
+
} else {
|
567
|
+
/* Fill in key if we were the first to add entry, then set flag. */
|
568
|
+
if (!entry->key.obj)
|
569
|
+
entry->key = *key;
|
570
|
+
entry->flags |= flag;
|
571
|
+
}
|
572
|
+
*entryp = entry;
|
573
|
+
return JS_TRUE;
|
574
|
+
|
575
|
+
outofmem:
|
576
|
+
JS_ReportOutOfMemory(cx);
|
577
|
+
return JS_FALSE;
|
578
|
+
}
|
579
|
+
|
580
|
+
void
|
581
|
+
js_StopResolving(JSContext *cx, JSResolvingKey *key, uint32 flag,
|
582
|
+
JSResolvingEntry *entry, uint32 generation)
|
583
|
+
{
|
584
|
+
JSDHashTable *table;
|
585
|
+
|
586
|
+
/*
|
587
|
+
* Clear flag from entry->flags and return early if other flags remain.
|
588
|
+
* We must take care to re-lookup entry if the table has changed since
|
589
|
+
* it was found by js_StartResolving.
|
590
|
+
*/
|
591
|
+
table = cx->resolvingTable;
|
592
|
+
if (!entry || table->generation != generation) {
|
593
|
+
entry = (JSResolvingEntry *)
|
594
|
+
JS_DHashTableOperate(table, key, JS_DHASH_LOOKUP);
|
595
|
+
}
|
596
|
+
JS_ASSERT(JS_DHASH_ENTRY_IS_BUSY(&entry->hdr));
|
597
|
+
entry->flags &= ~flag;
|
598
|
+
if (entry->flags)
|
599
|
+
return;
|
600
|
+
|
601
|
+
/*
|
602
|
+
* Do a raw remove only if fewer entries were removed than would cause
|
603
|
+
* alpha to be less than .5 (alpha is at most .75). Otherwise, we just
|
604
|
+
* call JS_DHashTableOperate to re-lookup the key and remove its entry,
|
605
|
+
* compressing or shrinking the table as needed.
|
606
|
+
*/
|
607
|
+
if (table->removedCount < JS_DHASH_TABLE_SIZE(table) >> 2)
|
608
|
+
JS_DHashTableRawRemove(table, &entry->hdr);
|
609
|
+
else
|
610
|
+
JS_DHashTableOperate(table, key, JS_DHASH_REMOVE);
|
611
|
+
}
|
612
|
+
|
613
|
+
JSBool
|
614
|
+
js_EnterLocalRootScope(JSContext *cx)
|
615
|
+
{
|
616
|
+
JSLocalRootStack *lrs;
|
617
|
+
int mark;
|
618
|
+
|
619
|
+
lrs = cx->localRootStack;
|
620
|
+
if (!lrs) {
|
621
|
+
lrs = (JSLocalRootStack *) JS_malloc(cx, sizeof *lrs);
|
622
|
+
if (!lrs)
|
623
|
+
return JS_FALSE;
|
624
|
+
lrs->scopeMark = JSLRS_NULL_MARK;
|
625
|
+
lrs->rootCount = 0;
|
626
|
+
lrs->topChunk = &lrs->firstChunk;
|
627
|
+
lrs->firstChunk.down = NULL;
|
628
|
+
cx->localRootStack = lrs;
|
629
|
+
}
|
630
|
+
|
631
|
+
/* Push lrs->scopeMark to save it for restore when leaving. */
|
632
|
+
mark = js_PushLocalRoot(cx, lrs, INT_TO_JSVAL(lrs->scopeMark));
|
633
|
+
if (mark < 0)
|
634
|
+
return JS_FALSE;
|
635
|
+
lrs->scopeMark = (uint32) mark;
|
636
|
+
return JS_TRUE;
|
637
|
+
}
|
638
|
+
|
639
|
+
void
|
640
|
+
js_LeaveLocalRootScopeWithResult(JSContext *cx, jsval rval)
|
641
|
+
{
|
642
|
+
JSLocalRootStack *lrs;
|
643
|
+
uint32 mark, m, n;
|
644
|
+
JSLocalRootChunk *lrc;
|
645
|
+
|
646
|
+
/* Defend against buggy native callers. */
|
647
|
+
lrs = cx->localRootStack;
|
648
|
+
JS_ASSERT(lrs && lrs->rootCount != 0);
|
649
|
+
if (!lrs || lrs->rootCount == 0)
|
650
|
+
return;
|
651
|
+
|
652
|
+
mark = lrs->scopeMark;
|
653
|
+
JS_ASSERT(mark != JSLRS_NULL_MARK);
|
654
|
+
if (mark == JSLRS_NULL_MARK)
|
655
|
+
return;
|
656
|
+
|
657
|
+
/* Free any chunks being popped by this leave operation. */
|
658
|
+
m = mark >> JSLRS_CHUNK_SHIFT;
|
659
|
+
n = (lrs->rootCount - 1) >> JSLRS_CHUNK_SHIFT;
|
660
|
+
while (n > m) {
|
661
|
+
lrc = lrs->topChunk;
|
662
|
+
JS_ASSERT(lrc != &lrs->firstChunk);
|
663
|
+
lrs->topChunk = lrc->down;
|
664
|
+
JS_free(cx, lrc);
|
665
|
+
--n;
|
666
|
+
}
|
667
|
+
|
668
|
+
/*
|
669
|
+
* Pop the scope, restoring lrs->scopeMark. If rval is a GC-thing, push
|
670
|
+
* it on the caller's scope, or store it in lastInternalResult if we are
|
671
|
+
* leaving the outermost scope. We don't need to allocate a new lrc
|
672
|
+
* because we can overwrite the old mark's slot with rval.
|
673
|
+
*/
|
674
|
+
lrc = lrs->topChunk;
|
675
|
+
m = mark & JSLRS_CHUNK_MASK;
|
676
|
+
lrs->scopeMark = (uint32) JSVAL_TO_INT(lrc->roots[m]);
|
677
|
+
if (JSVAL_IS_GCTHING(rval) && !JSVAL_IS_NULL(rval)) {
|
678
|
+
if (mark == 0) {
|
679
|
+
cx->weakRoots.lastInternalResult = rval;
|
680
|
+
} else {
|
681
|
+
/*
|
682
|
+
* Increment m to avoid the "else if (m == 0)" case below. If
|
683
|
+
* rval is not a GC-thing, that case would take care of freeing
|
684
|
+
* any chunk that contained only the old mark. Since rval *is*
|
685
|
+
* a GC-thing here, we want to reuse that old mark's slot.
|
686
|
+
*/
|
687
|
+
lrc->roots[m++] = rval;
|
688
|
+
++mark;
|
689
|
+
}
|
690
|
+
}
|
691
|
+
lrs->rootCount = (uint32) mark;
|
692
|
+
|
693
|
+
/*
|
694
|
+
* Free the stack eagerly, risking malloc churn. The alternative would
|
695
|
+
* require an lrs->entryCount member, maintained by Enter and Leave, and
|
696
|
+
* tested by the GC in addition to the cx->localRootStack non-null test.
|
697
|
+
*
|
698
|
+
* That approach would risk hoarding 264 bytes (net) per context. Right
|
699
|
+
* now it seems better to give fresh (dirty in CPU write-back cache, and
|
700
|
+
* the data is no longer needed) memory back to the malloc heap.
|
701
|
+
*/
|
702
|
+
if (mark == 0) {
|
703
|
+
cx->localRootStack = NULL;
|
704
|
+
JS_free(cx, lrs);
|
705
|
+
} else if (m == 0) {
|
706
|
+
lrs->topChunk = lrc->down;
|
707
|
+
JS_free(cx, lrc);
|
708
|
+
}
|
709
|
+
}
|
710
|
+
|
711
|
+
void
|
712
|
+
js_ForgetLocalRoot(JSContext *cx, jsval v)
|
713
|
+
{
|
714
|
+
JSLocalRootStack *lrs;
|
715
|
+
uint32 i, j, m, n, mark;
|
716
|
+
JSLocalRootChunk *lrc, *lrc2;
|
717
|
+
jsval top;
|
718
|
+
|
719
|
+
lrs = cx->localRootStack;
|
720
|
+
JS_ASSERT(lrs && lrs->rootCount);
|
721
|
+
if (!lrs || lrs->rootCount == 0)
|
722
|
+
return;
|
723
|
+
|
724
|
+
/* Prepare to pop the top-most value from the stack. */
|
725
|
+
n = lrs->rootCount - 1;
|
726
|
+
m = n & JSLRS_CHUNK_MASK;
|
727
|
+
lrc = lrs->topChunk;
|
728
|
+
top = lrc->roots[m];
|
729
|
+
|
730
|
+
/* Be paranoid about calls on an empty scope. */
|
731
|
+
mark = lrs->scopeMark;
|
732
|
+
JS_ASSERT(mark < n);
|
733
|
+
if (mark >= n)
|
734
|
+
return;
|
735
|
+
|
736
|
+
/* If v was not the last root pushed in the top scope, find it. */
|
737
|
+
if (top != v) {
|
738
|
+
/* Search downward in case v was recently pushed. */
|
739
|
+
i = n;
|
740
|
+
j = m;
|
741
|
+
lrc2 = lrc;
|
742
|
+
while (--i > mark) {
|
743
|
+
if (j == 0)
|
744
|
+
lrc2 = lrc2->down;
|
745
|
+
j = i & JSLRS_CHUNK_MASK;
|
746
|
+
if (lrc2->roots[j] == v)
|
747
|
+
break;
|
748
|
+
}
|
749
|
+
|
750
|
+
/* If we didn't find v in this scope, assert and bail out. */
|
751
|
+
JS_ASSERT(i != mark);
|
752
|
+
if (i == mark)
|
753
|
+
return;
|
754
|
+
|
755
|
+
/* Swap top and v so common tail code can pop v. */
|
756
|
+
lrc2->roots[j] = top;
|
757
|
+
}
|
758
|
+
|
759
|
+
/* Pop the last value from the stack. */
|
760
|
+
lrc->roots[m] = JSVAL_NULL;
|
761
|
+
lrs->rootCount = n;
|
762
|
+
if (m == 0) {
|
763
|
+
JS_ASSERT(n != 0);
|
764
|
+
JS_ASSERT(lrc != &lrs->firstChunk);
|
765
|
+
lrs->topChunk = lrc->down;
|
766
|
+
JS_free(cx, lrc);
|
767
|
+
}
|
768
|
+
}
|
769
|
+
|
770
|
+
int
|
771
|
+
js_PushLocalRoot(JSContext *cx, JSLocalRootStack *lrs, jsval v)
|
772
|
+
{
|
773
|
+
uint32 n, m;
|
774
|
+
JSLocalRootChunk *lrc;
|
775
|
+
|
776
|
+
n = lrs->rootCount;
|
777
|
+
m = n & JSLRS_CHUNK_MASK;
|
778
|
+
if (n == 0 || m != 0) {
|
779
|
+
/*
|
780
|
+
* At start of first chunk, or not at start of a non-first top chunk.
|
781
|
+
* Check for lrs->rootCount overflow.
|
782
|
+
*/
|
783
|
+
if ((uint32)(n + 1) == 0) {
|
784
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
785
|
+
JSMSG_TOO_MANY_LOCAL_ROOTS);
|
786
|
+
return -1;
|
787
|
+
}
|
788
|
+
lrc = lrs->topChunk;
|
789
|
+
JS_ASSERT(n != 0 || lrc == &lrs->firstChunk);
|
790
|
+
} else {
|
791
|
+
/*
|
792
|
+
* After lrs->firstChunk, trying to index at a power-of-two chunk
|
793
|
+
* boundary: need a new chunk.
|
794
|
+
*/
|
795
|
+
lrc = (JSLocalRootChunk *) JS_malloc(cx, sizeof *lrc);
|
796
|
+
if (!lrc)
|
797
|
+
return -1;
|
798
|
+
lrc->down = lrs->topChunk;
|
799
|
+
lrs->topChunk = lrc;
|
800
|
+
}
|
801
|
+
lrs->rootCount = n + 1;
|
802
|
+
lrc->roots[m] = v;
|
803
|
+
return (int) n;
|
804
|
+
}
|
805
|
+
|
806
|
+
void
|
807
|
+
js_TraceLocalRoots(JSTracer *trc, JSLocalRootStack *lrs)
|
808
|
+
{
|
809
|
+
uint32 n, m, mark;
|
810
|
+
JSLocalRootChunk *lrc;
|
811
|
+
jsval v;
|
812
|
+
|
813
|
+
n = lrs->rootCount;
|
814
|
+
if (n == 0)
|
815
|
+
return;
|
816
|
+
|
817
|
+
mark = lrs->scopeMark;
|
818
|
+
lrc = lrs->topChunk;
|
819
|
+
do {
|
820
|
+
while (--n > mark) {
|
821
|
+
m = n & JSLRS_CHUNK_MASK;
|
822
|
+
v = lrc->roots[m];
|
823
|
+
JS_ASSERT(JSVAL_IS_GCTHING(v) && v != JSVAL_NULL);
|
824
|
+
JS_SET_TRACING_INDEX(trc, "local_root", n);
|
825
|
+
js_CallValueTracerIfGCThing(trc, v);
|
826
|
+
if (m == 0)
|
827
|
+
lrc = lrc->down;
|
828
|
+
}
|
829
|
+
m = n & JSLRS_CHUNK_MASK;
|
830
|
+
mark = JSVAL_TO_INT(lrc->roots[m]);
|
831
|
+
if (m == 0)
|
832
|
+
lrc = lrc->down;
|
833
|
+
} while (n != 0);
|
834
|
+
JS_ASSERT(!lrc);
|
835
|
+
}
|
836
|
+
|
837
|
+
static void
|
838
|
+
ReportError(JSContext *cx, const char *message, JSErrorReport *reportp)
|
839
|
+
{
|
840
|
+
/*
|
841
|
+
* Check the error report, and set a JavaScript-catchable exception
|
842
|
+
* if the error is defined to have an associated exception. If an
|
843
|
+
* exception is thrown, then the JSREPORT_EXCEPTION flag will be set
|
844
|
+
* on the error report, and exception-aware hosts should ignore it.
|
845
|
+
*/
|
846
|
+
JS_ASSERT(reportp);
|
847
|
+
if (reportp->errorNumber == JSMSG_UNCAUGHT_EXCEPTION)
|
848
|
+
reportp->flags |= JSREPORT_EXCEPTION;
|
849
|
+
|
850
|
+
/*
|
851
|
+
* Call the error reporter only if an exception wasn't raised.
|
852
|
+
*
|
853
|
+
* If an exception was raised, then we call the debugErrorHook
|
854
|
+
* (if present) to give it a chance to see the error before it
|
855
|
+
* propagates out of scope. This is needed for compatability
|
856
|
+
* with the old scheme.
|
857
|
+
*/
|
858
|
+
if (!js_ErrorToException(cx, message, reportp)) {
|
859
|
+
js_ReportErrorAgain(cx, message, reportp);
|
860
|
+
} else if (cx->debugHooks->debugErrorHook && cx->errorReporter) {
|
861
|
+
JSDebugErrorHook hook = cx->debugHooks->debugErrorHook;
|
862
|
+
/* test local in case debugErrorHook changed on another thread */
|
863
|
+
if (hook)
|
864
|
+
hook(cx, message, reportp, cx->debugHooks->debugErrorHookData);
|
865
|
+
}
|
866
|
+
}
|
867
|
+
|
868
|
+
/*
|
869
|
+
* We don't post an exception in this case, since doing so runs into
|
870
|
+
* complications of pre-allocating an exception object which required
|
871
|
+
* running the Exception class initializer early etc.
|
872
|
+
* Instead we just invoke the errorReporter with an "Out Of Memory"
|
873
|
+
* type message, and then hope the process ends swiftly.
|
874
|
+
*/
|
875
|
+
void
|
876
|
+
js_ReportOutOfMemory(JSContext *cx)
|
877
|
+
{
|
878
|
+
JSStackFrame *fp;
|
879
|
+
JSErrorReport report;
|
880
|
+
JSErrorReporter onError = cx->errorReporter;
|
881
|
+
|
882
|
+
/* Get the message for this error, but we won't expand any arguments. */
|
883
|
+
const JSErrorFormatString *efs =
|
884
|
+
js_GetLocalizedErrorMessage(cx, NULL, NULL, JSMSG_OUT_OF_MEMORY);
|
885
|
+
const char *msg = efs ? efs->format : "Out of memory";
|
886
|
+
|
887
|
+
/* Fill out the report, but don't do anything that requires allocation. */
|
888
|
+
memset(&report, 0, sizeof (struct JSErrorReport));
|
889
|
+
report.flags = JSREPORT_ERROR;
|
890
|
+
report.errorNumber = JSMSG_OUT_OF_MEMORY;
|
891
|
+
|
892
|
+
/*
|
893
|
+
* Walk stack until we find a frame that is associated with some script
|
894
|
+
* rather than a native frame.
|
895
|
+
*/
|
896
|
+
for (fp = cx->fp; fp; fp = fp->down) {
|
897
|
+
if (fp->regs) {
|
898
|
+
report.filename = fp->script->filename;
|
899
|
+
report.lineno = js_PCToLineNumber(cx, fp->script, fp->regs->pc);
|
900
|
+
break;
|
901
|
+
}
|
902
|
+
}
|
903
|
+
|
904
|
+
/*
|
905
|
+
* If debugErrorHook is present then we give it a chance to veto sending
|
906
|
+
* the error on to the regular ErrorReporter. We also clear a pending
|
907
|
+
* exception if any now so the hooks can replace the out-of-memory error
|
908
|
+
* by a script-catchable exception.
|
909
|
+
*/
|
910
|
+
cx->throwing = JS_FALSE;
|
911
|
+
if (onError) {
|
912
|
+
JSDebugErrorHook hook = cx->debugHooks->debugErrorHook;
|
913
|
+
if (hook &&
|
914
|
+
!hook(cx, msg, &report, cx->debugHooks->debugErrorHookData)) {
|
915
|
+
onError = NULL;
|
916
|
+
}
|
917
|
+
}
|
918
|
+
|
919
|
+
if (onError)
|
920
|
+
onError(cx, msg, &report);
|
921
|
+
}
|
922
|
+
|
923
|
+
void
|
924
|
+
js_ReportOutOfScriptQuota(JSContext *cx)
|
925
|
+
{
|
926
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
927
|
+
JSMSG_SCRIPT_STACK_QUOTA);
|
928
|
+
}
|
929
|
+
|
930
|
+
void
|
931
|
+
js_ReportOverRecursed(JSContext *cx)
|
932
|
+
{
|
933
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_OVER_RECURSED);
|
934
|
+
}
|
935
|
+
|
936
|
+
void
|
937
|
+
js_ReportAllocationOverflow(JSContext *cx)
|
938
|
+
{
|
939
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_ALLOC_OVERFLOW);
|
940
|
+
}
|
941
|
+
|
942
|
+
JSBool
|
943
|
+
js_ReportErrorVA(JSContext *cx, uintN flags, const char *format, va_list ap)
|
944
|
+
{
|
945
|
+
char *message;
|
946
|
+
jschar *ucmessage;
|
947
|
+
size_t messagelen;
|
948
|
+
JSStackFrame *fp;
|
949
|
+
JSErrorReport report;
|
950
|
+
JSBool warning;
|
951
|
+
|
952
|
+
if ((flags & JSREPORT_STRICT) && !JS_HAS_STRICT_OPTION(cx))
|
953
|
+
return JS_TRUE;
|
954
|
+
|
955
|
+
message = JS_vsmprintf(format, ap);
|
956
|
+
if (!message)
|
957
|
+
return JS_FALSE;
|
958
|
+
messagelen = strlen(message);
|
959
|
+
|
960
|
+
memset(&report, 0, sizeof (struct JSErrorReport));
|
961
|
+
report.flags = flags;
|
962
|
+
report.errorNumber = JSMSG_USER_DEFINED_ERROR;
|
963
|
+
report.ucmessage = ucmessage = js_InflateString(cx, message, &messagelen);
|
964
|
+
|
965
|
+
/* Find the top-most active script frame, for best line number blame. */
|
966
|
+
for (fp = cx->fp; fp; fp = fp->down) {
|
967
|
+
if (fp->regs) {
|
968
|
+
report.filename = fp->script->filename;
|
969
|
+
report.lineno = js_PCToLineNumber(cx, fp->script, fp->regs->pc);
|
970
|
+
break;
|
971
|
+
}
|
972
|
+
}
|
973
|
+
|
974
|
+
warning = JSREPORT_IS_WARNING(report.flags);
|
975
|
+
if (warning && JS_HAS_WERROR_OPTION(cx)) {
|
976
|
+
report.flags &= ~JSREPORT_WARNING;
|
977
|
+
warning = JS_FALSE;
|
978
|
+
}
|
979
|
+
|
980
|
+
ReportError(cx, message, &report);
|
981
|
+
free(message);
|
982
|
+
JS_free(cx, ucmessage);
|
983
|
+
return warning;
|
984
|
+
}
|
985
|
+
|
986
|
+
/*
|
987
|
+
* The arguments from ap need to be packaged up into an array and stored
|
988
|
+
* into the report struct.
|
989
|
+
*
|
990
|
+
* The format string addressed by the error number may contain operands
|
991
|
+
* identified by the format {N}, where N is a decimal digit. Each of these
|
992
|
+
* is to be replaced by the Nth argument from the va_list. The complete
|
993
|
+
* message is placed into reportp->ucmessage converted to a JSString.
|
994
|
+
*
|
995
|
+
* Returns true if the expansion succeeds (can fail if out of memory).
|
996
|
+
*/
|
997
|
+
JSBool
|
998
|
+
js_ExpandErrorArguments(JSContext *cx, JSErrorCallback callback,
|
999
|
+
void *userRef, const uintN errorNumber,
|
1000
|
+
char **messagep, JSErrorReport *reportp,
|
1001
|
+
JSBool *warningp, JSBool charArgs, va_list ap)
|
1002
|
+
{
|
1003
|
+
const JSErrorFormatString *efs;
|
1004
|
+
int i;
|
1005
|
+
int argCount;
|
1006
|
+
|
1007
|
+
*warningp = JSREPORT_IS_WARNING(reportp->flags);
|
1008
|
+
if (*warningp && JS_HAS_WERROR_OPTION(cx)) {
|
1009
|
+
reportp->flags &= ~JSREPORT_WARNING;
|
1010
|
+
*warningp = JS_FALSE;
|
1011
|
+
}
|
1012
|
+
|
1013
|
+
*messagep = NULL;
|
1014
|
+
|
1015
|
+
/* Most calls supply js_GetErrorMessage; if this is so, assume NULL. */
|
1016
|
+
if (!callback || callback == js_GetErrorMessage)
|
1017
|
+
efs = js_GetLocalizedErrorMessage(cx, userRef, NULL, errorNumber);
|
1018
|
+
else
|
1019
|
+
efs = callback(userRef, NULL, errorNumber);
|
1020
|
+
if (efs) {
|
1021
|
+
size_t totalArgsLength = 0;
|
1022
|
+
size_t argLengths[10]; /* only {0} thru {9} supported */
|
1023
|
+
argCount = efs->argCount;
|
1024
|
+
JS_ASSERT(argCount <= 10);
|
1025
|
+
if (argCount > 0) {
|
1026
|
+
/*
|
1027
|
+
* Gather the arguments into an array, and accumulate
|
1028
|
+
* their sizes. We allocate 1 more than necessary and
|
1029
|
+
* null it out to act as the caboose when we free the
|
1030
|
+
* pointers later.
|
1031
|
+
*/
|
1032
|
+
reportp->messageArgs = (const jschar **)
|
1033
|
+
JS_malloc(cx, sizeof(jschar *) * (argCount + 1));
|
1034
|
+
if (!reportp->messageArgs)
|
1035
|
+
return JS_FALSE;
|
1036
|
+
reportp->messageArgs[argCount] = NULL;
|
1037
|
+
for (i = 0; i < argCount; i++) {
|
1038
|
+
if (charArgs) {
|
1039
|
+
char *charArg = va_arg(ap, char *);
|
1040
|
+
size_t charArgLength = strlen(charArg);
|
1041
|
+
reportp->messageArgs[i]
|
1042
|
+
= js_InflateString(cx, charArg, &charArgLength);
|
1043
|
+
if (!reportp->messageArgs[i])
|
1044
|
+
goto error;
|
1045
|
+
} else {
|
1046
|
+
reportp->messageArgs[i] = va_arg(ap, jschar *);
|
1047
|
+
}
|
1048
|
+
argLengths[i] = js_strlen(reportp->messageArgs[i]);
|
1049
|
+
totalArgsLength += argLengths[i];
|
1050
|
+
}
|
1051
|
+
/* NULL-terminate for easy copying. */
|
1052
|
+
reportp->messageArgs[i] = NULL;
|
1053
|
+
}
|
1054
|
+
/*
|
1055
|
+
* Parse the error format, substituting the argument X
|
1056
|
+
* for {X} in the format.
|
1057
|
+
*/
|
1058
|
+
if (argCount > 0) {
|
1059
|
+
if (efs->format) {
|
1060
|
+
jschar *buffer, *fmt, *out;
|
1061
|
+
int expandedArgs = 0;
|
1062
|
+
size_t expandedLength;
|
1063
|
+
size_t len = strlen(efs->format);
|
1064
|
+
|
1065
|
+
buffer = fmt = js_InflateString (cx, efs->format, &len);
|
1066
|
+
if (!buffer)
|
1067
|
+
goto error;
|
1068
|
+
expandedLength = len
|
1069
|
+
- (3 * argCount) /* exclude the {n} */
|
1070
|
+
+ totalArgsLength;
|
1071
|
+
|
1072
|
+
/*
|
1073
|
+
* Note - the above calculation assumes that each argument
|
1074
|
+
* is used once and only once in the expansion !!!
|
1075
|
+
*/
|
1076
|
+
reportp->ucmessage = out = (jschar *)
|
1077
|
+
JS_malloc(cx, (expandedLength + 1) * sizeof(jschar));
|
1078
|
+
if (!out) {
|
1079
|
+
JS_free (cx, buffer);
|
1080
|
+
goto error;
|
1081
|
+
}
|
1082
|
+
while (*fmt) {
|
1083
|
+
if (*fmt == '{') {
|
1084
|
+
if (isdigit(fmt[1])) {
|
1085
|
+
int d = JS7_UNDEC(fmt[1]);
|
1086
|
+
JS_ASSERT(d < argCount);
|
1087
|
+
js_strncpy(out, reportp->messageArgs[d],
|
1088
|
+
argLengths[d]);
|
1089
|
+
out += argLengths[d];
|
1090
|
+
fmt += 3;
|
1091
|
+
expandedArgs++;
|
1092
|
+
continue;
|
1093
|
+
}
|
1094
|
+
}
|
1095
|
+
*out++ = *fmt++;
|
1096
|
+
}
|
1097
|
+
JS_ASSERT(expandedArgs == argCount);
|
1098
|
+
*out = 0;
|
1099
|
+
JS_free (cx, buffer);
|
1100
|
+
*messagep =
|
1101
|
+
js_DeflateString(cx, reportp->ucmessage,
|
1102
|
+
(size_t)(out - reportp->ucmessage));
|
1103
|
+
if (!*messagep)
|
1104
|
+
goto error;
|
1105
|
+
}
|
1106
|
+
} else {
|
1107
|
+
/*
|
1108
|
+
* Zero arguments: the format string (if it exists) is the
|
1109
|
+
* entire message.
|
1110
|
+
*/
|
1111
|
+
if (efs->format) {
|
1112
|
+
size_t len;
|
1113
|
+
*messagep = JS_strdup(cx, efs->format);
|
1114
|
+
if (!*messagep)
|
1115
|
+
goto error;
|
1116
|
+
len = strlen(*messagep);
|
1117
|
+
reportp->ucmessage = js_InflateString(cx, *messagep, &len);
|
1118
|
+
if (!reportp->ucmessage)
|
1119
|
+
goto error;
|
1120
|
+
}
|
1121
|
+
}
|
1122
|
+
}
|
1123
|
+
if (*messagep == NULL) {
|
1124
|
+
/* where's the right place for this ??? */
|
1125
|
+
const char *defaultErrorMessage
|
1126
|
+
= "No error message available for error number %d";
|
1127
|
+
size_t nbytes = strlen(defaultErrorMessage) + 16;
|
1128
|
+
*messagep = (char *)JS_malloc(cx, nbytes);
|
1129
|
+
if (!*messagep)
|
1130
|
+
goto error;
|
1131
|
+
JS_snprintf(*messagep, nbytes, defaultErrorMessage, errorNumber);
|
1132
|
+
}
|
1133
|
+
return JS_TRUE;
|
1134
|
+
|
1135
|
+
error:
|
1136
|
+
if (reportp->messageArgs) {
|
1137
|
+
/* free the arguments only if we allocated them */
|
1138
|
+
if (charArgs) {
|
1139
|
+
i = 0;
|
1140
|
+
while (reportp->messageArgs[i])
|
1141
|
+
JS_free(cx, (void *)reportp->messageArgs[i++]);
|
1142
|
+
}
|
1143
|
+
JS_free(cx, (void *)reportp->messageArgs);
|
1144
|
+
reportp->messageArgs = NULL;
|
1145
|
+
}
|
1146
|
+
if (reportp->ucmessage) {
|
1147
|
+
JS_free(cx, (void *)reportp->ucmessage);
|
1148
|
+
reportp->ucmessage = NULL;
|
1149
|
+
}
|
1150
|
+
if (*messagep) {
|
1151
|
+
JS_free(cx, (void *)*messagep);
|
1152
|
+
*messagep = NULL;
|
1153
|
+
}
|
1154
|
+
return JS_FALSE;
|
1155
|
+
}
|
1156
|
+
|
1157
|
+
JSBool
|
1158
|
+
js_ReportErrorNumberVA(JSContext *cx, uintN flags, JSErrorCallback callback,
|
1159
|
+
void *userRef, const uintN errorNumber,
|
1160
|
+
JSBool charArgs, va_list ap)
|
1161
|
+
{
|
1162
|
+
JSStackFrame *fp;
|
1163
|
+
JSErrorReport report;
|
1164
|
+
char *message;
|
1165
|
+
JSBool warning;
|
1166
|
+
|
1167
|
+
if ((flags & JSREPORT_STRICT) && !JS_HAS_STRICT_OPTION(cx))
|
1168
|
+
return JS_TRUE;
|
1169
|
+
|
1170
|
+
memset(&report, 0, sizeof (struct JSErrorReport));
|
1171
|
+
report.flags = flags;
|
1172
|
+
report.errorNumber = errorNumber;
|
1173
|
+
|
1174
|
+
/*
|
1175
|
+
* If we can't find out where the error was based on the current frame,
|
1176
|
+
* see if the next frame has a script/pc combo we can use.
|
1177
|
+
*/
|
1178
|
+
for (fp = cx->fp; fp; fp = fp->down) {
|
1179
|
+
if (fp->regs) {
|
1180
|
+
report.filename = fp->script->filename;
|
1181
|
+
report.lineno = js_PCToLineNumber(cx, fp->script, fp->regs->pc);
|
1182
|
+
break;
|
1183
|
+
}
|
1184
|
+
}
|
1185
|
+
|
1186
|
+
if (!js_ExpandErrorArguments(cx, callback, userRef, errorNumber,
|
1187
|
+
&message, &report, &warning, charArgs, ap)) {
|
1188
|
+
return JS_FALSE;
|
1189
|
+
}
|
1190
|
+
|
1191
|
+
ReportError(cx, message, &report);
|
1192
|
+
|
1193
|
+
if (message)
|
1194
|
+
JS_free(cx, message);
|
1195
|
+
if (report.messageArgs) {
|
1196
|
+
/*
|
1197
|
+
* js_ExpandErrorArguments owns its messageArgs only if it had to
|
1198
|
+
* inflate the arguments (from regular |char *|s).
|
1199
|
+
*/
|
1200
|
+
if (charArgs) {
|
1201
|
+
int i = 0;
|
1202
|
+
while (report.messageArgs[i])
|
1203
|
+
JS_free(cx, (void *)report.messageArgs[i++]);
|
1204
|
+
}
|
1205
|
+
JS_free(cx, (void *)report.messageArgs);
|
1206
|
+
}
|
1207
|
+
if (report.ucmessage)
|
1208
|
+
JS_free(cx, (void *)report.ucmessage);
|
1209
|
+
|
1210
|
+
return warning;
|
1211
|
+
}
|
1212
|
+
|
1213
|
+
JS_FRIEND_API(void)
|
1214
|
+
js_ReportErrorAgain(JSContext *cx, const char *message, JSErrorReport *reportp)
|
1215
|
+
{
|
1216
|
+
JSErrorReporter onError;
|
1217
|
+
|
1218
|
+
if (!message)
|
1219
|
+
return;
|
1220
|
+
|
1221
|
+
if (cx->lastMessage)
|
1222
|
+
free(cx->lastMessage);
|
1223
|
+
cx->lastMessage = JS_strdup(cx, message);
|
1224
|
+
if (!cx->lastMessage)
|
1225
|
+
return;
|
1226
|
+
onError = cx->errorReporter;
|
1227
|
+
|
1228
|
+
/*
|
1229
|
+
* If debugErrorHook is present then we give it a chance to veto
|
1230
|
+
* sending the error on to the regular ErrorReporter.
|
1231
|
+
*/
|
1232
|
+
if (onError) {
|
1233
|
+
JSDebugErrorHook hook = cx->debugHooks->debugErrorHook;
|
1234
|
+
if (hook &&
|
1235
|
+
!hook(cx, cx->lastMessage, reportp,
|
1236
|
+
cx->debugHooks->debugErrorHookData)) {
|
1237
|
+
onError = NULL;
|
1238
|
+
}
|
1239
|
+
}
|
1240
|
+
if (onError)
|
1241
|
+
onError(cx, cx->lastMessage, reportp);
|
1242
|
+
}
|
1243
|
+
|
1244
|
+
void
|
1245
|
+
js_ReportIsNotDefined(JSContext *cx, const char *name)
|
1246
|
+
{
|
1247
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_NOT_DEFINED, name);
|
1248
|
+
}
|
1249
|
+
|
1250
|
+
JSBool
|
1251
|
+
js_ReportIsNullOrUndefined(JSContext *cx, intN spindex, jsval v,
|
1252
|
+
JSString *fallback)
|
1253
|
+
{
|
1254
|
+
char *bytes;
|
1255
|
+
JSBool ok;
|
1256
|
+
|
1257
|
+
bytes = js_DecompileValueGenerator(cx, spindex, v, fallback);
|
1258
|
+
if (!bytes)
|
1259
|
+
return JS_FALSE;
|
1260
|
+
|
1261
|
+
if (strcmp(bytes, js_undefined_str) == 0 ||
|
1262
|
+
strcmp(bytes, js_null_str) == 0) {
|
1263
|
+
ok = JS_ReportErrorFlagsAndNumber(cx, JSREPORT_ERROR,
|
1264
|
+
js_GetErrorMessage, NULL,
|
1265
|
+
JSMSG_NO_PROPERTIES, bytes,
|
1266
|
+
NULL, NULL);
|
1267
|
+
} else if (JSVAL_IS_VOID(v)) {
|
1268
|
+
ok = JS_ReportErrorFlagsAndNumber(cx, JSREPORT_ERROR,
|
1269
|
+
js_GetErrorMessage, NULL,
|
1270
|
+
JSMSG_NULL_OR_UNDEFINED, bytes,
|
1271
|
+
js_undefined_str, NULL);
|
1272
|
+
} else {
|
1273
|
+
JS_ASSERT(JSVAL_IS_NULL(v));
|
1274
|
+
ok = JS_ReportErrorFlagsAndNumber(cx, JSREPORT_ERROR,
|
1275
|
+
js_GetErrorMessage, NULL,
|
1276
|
+
JSMSG_NULL_OR_UNDEFINED, bytes,
|
1277
|
+
js_null_str, NULL);
|
1278
|
+
}
|
1279
|
+
|
1280
|
+
JS_free(cx, bytes);
|
1281
|
+
return ok;
|
1282
|
+
}
|
1283
|
+
|
1284
|
+
JSBool
|
1285
|
+
js_ReportValueErrorFlags(JSContext *cx, uintN flags, const uintN errorNumber,
|
1286
|
+
intN spindex, jsval v, JSString *fallback,
|
1287
|
+
const char *arg1, const char *arg2)
|
1288
|
+
{
|
1289
|
+
char *bytes;
|
1290
|
+
JSBool ok;
|
1291
|
+
|
1292
|
+
JS_ASSERT(js_ErrorFormatString[errorNumber].argCount >= 1);
|
1293
|
+
JS_ASSERT(js_ErrorFormatString[errorNumber].argCount <= 3);
|
1294
|
+
bytes = js_DecompileValueGenerator(cx, spindex, v, fallback);
|
1295
|
+
if (!bytes)
|
1296
|
+
return JS_FALSE;
|
1297
|
+
|
1298
|
+
ok = JS_ReportErrorFlagsAndNumber(cx, flags, js_GetErrorMessage,
|
1299
|
+
NULL, errorNumber, bytes, arg1, arg2);
|
1300
|
+
JS_free(cx, bytes);
|
1301
|
+
return ok;
|
1302
|
+
}
|
1303
|
+
|
1304
|
+
#if defined DEBUG && defined XP_UNIX
|
1305
|
+
/* For gdb usage. */
|
1306
|
+
void js_traceon(JSContext *cx) { cx->tracefp = stderr; }
|
1307
|
+
void js_traceoff(JSContext *cx) { cx->tracefp = NULL; }
|
1308
|
+
#endif
|
1309
|
+
|
1310
|
+
JSErrorFormatString js_ErrorFormatString[JSErr_Limit] = {
|
1311
|
+
#define MSG_DEF(name, number, count, exception, format) \
|
1312
|
+
{ format, count, exception } ,
|
1313
|
+
#include "js.msg"
|
1314
|
+
#undef MSG_DEF
|
1315
|
+
};
|
1316
|
+
|
1317
|
+
const JSErrorFormatString *
|
1318
|
+
js_GetErrorMessage(void *userRef, const char *locale, const uintN errorNumber)
|
1319
|
+
{
|
1320
|
+
if ((errorNumber > 0) && (errorNumber < JSErr_Limit))
|
1321
|
+
return &js_ErrorFormatString[errorNumber];
|
1322
|
+
return NULL;
|
1323
|
+
}
|
1324
|
+
|
1325
|
+
JSBool
|
1326
|
+
js_ResetOperationCount(JSContext *cx)
|
1327
|
+
{
|
1328
|
+
JSScript *script;
|
1329
|
+
|
1330
|
+
JS_ASSERT(cx->operationCount <= 0);
|
1331
|
+
JS_ASSERT(cx->operationLimit > 0);
|
1332
|
+
|
1333
|
+
cx->operationCount = (int32) cx->operationLimit;
|
1334
|
+
if (cx->operationCallbackIsSet)
|
1335
|
+
return cx->operationCallback(cx);
|
1336
|
+
|
1337
|
+
if (cx->operationCallback) {
|
1338
|
+
/*
|
1339
|
+
* Invoke the deprecated branch callback. It may be called only when
|
1340
|
+
* the top-most frame is scripted or JSOPTION_NATIVE_BRANCH_CALLBACK
|
1341
|
+
* is set.
|
1342
|
+
*/
|
1343
|
+
script = cx->fp ? cx->fp->script : NULL;
|
1344
|
+
if (script || JS_HAS_OPTION(cx, JSOPTION_NATIVE_BRANCH_CALLBACK))
|
1345
|
+
return ((JSBranchCallback) cx->operationCallback)(cx, script);
|
1346
|
+
}
|
1347
|
+
return JS_TRUE;
|
1348
|
+
}
|