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,1371 @@
|
|
1
|
+
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2
|
+
* vim: set ts=8 sw=4 et tw=78:
|
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 standard exception implementation.
|
43
|
+
*/
|
44
|
+
|
45
|
+
#include "jsstddef.h"
|
46
|
+
#include <stdlib.h>
|
47
|
+
#include <string.h>
|
48
|
+
#include "jstypes.h"
|
49
|
+
#include "jsbit.h"
|
50
|
+
#include "jsutil.h" /* Added by JSIFY */
|
51
|
+
#include "jsprf.h"
|
52
|
+
#include "jsapi.h"
|
53
|
+
#include "jscntxt.h"
|
54
|
+
#include "jsconfig.h"
|
55
|
+
#include "jsdbgapi.h"
|
56
|
+
#include "jsexn.h"
|
57
|
+
#include "jsfun.h"
|
58
|
+
#include "jsinterp.h"
|
59
|
+
#include "jsnum.h"
|
60
|
+
#include "jsobj.h"
|
61
|
+
#include "jsopcode.h"
|
62
|
+
#include "jsscope.h"
|
63
|
+
#include "jsscript.h"
|
64
|
+
|
65
|
+
/* Forward declarations for js_ErrorClass's initializer. */
|
66
|
+
static JSBool
|
67
|
+
Exception(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval);
|
68
|
+
|
69
|
+
static void
|
70
|
+
exn_finalize(JSContext *cx, JSObject *obj);
|
71
|
+
|
72
|
+
static void
|
73
|
+
exn_trace(JSTracer *trc, JSObject *obj);
|
74
|
+
|
75
|
+
static void
|
76
|
+
exn_finalize(JSContext *cx, JSObject *obj);
|
77
|
+
|
78
|
+
static JSBool
|
79
|
+
exn_enumerate(JSContext *cx, JSObject *obj);
|
80
|
+
|
81
|
+
static JSBool
|
82
|
+
exn_resolve(JSContext *cx, JSObject *obj, jsval id, uintN flags,
|
83
|
+
JSObject **objp);
|
84
|
+
|
85
|
+
JSClass js_ErrorClass = {
|
86
|
+
js_Error_str,
|
87
|
+
JSCLASS_HAS_PRIVATE | JSCLASS_NEW_RESOLVE | JSCLASS_MARK_IS_TRACE |
|
88
|
+
JSCLASS_HAS_CACHED_PROTO(JSProto_Error),
|
89
|
+
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
|
90
|
+
exn_enumerate, (JSResolveOp)exn_resolve, JS_ConvertStub, exn_finalize,
|
91
|
+
NULL, NULL, NULL, Exception,
|
92
|
+
NULL, NULL, JS_CLASS_TRACE(exn_trace), NULL
|
93
|
+
};
|
94
|
+
|
95
|
+
typedef struct JSStackTraceElem {
|
96
|
+
JSString *funName;
|
97
|
+
size_t argc;
|
98
|
+
const char *filename;
|
99
|
+
uintN ulineno;
|
100
|
+
} JSStackTraceElem;
|
101
|
+
|
102
|
+
typedef struct JSExnPrivate {
|
103
|
+
/* A copy of the JSErrorReport originally generated. */
|
104
|
+
JSErrorReport *errorReport;
|
105
|
+
JSString *message;
|
106
|
+
JSString *filename;
|
107
|
+
uintN lineno;
|
108
|
+
size_t stackDepth;
|
109
|
+
JSStackTraceElem stackElems[1];
|
110
|
+
} JSExnPrivate;
|
111
|
+
|
112
|
+
static JSString *
|
113
|
+
StackTraceToString(JSContext *cx, JSExnPrivate *priv);
|
114
|
+
|
115
|
+
static JSErrorReport *
|
116
|
+
CopyErrorReport(JSContext *cx, JSErrorReport *report)
|
117
|
+
{
|
118
|
+
/*
|
119
|
+
* We use a single malloc block to make a deep copy of JSErrorReport with
|
120
|
+
* the following layout:
|
121
|
+
* JSErrorReport
|
122
|
+
* array of copies of report->messageArgs
|
123
|
+
* jschar array with characters for all messageArgs
|
124
|
+
* jschar array with characters for ucmessage
|
125
|
+
* jschar array with characters for uclinebuf and uctokenptr
|
126
|
+
* char array with characters for linebuf and tokenptr
|
127
|
+
* char array with characters for filename
|
128
|
+
* Such layout together with the properties enforced by the following
|
129
|
+
* asserts does not need any extra alignment padding.
|
130
|
+
*/
|
131
|
+
JS_STATIC_ASSERT(sizeof(JSErrorReport) % sizeof(const char *) == 0);
|
132
|
+
JS_STATIC_ASSERT(sizeof(const char *) % sizeof(jschar) == 0);
|
133
|
+
|
134
|
+
size_t filenameSize;
|
135
|
+
size_t linebufSize;
|
136
|
+
size_t uclinebufSize;
|
137
|
+
size_t ucmessageSize;
|
138
|
+
size_t i, argsArraySize, argsCopySize, argSize;
|
139
|
+
size_t mallocSize;
|
140
|
+
JSErrorReport *copy;
|
141
|
+
uint8 *cursor;
|
142
|
+
|
143
|
+
#define JS_CHARS_SIZE(jschars) ((js_strlen(jschars) + 1) * sizeof(jschar))
|
144
|
+
|
145
|
+
filenameSize = report->filename ? strlen(report->filename) + 1 : 0;
|
146
|
+
linebufSize = report->linebuf ? strlen(report->linebuf) + 1 : 0;
|
147
|
+
uclinebufSize = report->uclinebuf ? JS_CHARS_SIZE(report->uclinebuf) : 0;
|
148
|
+
ucmessageSize = 0;
|
149
|
+
argsArraySize = 0;
|
150
|
+
argsCopySize = 0;
|
151
|
+
if (report->ucmessage) {
|
152
|
+
ucmessageSize = JS_CHARS_SIZE(report->ucmessage);
|
153
|
+
if (report->messageArgs) {
|
154
|
+
for (i = 0; report->messageArgs[i]; ++i)
|
155
|
+
argsCopySize += JS_CHARS_SIZE(report->messageArgs[i]);
|
156
|
+
|
157
|
+
/* Non-null messageArgs should have at least one non-null arg. */
|
158
|
+
JS_ASSERT(i != 0);
|
159
|
+
argsArraySize = (i + 1) * sizeof(const jschar *);
|
160
|
+
}
|
161
|
+
}
|
162
|
+
|
163
|
+
/*
|
164
|
+
* The mallocSize can not overflow since it represents the sum of the
|
165
|
+
* sizes of already allocated objects.
|
166
|
+
*/
|
167
|
+
mallocSize = sizeof(JSErrorReport) + argsArraySize + argsCopySize +
|
168
|
+
ucmessageSize + uclinebufSize + linebufSize + filenameSize;
|
169
|
+
cursor = (uint8 *)JS_malloc(cx, mallocSize);
|
170
|
+
if (!cursor)
|
171
|
+
return NULL;
|
172
|
+
|
173
|
+
copy = (JSErrorReport *)cursor;
|
174
|
+
memset(cursor, 0, sizeof(JSErrorReport));
|
175
|
+
cursor += sizeof(JSErrorReport);
|
176
|
+
|
177
|
+
if (argsArraySize != 0) {
|
178
|
+
copy->messageArgs = (const jschar **)cursor;
|
179
|
+
cursor += argsArraySize;
|
180
|
+
for (i = 0; report->messageArgs[i]; ++i) {
|
181
|
+
copy->messageArgs[i] = (const jschar *)cursor;
|
182
|
+
argSize = JS_CHARS_SIZE(report->messageArgs[i]);
|
183
|
+
memcpy(cursor, report->messageArgs[i], argSize);
|
184
|
+
cursor += argSize;
|
185
|
+
}
|
186
|
+
copy->messageArgs[i] = NULL;
|
187
|
+
JS_ASSERT(cursor == (uint8 *)copy->messageArgs[0] + argsCopySize);
|
188
|
+
}
|
189
|
+
|
190
|
+
if (report->ucmessage) {
|
191
|
+
copy->ucmessage = (const jschar *)cursor;
|
192
|
+
memcpy(cursor, report->ucmessage, ucmessageSize);
|
193
|
+
cursor += ucmessageSize;
|
194
|
+
}
|
195
|
+
|
196
|
+
if (report->uclinebuf) {
|
197
|
+
copy->uclinebuf = (const jschar *)cursor;
|
198
|
+
memcpy(cursor, report->uclinebuf, uclinebufSize);
|
199
|
+
cursor += uclinebufSize;
|
200
|
+
if (report->uctokenptr) {
|
201
|
+
copy->uctokenptr = copy->uclinebuf + (report->uctokenptr -
|
202
|
+
report->uclinebuf);
|
203
|
+
}
|
204
|
+
}
|
205
|
+
|
206
|
+
if (report->linebuf) {
|
207
|
+
copy->linebuf = (const char *)cursor;
|
208
|
+
memcpy(cursor, report->linebuf, linebufSize);
|
209
|
+
cursor += linebufSize;
|
210
|
+
if (report->tokenptr) {
|
211
|
+
copy->tokenptr = copy->linebuf + (report->tokenptr -
|
212
|
+
report->linebuf);
|
213
|
+
}
|
214
|
+
}
|
215
|
+
|
216
|
+
if (report->filename) {
|
217
|
+
copy->filename = (const char *)cursor;
|
218
|
+
memcpy(cursor, report->filename, filenameSize);
|
219
|
+
}
|
220
|
+
JS_ASSERT(cursor + filenameSize == (uint8 *)copy + mallocSize);
|
221
|
+
|
222
|
+
/* Copy non-pointer members. */
|
223
|
+
copy->lineno = report->lineno;
|
224
|
+
copy->errorNumber = report->errorNumber;
|
225
|
+
|
226
|
+
/* Note that this is before it gets flagged with JSREPORT_EXCEPTION */
|
227
|
+
copy->flags = report->flags;
|
228
|
+
|
229
|
+
#undef JS_CHARS_SIZE
|
230
|
+
return copy;
|
231
|
+
}
|
232
|
+
|
233
|
+
static jsval *
|
234
|
+
GetStackTraceValueBuffer(JSExnPrivate *priv)
|
235
|
+
{
|
236
|
+
/*
|
237
|
+
* We use extra memory after JSExnPrivateInfo.stackElems to store jsvals
|
238
|
+
* that helps to produce more informative stack traces. The following
|
239
|
+
* assert allows us to assume that no gap after stackElems is necessary to
|
240
|
+
* align the buffer properly.
|
241
|
+
*/
|
242
|
+
JS_STATIC_ASSERT(sizeof(JSStackTraceElem) % sizeof(jsval) == 0);
|
243
|
+
|
244
|
+
return (jsval *)(priv->stackElems + priv->stackDepth);
|
245
|
+
}
|
246
|
+
|
247
|
+
static JSBool
|
248
|
+
InitExnPrivate(JSContext *cx, JSObject *exnObject, JSString *message,
|
249
|
+
JSString *filename, uintN lineno, JSErrorReport *report)
|
250
|
+
{
|
251
|
+
JSCheckAccessOp checkAccess;
|
252
|
+
JSErrorReporter older;
|
253
|
+
JSExceptionState *state;
|
254
|
+
jsval callerid, v;
|
255
|
+
JSStackFrame *fp, *fpstop;
|
256
|
+
size_t stackDepth, valueCount, size;
|
257
|
+
JSBool overflow;
|
258
|
+
JSExnPrivate *priv;
|
259
|
+
JSStackTraceElem *elem;
|
260
|
+
jsval *values;
|
261
|
+
|
262
|
+
JS_ASSERT(OBJ_GET_CLASS(cx, exnObject) == &js_ErrorClass);
|
263
|
+
|
264
|
+
/*
|
265
|
+
* Prepare stack trace data.
|
266
|
+
*
|
267
|
+
* Set aside any error reporter for cx and save its exception state
|
268
|
+
* so we can suppress any checkAccess failures. Such failures should stop
|
269
|
+
* the backtrace procedure, not result in a failure of this constructor.
|
270
|
+
*/
|
271
|
+
checkAccess = cx->runtime->checkObjectAccess;
|
272
|
+
older = JS_SetErrorReporter(cx, NULL);
|
273
|
+
state = JS_SaveExceptionState(cx);
|
274
|
+
|
275
|
+
callerid = ATOM_KEY(cx->runtime->atomState.callerAtom);
|
276
|
+
stackDepth = 0;
|
277
|
+
valueCount = 0;
|
278
|
+
for (fp = cx->fp; fp; fp = fp->down) {
|
279
|
+
if (fp->fun && fp->argv) {
|
280
|
+
v = JSVAL_NULL;
|
281
|
+
if (checkAccess &&
|
282
|
+
!checkAccess(cx, fp->callee, callerid, JSACC_READ, &v)) {
|
283
|
+
break;
|
284
|
+
}
|
285
|
+
valueCount += fp->argc;
|
286
|
+
}
|
287
|
+
++stackDepth;
|
288
|
+
}
|
289
|
+
JS_RestoreExceptionState(cx, state);
|
290
|
+
JS_SetErrorReporter(cx, older);
|
291
|
+
fpstop = fp;
|
292
|
+
|
293
|
+
size = offsetof(JSExnPrivate, stackElems);
|
294
|
+
overflow = (stackDepth > ((size_t)-1 - size) / sizeof(JSStackTraceElem));
|
295
|
+
size += stackDepth * sizeof(JSStackTraceElem);
|
296
|
+
overflow |= (valueCount > ((size_t)-1 - size) / sizeof(jsval));
|
297
|
+
size += valueCount * sizeof(jsval);
|
298
|
+
if (overflow) {
|
299
|
+
js_ReportAllocationOverflow(cx);
|
300
|
+
return JS_FALSE;
|
301
|
+
}
|
302
|
+
priv = (JSExnPrivate *)JS_malloc(cx, size);
|
303
|
+
if (!priv)
|
304
|
+
return JS_FALSE;
|
305
|
+
|
306
|
+
/*
|
307
|
+
* We initialize errorReport with a copy of report after setting the
|
308
|
+
* private slot, to prevent GC accessing a junk value we clear the field
|
309
|
+
* here.
|
310
|
+
*/
|
311
|
+
priv->errorReport = NULL;
|
312
|
+
priv->message = message;
|
313
|
+
priv->filename = filename;
|
314
|
+
priv->lineno = lineno;
|
315
|
+
priv->stackDepth = stackDepth;
|
316
|
+
|
317
|
+
values = GetStackTraceValueBuffer(priv);
|
318
|
+
elem = priv->stackElems;
|
319
|
+
for (fp = cx->fp; fp != fpstop; fp = fp->down) {
|
320
|
+
if (!fp->fun) {
|
321
|
+
elem->funName = NULL;
|
322
|
+
elem->argc = 0;
|
323
|
+
} else {
|
324
|
+
elem->funName = fp->fun->atom
|
325
|
+
? ATOM_TO_STRING(fp->fun->atom)
|
326
|
+
: cx->runtime->emptyString;
|
327
|
+
elem->argc = fp->argc;
|
328
|
+
memcpy(values, fp->argv, fp->argc * sizeof(jsval));
|
329
|
+
values += fp->argc;
|
330
|
+
}
|
331
|
+
elem->ulineno = 0;
|
332
|
+
elem->filename = NULL;
|
333
|
+
if (fp->script) {
|
334
|
+
elem->filename = fp->script->filename;
|
335
|
+
if (fp->regs)
|
336
|
+
elem->ulineno = js_PCToLineNumber(cx, fp->script, fp->regs->pc);
|
337
|
+
}
|
338
|
+
++elem;
|
339
|
+
}
|
340
|
+
JS_ASSERT(priv->stackElems + stackDepth == elem);
|
341
|
+
JS_ASSERT(GetStackTraceValueBuffer(priv) + valueCount == values);
|
342
|
+
|
343
|
+
STOBJ_SET_SLOT(exnObject, JSSLOT_PRIVATE, PRIVATE_TO_JSVAL(priv));
|
344
|
+
|
345
|
+
if (report) {
|
346
|
+
/*
|
347
|
+
* Construct a new copy of the error report struct. We can't use the
|
348
|
+
* error report struct that was passed in, because it's allocated on
|
349
|
+
* the stack, and also because it may point to transient data in the
|
350
|
+
* JSTokenStream.
|
351
|
+
*/
|
352
|
+
priv->errorReport = CopyErrorReport(cx, report);
|
353
|
+
if (!priv->errorReport) {
|
354
|
+
/* The finalizer realeases priv since it is in the private slot. */
|
355
|
+
return JS_FALSE;
|
356
|
+
}
|
357
|
+
}
|
358
|
+
|
359
|
+
return JS_TRUE;
|
360
|
+
}
|
361
|
+
|
362
|
+
static JSExnPrivate *
|
363
|
+
GetExnPrivate(JSContext *cx, JSObject *obj)
|
364
|
+
{
|
365
|
+
jsval privateValue;
|
366
|
+
JSExnPrivate *priv;
|
367
|
+
|
368
|
+
JS_ASSERT(OBJ_GET_CLASS(cx, obj) == &js_ErrorClass);
|
369
|
+
privateValue = OBJ_GET_SLOT(cx, obj, JSSLOT_PRIVATE);
|
370
|
+
if (JSVAL_IS_VOID(privateValue))
|
371
|
+
return NULL;
|
372
|
+
priv = (JSExnPrivate *)JSVAL_TO_PRIVATE(privateValue);
|
373
|
+
JS_ASSERT(priv);
|
374
|
+
return priv;
|
375
|
+
}
|
376
|
+
|
377
|
+
static void
|
378
|
+
exn_trace(JSTracer *trc, JSObject *obj)
|
379
|
+
{
|
380
|
+
JSExnPrivate *priv;
|
381
|
+
JSStackTraceElem *elem;
|
382
|
+
size_t vcount, i;
|
383
|
+
jsval *vp, v;
|
384
|
+
|
385
|
+
priv = GetExnPrivate(trc->context, obj);
|
386
|
+
if (priv) {
|
387
|
+
if (priv->message)
|
388
|
+
JS_CALL_STRING_TRACER(trc, priv->message, "exception message");
|
389
|
+
if (priv->filename)
|
390
|
+
JS_CALL_STRING_TRACER(trc, priv->filename, "exception filename");
|
391
|
+
|
392
|
+
elem = priv->stackElems;
|
393
|
+
for (vcount = i = 0; i != priv->stackDepth; ++i, ++elem) {
|
394
|
+
if (elem->funName) {
|
395
|
+
JS_CALL_STRING_TRACER(trc, elem->funName,
|
396
|
+
"stack trace function name");
|
397
|
+
}
|
398
|
+
if (IS_GC_MARKING_TRACER(trc) && elem->filename)
|
399
|
+
js_MarkScriptFilename(elem->filename);
|
400
|
+
vcount += elem->argc;
|
401
|
+
}
|
402
|
+
vp = GetStackTraceValueBuffer(priv);
|
403
|
+
for (i = 0; i != vcount; ++i, ++vp) {
|
404
|
+
v = *vp;
|
405
|
+
JS_CALL_VALUE_TRACER(trc, v, "stack trace argument");
|
406
|
+
}
|
407
|
+
}
|
408
|
+
}
|
409
|
+
|
410
|
+
static void
|
411
|
+
exn_finalize(JSContext *cx, JSObject *obj)
|
412
|
+
{
|
413
|
+
JSExnPrivate *priv;
|
414
|
+
|
415
|
+
priv = GetExnPrivate(cx, obj);
|
416
|
+
if (priv) {
|
417
|
+
if (priv->errorReport)
|
418
|
+
JS_free(cx, priv->errorReport);
|
419
|
+
JS_free(cx, priv);
|
420
|
+
}
|
421
|
+
}
|
422
|
+
|
423
|
+
static JSBool
|
424
|
+
exn_enumerate(JSContext *cx, JSObject *obj)
|
425
|
+
{
|
426
|
+
JSAtomState *atomState;
|
427
|
+
uintN i;
|
428
|
+
JSAtom *atom;
|
429
|
+
JSObject *pobj;
|
430
|
+
JSProperty *prop;
|
431
|
+
|
432
|
+
JS_STATIC_ASSERT(sizeof(JSAtomState) <= (size_t)(uint16)-1);
|
433
|
+
static const uint16 offsets[] = {
|
434
|
+
(uint16)offsetof(JSAtomState, messageAtom),
|
435
|
+
(uint16)offsetof(JSAtomState, fileNameAtom),
|
436
|
+
(uint16)offsetof(JSAtomState, lineNumberAtom),
|
437
|
+
(uint16)offsetof(JSAtomState, stackAtom),
|
438
|
+
};
|
439
|
+
|
440
|
+
atomState = &cx->runtime->atomState;
|
441
|
+
for (i = 0; i != JS_ARRAY_LENGTH(offsets); ++i) {
|
442
|
+
atom = *(JSAtom **)((uint8 *)atomState + offsets[i]);
|
443
|
+
if (!js_LookupProperty(cx, obj, ATOM_TO_JSID(atom), &pobj, &prop))
|
444
|
+
return JS_FALSE;
|
445
|
+
if (prop)
|
446
|
+
OBJ_DROP_PROPERTY(cx, pobj, prop);
|
447
|
+
}
|
448
|
+
return JS_TRUE;
|
449
|
+
}
|
450
|
+
|
451
|
+
static JSBool
|
452
|
+
exn_resolve(JSContext *cx, JSObject *obj, jsval id, uintN flags,
|
453
|
+
JSObject **objp)
|
454
|
+
{
|
455
|
+
JSExnPrivate *priv;
|
456
|
+
JSString *str;
|
457
|
+
JSAtom *atom;
|
458
|
+
JSString *stack;
|
459
|
+
const char *prop;
|
460
|
+
jsval v;
|
461
|
+
|
462
|
+
*objp = NULL;
|
463
|
+
priv = GetExnPrivate(cx, obj);
|
464
|
+
if (priv && JSVAL_IS_STRING(id)) {
|
465
|
+
str = JSVAL_TO_STRING(id);
|
466
|
+
|
467
|
+
atom = cx->runtime->atomState.messageAtom;
|
468
|
+
if (str == ATOM_TO_STRING(atom)) {
|
469
|
+
prop = js_message_str;
|
470
|
+
v = STRING_TO_JSVAL(priv->message);
|
471
|
+
goto define;
|
472
|
+
}
|
473
|
+
|
474
|
+
atom = cx->runtime->atomState.fileNameAtom;
|
475
|
+
if (str == ATOM_TO_STRING(atom)) {
|
476
|
+
prop = js_fileName_str;
|
477
|
+
v = STRING_TO_JSVAL(priv->filename);
|
478
|
+
goto define;
|
479
|
+
}
|
480
|
+
|
481
|
+
atom = cx->runtime->atomState.lineNumberAtom;
|
482
|
+
if (str == ATOM_TO_STRING(atom)) {
|
483
|
+
prop = js_lineNumber_str;
|
484
|
+
v = INT_TO_JSVAL(priv->lineno);
|
485
|
+
goto define;
|
486
|
+
}
|
487
|
+
|
488
|
+
atom = cx->runtime->atomState.stackAtom;
|
489
|
+
if (str == ATOM_TO_STRING(atom)) {
|
490
|
+
stack = StackTraceToString(cx, priv);
|
491
|
+
if (!stack)
|
492
|
+
return JS_FALSE;
|
493
|
+
|
494
|
+
/* Allow to GC all things that were used to build stack trace. */
|
495
|
+
priv->stackDepth = 0;
|
496
|
+
prop = js_stack_str;
|
497
|
+
v = STRING_TO_JSVAL(stack);
|
498
|
+
goto define;
|
499
|
+
}
|
500
|
+
}
|
501
|
+
return JS_TRUE;
|
502
|
+
|
503
|
+
define:
|
504
|
+
if (!JS_DefineProperty(cx, obj, prop, v, NULL, NULL, JSPROP_ENUMERATE))
|
505
|
+
return JS_FALSE;
|
506
|
+
*objp = obj;
|
507
|
+
return JS_TRUE;
|
508
|
+
}
|
509
|
+
|
510
|
+
JSErrorReport *
|
511
|
+
js_ErrorFromException(JSContext *cx, jsval exn)
|
512
|
+
{
|
513
|
+
JSObject *obj;
|
514
|
+
JSExnPrivate *priv;
|
515
|
+
|
516
|
+
if (JSVAL_IS_PRIMITIVE(exn))
|
517
|
+
return NULL;
|
518
|
+
obj = JSVAL_TO_OBJECT(exn);
|
519
|
+
if (OBJ_GET_CLASS(cx, obj) != &js_ErrorClass)
|
520
|
+
return NULL;
|
521
|
+
priv = GetExnPrivate(cx, obj);
|
522
|
+
if (!priv)
|
523
|
+
return NULL;
|
524
|
+
return priv->errorReport;
|
525
|
+
}
|
526
|
+
|
527
|
+
struct JSExnSpec {
|
528
|
+
int protoIndex;
|
529
|
+
const char *name;
|
530
|
+
JSProtoKey key;
|
531
|
+
JSNative native;
|
532
|
+
};
|
533
|
+
|
534
|
+
/*
|
535
|
+
* All *Error constructors share the same JSClass, js_ErrorClass. But each
|
536
|
+
* constructor function for an *Error class must have a distinct native 'call'
|
537
|
+
* function pointer, in order for instanceof to work properly across multiple
|
538
|
+
* standard class sets. See jsfun.c:fun_hasInstance.
|
539
|
+
*/
|
540
|
+
#define MAKE_EXCEPTION_CTOR(name) \
|
541
|
+
static JSBool \
|
542
|
+
name(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) \
|
543
|
+
{ \
|
544
|
+
return Exception(cx, obj, argc, argv, rval); \
|
545
|
+
}
|
546
|
+
|
547
|
+
MAKE_EXCEPTION_CTOR(Error)
|
548
|
+
MAKE_EXCEPTION_CTOR(InternalError)
|
549
|
+
MAKE_EXCEPTION_CTOR(EvalError)
|
550
|
+
MAKE_EXCEPTION_CTOR(RangeError)
|
551
|
+
MAKE_EXCEPTION_CTOR(ReferenceError)
|
552
|
+
MAKE_EXCEPTION_CTOR(SyntaxError)
|
553
|
+
MAKE_EXCEPTION_CTOR(TypeError)
|
554
|
+
MAKE_EXCEPTION_CTOR(URIError)
|
555
|
+
|
556
|
+
#undef MAKE_EXCEPTION_CTOR
|
557
|
+
|
558
|
+
static struct JSExnSpec exceptions[] = {
|
559
|
+
{JSEXN_NONE, js_Error_str, JSProto_Error, Error},
|
560
|
+
{JSEXN_ERR, js_InternalError_str, JSProto_InternalError, InternalError},
|
561
|
+
{JSEXN_ERR, js_EvalError_str, JSProto_EvalError, EvalError},
|
562
|
+
{JSEXN_ERR, js_RangeError_str, JSProto_RangeError, RangeError},
|
563
|
+
{JSEXN_ERR, js_ReferenceError_str, JSProto_ReferenceError, ReferenceError},
|
564
|
+
{JSEXN_ERR, js_SyntaxError_str, JSProto_SyntaxError, SyntaxError},
|
565
|
+
{JSEXN_ERR, js_TypeError_str, JSProto_TypeError, TypeError},
|
566
|
+
{JSEXN_ERR, js_URIError_str, JSProto_URIError, URIError},
|
567
|
+
{0, NULL, JSProto_Null, NULL}
|
568
|
+
};
|
569
|
+
|
570
|
+
static JSString *
|
571
|
+
ValueToShortSource(JSContext *cx, jsval v)
|
572
|
+
{
|
573
|
+
JSString *str;
|
574
|
+
|
575
|
+
/* Avoid toSource bloat and fallibility for object types. */
|
576
|
+
if (JSVAL_IS_PRIMITIVE(v)) {
|
577
|
+
str = js_ValueToSource(cx, v);
|
578
|
+
} else if (VALUE_IS_FUNCTION(cx, v)) {
|
579
|
+
/*
|
580
|
+
* XXX Avoid function decompilation bloat for now.
|
581
|
+
*/
|
582
|
+
str = JS_GetFunctionId(JS_ValueToFunction(cx, v));
|
583
|
+
if (!str && !(str = js_ValueToSource(cx, v))) {
|
584
|
+
/*
|
585
|
+
* Continue to soldier on if the function couldn't be
|
586
|
+
* converted into a string.
|
587
|
+
*/
|
588
|
+
JS_ClearPendingException(cx);
|
589
|
+
str = JS_NewStringCopyZ(cx, "[unknown function]");
|
590
|
+
}
|
591
|
+
} else {
|
592
|
+
/*
|
593
|
+
* XXX Avoid toString on objects, it takes too long and uses too much
|
594
|
+
* memory, for too many classes (see Mozilla bug 166743).
|
595
|
+
*/
|
596
|
+
char buf[100];
|
597
|
+
JS_snprintf(buf, sizeof buf, "[object %s]",
|
598
|
+
OBJ_GET_CLASS(cx, JSVAL_TO_OBJECT(v))->name);
|
599
|
+
str = JS_NewStringCopyZ(cx, buf);
|
600
|
+
}
|
601
|
+
return str;
|
602
|
+
}
|
603
|
+
|
604
|
+
static JSString *
|
605
|
+
StackTraceToString(JSContext *cx, JSExnPrivate *priv)
|
606
|
+
{
|
607
|
+
jschar *stackbuf;
|
608
|
+
size_t stacklen, stackmax;
|
609
|
+
JSStackTraceElem *elem, *endElem;
|
610
|
+
jsval *values;
|
611
|
+
size_t i;
|
612
|
+
JSString *str;
|
613
|
+
const char *cp;
|
614
|
+
char ulnbuf[11];
|
615
|
+
|
616
|
+
/* After this point, failing control flow must goto bad. */
|
617
|
+
stackbuf = NULL;
|
618
|
+
stacklen = stackmax = 0;
|
619
|
+
|
620
|
+
/* Limit the stackbuf length to a reasonable value to avoid overflow checks. */
|
621
|
+
#define STACK_LENGTH_LIMIT JS_BIT(20)
|
622
|
+
|
623
|
+
#define APPEND_CHAR_TO_STACK(c) \
|
624
|
+
JS_BEGIN_MACRO \
|
625
|
+
if (stacklen == stackmax) { \
|
626
|
+
void *ptr_; \
|
627
|
+
if (stackmax >= STACK_LENGTH_LIMIT) \
|
628
|
+
goto done; \
|
629
|
+
stackmax = stackmax ? 2 * stackmax : 64; \
|
630
|
+
ptr_ = JS_realloc(cx, stackbuf, (stackmax+1) * sizeof(jschar)); \
|
631
|
+
if (!ptr_) \
|
632
|
+
goto bad; \
|
633
|
+
stackbuf = (jschar *) ptr_; \
|
634
|
+
} \
|
635
|
+
stackbuf[stacklen++] = (c); \
|
636
|
+
JS_END_MACRO
|
637
|
+
|
638
|
+
#define APPEND_STRING_TO_STACK(str) \
|
639
|
+
JS_BEGIN_MACRO \
|
640
|
+
JSString *str_ = str; \
|
641
|
+
jschar *chars_; \
|
642
|
+
size_t length_; \
|
643
|
+
\
|
644
|
+
JSSTRING_CHARS_AND_LENGTH(str_, chars_, length_); \
|
645
|
+
if (length_ > stackmax - stacklen) { \
|
646
|
+
void *ptr_; \
|
647
|
+
if (stackmax >= STACK_LENGTH_LIMIT || \
|
648
|
+
length_ >= STACK_LENGTH_LIMIT - stacklen) { \
|
649
|
+
goto done; \
|
650
|
+
} \
|
651
|
+
stackmax = JS_BIT(JS_CeilingLog2(stacklen + length_)); \
|
652
|
+
ptr_ = JS_realloc(cx, stackbuf, (stackmax+1) * sizeof(jschar)); \
|
653
|
+
if (!ptr_) \
|
654
|
+
goto bad; \
|
655
|
+
stackbuf = (jschar *) ptr_; \
|
656
|
+
} \
|
657
|
+
js_strncpy(stackbuf + stacklen, chars_, length_); \
|
658
|
+
stacklen += length_; \
|
659
|
+
JS_END_MACRO
|
660
|
+
|
661
|
+
values = GetStackTraceValueBuffer(priv);
|
662
|
+
elem = priv->stackElems;
|
663
|
+
for (endElem = elem + priv->stackDepth; elem != endElem; elem++) {
|
664
|
+
if (elem->funName) {
|
665
|
+
APPEND_STRING_TO_STACK(elem->funName);
|
666
|
+
APPEND_CHAR_TO_STACK('(');
|
667
|
+
for (i = 0; i != elem->argc; i++, values++) {
|
668
|
+
if (i > 0)
|
669
|
+
APPEND_CHAR_TO_STACK(',');
|
670
|
+
str = ValueToShortSource(cx, *values);
|
671
|
+
if (!str)
|
672
|
+
goto bad;
|
673
|
+
APPEND_STRING_TO_STACK(str);
|
674
|
+
}
|
675
|
+
APPEND_CHAR_TO_STACK(')');
|
676
|
+
}
|
677
|
+
APPEND_CHAR_TO_STACK('@');
|
678
|
+
if (elem->filename) {
|
679
|
+
for (cp = elem->filename; *cp; cp++)
|
680
|
+
APPEND_CHAR_TO_STACK(*cp);
|
681
|
+
}
|
682
|
+
APPEND_CHAR_TO_STACK(':');
|
683
|
+
JS_snprintf(ulnbuf, sizeof ulnbuf, "%u", elem->ulineno);
|
684
|
+
for (cp = ulnbuf; *cp; cp++)
|
685
|
+
APPEND_CHAR_TO_STACK(*cp);
|
686
|
+
APPEND_CHAR_TO_STACK('\n');
|
687
|
+
}
|
688
|
+
#undef APPEND_CHAR_TO_STACK
|
689
|
+
#undef APPEND_STRING_TO_STACK
|
690
|
+
#undef STACK_LENGTH_LIMIT
|
691
|
+
|
692
|
+
done:
|
693
|
+
if (stacklen == 0) {
|
694
|
+
JS_ASSERT(!stackbuf);
|
695
|
+
return cx->runtime->emptyString;
|
696
|
+
}
|
697
|
+
if (stacklen < stackmax) {
|
698
|
+
/*
|
699
|
+
* Realloc can fail when shrinking on some FreeBSD versions, so
|
700
|
+
* don't use JS_realloc here; simply let the oversized allocation
|
701
|
+
* be owned by the string in that rare case.
|
702
|
+
*/
|
703
|
+
void *shrunk = JS_realloc(cx, stackbuf, (stacklen+1) * sizeof(jschar));
|
704
|
+
if (shrunk)
|
705
|
+
stackbuf = (jschar *) shrunk;
|
706
|
+
}
|
707
|
+
|
708
|
+
stackbuf[stacklen] = 0;
|
709
|
+
str = js_NewString(cx, stackbuf, stacklen);
|
710
|
+
if (str)
|
711
|
+
return str;
|
712
|
+
|
713
|
+
bad:
|
714
|
+
if (stackbuf)
|
715
|
+
JS_free(cx, stackbuf);
|
716
|
+
return NULL;
|
717
|
+
}
|
718
|
+
|
719
|
+
/* XXXbe Consolidate the ugly truth that we don't treat filename as UTF-8
|
720
|
+
with these two functions. */
|
721
|
+
static JSString *
|
722
|
+
FilenameToString(JSContext *cx, const char *filename)
|
723
|
+
{
|
724
|
+
return JS_NewStringCopyZ(cx, filename);
|
725
|
+
}
|
726
|
+
|
727
|
+
static const char *
|
728
|
+
StringToFilename(JSContext *cx, JSString *str)
|
729
|
+
{
|
730
|
+
return js_GetStringBytes(cx, str);
|
731
|
+
}
|
732
|
+
|
733
|
+
static JSBool
|
734
|
+
Exception(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
735
|
+
{
|
736
|
+
uint32 lineno;
|
737
|
+
JSString *message, *filename;
|
738
|
+
JSStackFrame *fp;
|
739
|
+
|
740
|
+
if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) {
|
741
|
+
/*
|
742
|
+
* ECMA ed. 3, 15.11.1 requires Error, etc., to construct even when
|
743
|
+
* called as functions, without operator new. But as we do not give
|
744
|
+
* each constructor a distinct JSClass, whose .name member is used by
|
745
|
+
* js_NewObject to find the class prototype, we must get the class
|
746
|
+
* prototype ourselves.
|
747
|
+
*/
|
748
|
+
if (!OBJ_GET_PROPERTY(cx, JSVAL_TO_OBJECT(argv[-2]),
|
749
|
+
ATOM_TO_JSID(cx->runtime->atomState
|
750
|
+
.classPrototypeAtom),
|
751
|
+
rval))
|
752
|
+
return JS_FALSE;
|
753
|
+
obj = js_NewObject(cx, &js_ErrorClass, JSVAL_TO_OBJECT(*rval), NULL, 0);
|
754
|
+
if (!obj)
|
755
|
+
return JS_FALSE;
|
756
|
+
*rval = OBJECT_TO_JSVAL(obj);
|
757
|
+
}
|
758
|
+
|
759
|
+
/*
|
760
|
+
* If it's a new object of class Exception, then null out the private
|
761
|
+
* data so that the finalizer doesn't attempt to free it.
|
762
|
+
*/
|
763
|
+
if (OBJ_GET_CLASS(cx, obj) == &js_ErrorClass)
|
764
|
+
STOBJ_SET_SLOT(obj, JSSLOT_PRIVATE, JSVAL_VOID);
|
765
|
+
|
766
|
+
/* Set the 'message' property. */
|
767
|
+
if (argc != 0) {
|
768
|
+
message = js_ValueToString(cx, argv[0]);
|
769
|
+
if (!message)
|
770
|
+
return JS_FALSE;
|
771
|
+
argv[0] = STRING_TO_JSVAL(message);
|
772
|
+
} else {
|
773
|
+
message = cx->runtime->emptyString;
|
774
|
+
}
|
775
|
+
|
776
|
+
/* Set the 'fileName' property. */
|
777
|
+
if (argc > 1) {
|
778
|
+
filename = js_ValueToString(cx, argv[1]);
|
779
|
+
if (!filename)
|
780
|
+
return JS_FALSE;
|
781
|
+
argv[1] = STRING_TO_JSVAL(filename);
|
782
|
+
fp = NULL;
|
783
|
+
} else {
|
784
|
+
fp = JS_GetScriptedCaller(cx, NULL);
|
785
|
+
if (fp) {
|
786
|
+
filename = FilenameToString(cx, fp->script->filename);
|
787
|
+
if (!filename)
|
788
|
+
return JS_FALSE;
|
789
|
+
} else {
|
790
|
+
filename = cx->runtime->emptyString;
|
791
|
+
}
|
792
|
+
}
|
793
|
+
|
794
|
+
/* Set the 'lineNumber' property. */
|
795
|
+
if (argc > 2) {
|
796
|
+
lineno = js_ValueToECMAUint32(cx, &argv[2]);
|
797
|
+
if (JSVAL_IS_NULL(argv[2]))
|
798
|
+
return JS_FALSE;
|
799
|
+
} else {
|
800
|
+
if (!fp)
|
801
|
+
fp = JS_GetScriptedCaller(cx, NULL);
|
802
|
+
lineno = (fp && fp->regs)
|
803
|
+
? js_PCToLineNumber(cx, fp->script, fp->regs->pc)
|
804
|
+
: 0;
|
805
|
+
}
|
806
|
+
|
807
|
+
return (OBJ_GET_CLASS(cx, obj) != &js_ErrorClass) ||
|
808
|
+
InitExnPrivate(cx, obj, message, filename, lineno, NULL);
|
809
|
+
}
|
810
|
+
|
811
|
+
/*
|
812
|
+
* Convert to string.
|
813
|
+
*
|
814
|
+
* This method only uses JavaScript-modifiable properties name, message. It
|
815
|
+
* is left to the host to check for private data and report filename and line
|
816
|
+
* number information along with this message.
|
817
|
+
*/
|
818
|
+
static JSBool
|
819
|
+
exn_toString(JSContext *cx, uintN argc, jsval *vp)
|
820
|
+
{
|
821
|
+
JSObject *obj;
|
822
|
+
jsval v;
|
823
|
+
JSString *name, *message, *result;
|
824
|
+
jschar *chars, *cp;
|
825
|
+
size_t name_length, message_length, length;
|
826
|
+
|
827
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
828
|
+
if (!obj ||
|
829
|
+
!OBJ_GET_PROPERTY(cx, obj,
|
830
|
+
ATOM_TO_JSID(cx->runtime->atomState.nameAtom),
|
831
|
+
&v)) {
|
832
|
+
return JS_FALSE;
|
833
|
+
}
|
834
|
+
name = JSVAL_IS_STRING(v) ? JSVAL_TO_STRING(v) : cx->runtime->emptyString;
|
835
|
+
*vp = STRING_TO_JSVAL(name);
|
836
|
+
|
837
|
+
if (!JS_GetProperty(cx, obj, js_message_str, &v))
|
838
|
+
return JS_FALSE;
|
839
|
+
message = JSVAL_IS_STRING(v) ? JSVAL_TO_STRING(v)
|
840
|
+
: cx->runtime->emptyString;
|
841
|
+
|
842
|
+
if (JSSTRING_LENGTH(message) != 0) {
|
843
|
+
name_length = JSSTRING_LENGTH(name);
|
844
|
+
message_length = JSSTRING_LENGTH(message);
|
845
|
+
length = (name_length ? name_length + 2 : 0) + message_length;
|
846
|
+
cp = chars = (jschar *) JS_malloc(cx, (length + 1) * sizeof(jschar));
|
847
|
+
if (!chars)
|
848
|
+
return JS_FALSE;
|
849
|
+
|
850
|
+
if (name_length) {
|
851
|
+
js_strncpy(cp, JSSTRING_CHARS(name), name_length);
|
852
|
+
cp += name_length;
|
853
|
+
*cp++ = ':'; *cp++ = ' ';
|
854
|
+
}
|
855
|
+
js_strncpy(cp, JSSTRING_CHARS(message), message_length);
|
856
|
+
cp += message_length;
|
857
|
+
*cp = 0;
|
858
|
+
|
859
|
+
result = js_NewString(cx, chars, length);
|
860
|
+
if (!result) {
|
861
|
+
JS_free(cx, chars);
|
862
|
+
return JS_FALSE;
|
863
|
+
}
|
864
|
+
} else {
|
865
|
+
result = name;
|
866
|
+
}
|
867
|
+
|
868
|
+
*vp = STRING_TO_JSVAL(result);
|
869
|
+
return JS_TRUE;
|
870
|
+
}
|
871
|
+
|
872
|
+
#if JS_HAS_TOSOURCE
|
873
|
+
/*
|
874
|
+
* Return a string that may eval to something similar to the original object.
|
875
|
+
*/
|
876
|
+
static JSBool
|
877
|
+
exn_toSource(JSContext *cx, uintN argc, jsval *vp)
|
878
|
+
{
|
879
|
+
JSObject *obj;
|
880
|
+
JSString *name, *message, *filename, *lineno_as_str, *result;
|
881
|
+
jsval localroots[3] = {JSVAL_NULL, JSVAL_NULL, JSVAL_NULL};
|
882
|
+
JSTempValueRooter tvr;
|
883
|
+
JSBool ok;
|
884
|
+
uint32 lineno;
|
885
|
+
size_t lineno_length, name_length, message_length, filename_length, length;
|
886
|
+
jschar *chars, *cp;
|
887
|
+
|
888
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
889
|
+
if (!obj ||
|
890
|
+
!OBJ_GET_PROPERTY(cx, obj,
|
891
|
+
ATOM_TO_JSID(cx->runtime->atomState.nameAtom),
|
892
|
+
vp)) {
|
893
|
+
return JS_FALSE;
|
894
|
+
}
|
895
|
+
name = js_ValueToString(cx, *vp);
|
896
|
+
if (!name)
|
897
|
+
return JS_FALSE;
|
898
|
+
*vp = STRING_TO_JSVAL(name);
|
899
|
+
|
900
|
+
/* After this, control must flow through label out: to exit. */
|
901
|
+
JS_PUSH_TEMP_ROOT(cx, 3, localroots, &tvr);
|
902
|
+
|
903
|
+
ok = JS_GetProperty(cx, obj, js_message_str, &localroots[0]) &&
|
904
|
+
(message = js_ValueToSource(cx, localroots[0]));
|
905
|
+
if (!ok)
|
906
|
+
goto out;
|
907
|
+
localroots[0] = STRING_TO_JSVAL(message);
|
908
|
+
|
909
|
+
ok = JS_GetProperty(cx, obj, js_fileName_str, &localroots[1]) &&
|
910
|
+
(filename = js_ValueToSource(cx, localroots[1]));
|
911
|
+
if (!ok)
|
912
|
+
goto out;
|
913
|
+
localroots[1] = STRING_TO_JSVAL(filename);
|
914
|
+
|
915
|
+
ok = JS_GetProperty(cx, obj, js_lineNumber_str, &localroots[2]);
|
916
|
+
if (!ok)
|
917
|
+
goto out;
|
918
|
+
lineno = js_ValueToECMAUint32 (cx, &localroots[2]);
|
919
|
+
ok = !JSVAL_IS_NULL(localroots[2]);
|
920
|
+
if (!ok)
|
921
|
+
goto out;
|
922
|
+
|
923
|
+
if (lineno != 0) {
|
924
|
+
lineno_as_str = js_ValueToString(cx, localroots[2]);
|
925
|
+
if (!lineno_as_str) {
|
926
|
+
ok = JS_FALSE;
|
927
|
+
goto out;
|
928
|
+
}
|
929
|
+
lineno_length = JSSTRING_LENGTH(lineno_as_str);
|
930
|
+
} else {
|
931
|
+
lineno_as_str = NULL;
|
932
|
+
lineno_length = 0;
|
933
|
+
}
|
934
|
+
|
935
|
+
/* Magic 8, for the characters in ``(new ())''. */
|
936
|
+
name_length = JSSTRING_LENGTH(name);
|
937
|
+
message_length = JSSTRING_LENGTH(message);
|
938
|
+
length = 8 + name_length + message_length;
|
939
|
+
|
940
|
+
filename_length = JSSTRING_LENGTH(filename);
|
941
|
+
if (filename_length != 0) {
|
942
|
+
/* append filename as ``, {filename}'' */
|
943
|
+
length += 2 + filename_length;
|
944
|
+
if (lineno_as_str) {
|
945
|
+
/* append lineno as ``, {lineno_as_str}'' */
|
946
|
+
length += 2 + lineno_length;
|
947
|
+
}
|
948
|
+
} else {
|
949
|
+
if (lineno_as_str) {
|
950
|
+
/*
|
951
|
+
* no filename, but have line number,
|
952
|
+
* need to append ``, "", {lineno_as_str}''
|
953
|
+
*/
|
954
|
+
length += 6 + lineno_length;
|
955
|
+
}
|
956
|
+
}
|
957
|
+
|
958
|
+
cp = chars = (jschar *) JS_malloc(cx, (length + 1) * sizeof(jschar));
|
959
|
+
if (!chars) {
|
960
|
+
ok = JS_FALSE;
|
961
|
+
goto out;
|
962
|
+
}
|
963
|
+
|
964
|
+
*cp++ = '('; *cp++ = 'n'; *cp++ = 'e'; *cp++ = 'w'; *cp++ = ' ';
|
965
|
+
js_strncpy(cp, JSSTRING_CHARS(name), name_length);
|
966
|
+
cp += name_length;
|
967
|
+
*cp++ = '(';
|
968
|
+
if (message_length != 0) {
|
969
|
+
js_strncpy(cp, JSSTRING_CHARS(message), message_length);
|
970
|
+
cp += message_length;
|
971
|
+
}
|
972
|
+
|
973
|
+
if (filename_length != 0) {
|
974
|
+
/* append filename as ``, {filename}'' */
|
975
|
+
*cp++ = ','; *cp++ = ' ';
|
976
|
+
js_strncpy(cp, JSSTRING_CHARS(filename), filename_length);
|
977
|
+
cp += filename_length;
|
978
|
+
} else {
|
979
|
+
if (lineno_as_str) {
|
980
|
+
/*
|
981
|
+
* no filename, but have line number,
|
982
|
+
* need to append ``, "", {lineno_as_str}''
|
983
|
+
*/
|
984
|
+
*cp++ = ','; *cp++ = ' '; *cp++ = '"'; *cp++ = '"';
|
985
|
+
}
|
986
|
+
}
|
987
|
+
if (lineno_as_str) {
|
988
|
+
/* append lineno as ``, {lineno_as_str}'' */
|
989
|
+
*cp++ = ','; *cp++ = ' ';
|
990
|
+
js_strncpy(cp, JSSTRING_CHARS(lineno_as_str), lineno_length);
|
991
|
+
cp += lineno_length;
|
992
|
+
}
|
993
|
+
|
994
|
+
*cp++ = ')'; *cp++ = ')'; *cp = 0;
|
995
|
+
|
996
|
+
result = js_NewString(cx, chars, length);
|
997
|
+
if (!result) {
|
998
|
+
JS_free(cx, chars);
|
999
|
+
ok = JS_FALSE;
|
1000
|
+
goto out;
|
1001
|
+
}
|
1002
|
+
*vp = STRING_TO_JSVAL(result);
|
1003
|
+
ok = JS_TRUE;
|
1004
|
+
|
1005
|
+
out:
|
1006
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
1007
|
+
return ok;
|
1008
|
+
}
|
1009
|
+
#endif
|
1010
|
+
|
1011
|
+
static JSFunctionSpec exception_methods[] = {
|
1012
|
+
#if JS_HAS_TOSOURCE
|
1013
|
+
JS_FN(js_toSource_str, exn_toSource, 0,0,0),
|
1014
|
+
#endif
|
1015
|
+
JS_FN(js_toString_str, exn_toString, 0,0,0),
|
1016
|
+
JS_FS_END
|
1017
|
+
};
|
1018
|
+
|
1019
|
+
JSObject *
|
1020
|
+
js_InitExceptionClasses(JSContext *cx, JSObject *obj)
|
1021
|
+
{
|
1022
|
+
JSObject *obj_proto, *protos[JSEXN_LIMIT];
|
1023
|
+
int i;
|
1024
|
+
|
1025
|
+
/*
|
1026
|
+
* If lazy class initialization occurs for any Error subclass, then all
|
1027
|
+
* classes are initialized, starting with Error. To avoid reentry and
|
1028
|
+
* redundant initialization, we must not pass a null proto parameter to
|
1029
|
+
* js_NewObject below, when called for the Error superclass. We need to
|
1030
|
+
* ensure that Object.prototype is the proto of Error.prototype.
|
1031
|
+
*
|
1032
|
+
* See the equivalent code to ensure that parent_proto is non-null when
|
1033
|
+
* JS_InitClass calls js_NewObject, in jsapi.c.
|
1034
|
+
*/
|
1035
|
+
if (!js_GetClassPrototype(cx, obj, INT_TO_JSID(JSProto_Object),
|
1036
|
+
&obj_proto)) {
|
1037
|
+
return NULL;
|
1038
|
+
}
|
1039
|
+
|
1040
|
+
if (!js_EnterLocalRootScope(cx))
|
1041
|
+
return NULL;
|
1042
|
+
|
1043
|
+
/* Initialize the prototypes first. */
|
1044
|
+
for (i = 0; exceptions[i].name != 0; i++) {
|
1045
|
+
JSAtom *atom;
|
1046
|
+
JSFunction *fun;
|
1047
|
+
JSString *nameString;
|
1048
|
+
int protoIndex = exceptions[i].protoIndex;
|
1049
|
+
|
1050
|
+
/* Make the prototype for the current constructor name. */
|
1051
|
+
protos[i] = js_NewObject(cx, &js_ErrorClass,
|
1052
|
+
(protoIndex != JSEXN_NONE)
|
1053
|
+
? protos[protoIndex]
|
1054
|
+
: obj_proto,
|
1055
|
+
obj, 0);
|
1056
|
+
if (!protos[i])
|
1057
|
+
break;
|
1058
|
+
|
1059
|
+
/* So exn_finalize knows whether to destroy private data. */
|
1060
|
+
STOBJ_SET_SLOT(protos[i], JSSLOT_PRIVATE, JSVAL_VOID);
|
1061
|
+
|
1062
|
+
/* Make a constructor function for the current name. */
|
1063
|
+
atom = cx->runtime->atomState.classAtoms[exceptions[i].key];
|
1064
|
+
fun = js_DefineFunction(cx, obj, atom, exceptions[i].native, 3, 0);
|
1065
|
+
if (!fun)
|
1066
|
+
break;
|
1067
|
+
|
1068
|
+
/* Make this constructor make objects of class Exception. */
|
1069
|
+
fun->u.n.clasp = &js_ErrorClass;
|
1070
|
+
|
1071
|
+
/* Make the prototype and constructor links. */
|
1072
|
+
if (!js_SetClassPrototype(cx, FUN_OBJECT(fun), protos[i],
|
1073
|
+
JSPROP_READONLY | JSPROP_PERMANENT)) {
|
1074
|
+
break;
|
1075
|
+
}
|
1076
|
+
|
1077
|
+
/* proto bootstrap bit from JS_InitClass omitted. */
|
1078
|
+
nameString = JS_NewStringCopyZ(cx, exceptions[i].name);
|
1079
|
+
if (!nameString)
|
1080
|
+
break;
|
1081
|
+
|
1082
|
+
/* Add the name property to the prototype. */
|
1083
|
+
if (!JS_DefineProperty(cx, protos[i], js_name_str,
|
1084
|
+
STRING_TO_JSVAL(nameString),
|
1085
|
+
NULL, NULL,
|
1086
|
+
JSPROP_ENUMERATE)) {
|
1087
|
+
break;
|
1088
|
+
}
|
1089
|
+
|
1090
|
+
/* Finally, stash the constructor for later uses. */
|
1091
|
+
if (!js_SetClassObject(cx, obj, exceptions[i].key, FUN_OBJECT(fun)))
|
1092
|
+
break;
|
1093
|
+
}
|
1094
|
+
|
1095
|
+
js_LeaveLocalRootScope(cx);
|
1096
|
+
if (exceptions[i].name)
|
1097
|
+
return NULL;
|
1098
|
+
|
1099
|
+
/*
|
1100
|
+
* Add an empty message property. (To Exception.prototype only,
|
1101
|
+
* because this property will be the same for all the exception
|
1102
|
+
* protos.)
|
1103
|
+
*/
|
1104
|
+
if (!JS_DefineProperty(cx, protos[0], js_message_str,
|
1105
|
+
STRING_TO_JSVAL(cx->runtime->emptyString),
|
1106
|
+
NULL, NULL, JSPROP_ENUMERATE)) {
|
1107
|
+
return NULL;
|
1108
|
+
}
|
1109
|
+
if (!JS_DefineProperty(cx, protos[0], js_fileName_str,
|
1110
|
+
STRING_TO_JSVAL(cx->runtime->emptyString),
|
1111
|
+
NULL, NULL, JSPROP_ENUMERATE)) {
|
1112
|
+
return NULL;
|
1113
|
+
}
|
1114
|
+
if (!JS_DefineProperty(cx, protos[0], js_lineNumber_str,
|
1115
|
+
INT_TO_JSVAL(0),
|
1116
|
+
NULL, NULL, JSPROP_ENUMERATE)) {
|
1117
|
+
return NULL;
|
1118
|
+
}
|
1119
|
+
|
1120
|
+
/*
|
1121
|
+
* Add methods only to Exception.prototype, because ostensibly all
|
1122
|
+
* exception types delegate to that.
|
1123
|
+
*/
|
1124
|
+
if (!JS_DefineFunctions(cx, protos[0], exception_methods))
|
1125
|
+
return NULL;
|
1126
|
+
|
1127
|
+
return protos[0];
|
1128
|
+
}
|
1129
|
+
|
1130
|
+
const JSErrorFormatString*
|
1131
|
+
js_GetLocalizedErrorMessage(JSContext* cx, void *userRef, const char *locale,
|
1132
|
+
const uintN errorNumber)
|
1133
|
+
{
|
1134
|
+
const JSErrorFormatString *errorString = NULL;
|
1135
|
+
|
1136
|
+
if (cx->localeCallbacks && cx->localeCallbacks->localeGetErrorMessage) {
|
1137
|
+
errorString = cx->localeCallbacks
|
1138
|
+
->localeGetErrorMessage(userRef, locale, errorNumber);
|
1139
|
+
}
|
1140
|
+
if (!errorString)
|
1141
|
+
errorString = js_GetErrorMessage(userRef, locale, errorNumber);
|
1142
|
+
return errorString;
|
1143
|
+
}
|
1144
|
+
|
1145
|
+
#if defined ( DEBUG_mccabe ) && defined ( PRINTNAMES )
|
1146
|
+
/* For use below... get character strings for error name and exception name */
|
1147
|
+
static struct exnname { char *name; char *exception; } errortoexnname[] = {
|
1148
|
+
#define MSG_DEF(name, number, count, exception, format) \
|
1149
|
+
{#name, #exception},
|
1150
|
+
#include "js.msg"
|
1151
|
+
#undef MSG_DEF
|
1152
|
+
};
|
1153
|
+
#endif /* DEBUG */
|
1154
|
+
|
1155
|
+
JSBool
|
1156
|
+
js_ErrorToException(JSContext *cx, const char *message, JSErrorReport *reportp)
|
1157
|
+
{
|
1158
|
+
JSErrNum errorNumber;
|
1159
|
+
const JSErrorFormatString *errorString;
|
1160
|
+
JSExnType exn;
|
1161
|
+
jsval tv[4];
|
1162
|
+
JSTempValueRooter tvr;
|
1163
|
+
JSBool ok;
|
1164
|
+
JSObject *errProto, *errObject;
|
1165
|
+
JSString *messageStr, *filenameStr;
|
1166
|
+
|
1167
|
+
/*
|
1168
|
+
* Tell our caller to report immediately if cx has no active frames, or if
|
1169
|
+
* this report is just a warning.
|
1170
|
+
*/
|
1171
|
+
JS_ASSERT(reportp);
|
1172
|
+
if (!cx->fp || JSREPORT_IS_WARNING(reportp->flags))
|
1173
|
+
return JS_FALSE;
|
1174
|
+
|
1175
|
+
/* Find the exception index associated with this error. */
|
1176
|
+
errorNumber = (JSErrNum) reportp->errorNumber;
|
1177
|
+
errorString = js_GetLocalizedErrorMessage(cx, NULL, NULL, errorNumber);
|
1178
|
+
exn = errorString ? (JSExnType) errorString->exnType : JSEXN_NONE;
|
1179
|
+
JS_ASSERT(exn < JSEXN_LIMIT);
|
1180
|
+
|
1181
|
+
#if defined( DEBUG_mccabe ) && defined ( PRINTNAMES )
|
1182
|
+
/* Print the error name and the associated exception name to stderr */
|
1183
|
+
fprintf(stderr, "%s\t%s\n",
|
1184
|
+
errortoexnname[errorNumber].name,
|
1185
|
+
errortoexnname[errorNumber].exception);
|
1186
|
+
#endif
|
1187
|
+
|
1188
|
+
/*
|
1189
|
+
* Return false (no exception raised) if no exception is associated
|
1190
|
+
* with the given error number.
|
1191
|
+
*/
|
1192
|
+
if (exn == JSEXN_NONE)
|
1193
|
+
return JS_FALSE;
|
1194
|
+
|
1195
|
+
/*
|
1196
|
+
* Prevent runaway recursion, via cx->generatingError. If an out-of-memory
|
1197
|
+
* error occurs, no exception object will be created, but we don't assume
|
1198
|
+
* that OOM is the only kind of error that subroutines of this function
|
1199
|
+
* called below might raise.
|
1200
|
+
*/
|
1201
|
+
if (cx->generatingError)
|
1202
|
+
return JS_FALSE;
|
1203
|
+
|
1204
|
+
/* After this point the control must flow through the label out. */
|
1205
|
+
cx->generatingError = JS_TRUE;
|
1206
|
+
|
1207
|
+
/* Protect the newly-created strings below from nesting GCs. */
|
1208
|
+
memset(tv, 0, sizeof tv);
|
1209
|
+
JS_PUSH_TEMP_ROOT(cx, JS_ARRAY_LENGTH(tv), tv, &tvr);
|
1210
|
+
|
1211
|
+
/*
|
1212
|
+
* Try to get an appropriate prototype by looking up the corresponding
|
1213
|
+
* exception constructor name in the scope chain of the current context's
|
1214
|
+
* top stack frame, or in the global object if no frame is active.
|
1215
|
+
*/
|
1216
|
+
ok = js_GetClassPrototype(cx, NULL, INT_TO_JSID(exceptions[exn].key),
|
1217
|
+
&errProto);
|
1218
|
+
if (!ok)
|
1219
|
+
goto out;
|
1220
|
+
tv[0] = OBJECT_TO_JSVAL(errProto);
|
1221
|
+
|
1222
|
+
errObject = js_NewObject(cx, &js_ErrorClass, errProto, NULL, 0);
|
1223
|
+
if (!errObject) {
|
1224
|
+
ok = JS_FALSE;
|
1225
|
+
goto out;
|
1226
|
+
}
|
1227
|
+
tv[1] = OBJECT_TO_JSVAL(errObject);
|
1228
|
+
|
1229
|
+
messageStr = JS_NewStringCopyZ(cx, message);
|
1230
|
+
if (!messageStr) {
|
1231
|
+
ok = JS_FALSE;
|
1232
|
+
goto out;
|
1233
|
+
}
|
1234
|
+
tv[2] = STRING_TO_JSVAL(messageStr);
|
1235
|
+
|
1236
|
+
filenameStr = JS_NewStringCopyZ(cx, reportp->filename);
|
1237
|
+
if (!filenameStr) {
|
1238
|
+
ok = JS_FALSE;
|
1239
|
+
goto out;
|
1240
|
+
}
|
1241
|
+
tv[3] = STRING_TO_JSVAL(filenameStr);
|
1242
|
+
|
1243
|
+
ok = InitExnPrivate(cx, errObject, messageStr, filenameStr,
|
1244
|
+
reportp->lineno, reportp);
|
1245
|
+
if (!ok)
|
1246
|
+
goto out;
|
1247
|
+
|
1248
|
+
JS_SetPendingException(cx, OBJECT_TO_JSVAL(errObject));
|
1249
|
+
|
1250
|
+
/* Flag the error report passed in to indicate an exception was raised. */
|
1251
|
+
reportp->flags |= JSREPORT_EXCEPTION;
|
1252
|
+
|
1253
|
+
out:
|
1254
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
1255
|
+
cx->generatingError = JS_FALSE;
|
1256
|
+
return ok;
|
1257
|
+
}
|
1258
|
+
|
1259
|
+
JSBool
|
1260
|
+
js_ReportUncaughtException(JSContext *cx)
|
1261
|
+
{
|
1262
|
+
jsval exn;
|
1263
|
+
JSObject *exnObject;
|
1264
|
+
jsval roots[5];
|
1265
|
+
JSTempValueRooter tvr;
|
1266
|
+
JSErrorReport *reportp, report;
|
1267
|
+
JSString *str;
|
1268
|
+
const char *bytes;
|
1269
|
+
JSBool ok;
|
1270
|
+
|
1271
|
+
if (!JS_IsExceptionPending(cx))
|
1272
|
+
return JS_TRUE;
|
1273
|
+
|
1274
|
+
if (!JS_GetPendingException(cx, &exn))
|
1275
|
+
return JS_FALSE;
|
1276
|
+
|
1277
|
+
memset(roots, 0, sizeof roots);
|
1278
|
+
JS_PUSH_TEMP_ROOT(cx, JS_ARRAY_LENGTH(roots), roots, &tvr);
|
1279
|
+
|
1280
|
+
/*
|
1281
|
+
* Because js_ValueToString below could error and an exception object
|
1282
|
+
* could become unrooted, we must root exnObject. Later, if exnObject is
|
1283
|
+
* non-null, we need to root other intermediates, so allocate an operand
|
1284
|
+
* stack segment to protect all of these values.
|
1285
|
+
*/
|
1286
|
+
if (JSVAL_IS_PRIMITIVE(exn)) {
|
1287
|
+
exnObject = NULL;
|
1288
|
+
} else {
|
1289
|
+
exnObject = JSVAL_TO_OBJECT(exn);
|
1290
|
+
roots[0] = exn;
|
1291
|
+
}
|
1292
|
+
|
1293
|
+
JS_ClearPendingException(cx);
|
1294
|
+
reportp = js_ErrorFromException(cx, exn);
|
1295
|
+
|
1296
|
+
/* XXX L10N angels cry once again (see also jsemit.c, /L10N gaffes/) */
|
1297
|
+
str = js_ValueToString(cx, exn);
|
1298
|
+
if (!str) {
|
1299
|
+
bytes = "unknown (can't convert to string)";
|
1300
|
+
} else {
|
1301
|
+
roots[1] = STRING_TO_JSVAL(str);
|
1302
|
+
bytes = js_GetStringBytes(cx, str);
|
1303
|
+
if (!bytes) {
|
1304
|
+
ok = JS_FALSE;
|
1305
|
+
goto out;
|
1306
|
+
}
|
1307
|
+
}
|
1308
|
+
ok = JS_TRUE;
|
1309
|
+
|
1310
|
+
if (!reportp &&
|
1311
|
+
exnObject &&
|
1312
|
+
OBJ_GET_CLASS(cx, exnObject) == &js_ErrorClass) {
|
1313
|
+
const char *filename;
|
1314
|
+
uint32 lineno;
|
1315
|
+
|
1316
|
+
ok = JS_GetProperty(cx, exnObject, js_message_str, &roots[2]);
|
1317
|
+
if (!ok)
|
1318
|
+
goto out;
|
1319
|
+
if (JSVAL_IS_STRING(roots[2])) {
|
1320
|
+
bytes = js_GetStringBytes(cx, JSVAL_TO_STRING(roots[2]));
|
1321
|
+
if (!bytes) {
|
1322
|
+
ok = JS_FALSE;
|
1323
|
+
goto out;
|
1324
|
+
}
|
1325
|
+
}
|
1326
|
+
|
1327
|
+
ok = JS_GetProperty(cx, exnObject, js_fileName_str, &roots[3]);
|
1328
|
+
if (!ok)
|
1329
|
+
goto out;
|
1330
|
+
str = js_ValueToString(cx, roots[3]);
|
1331
|
+
if (!str) {
|
1332
|
+
ok = JS_FALSE;
|
1333
|
+
goto out;
|
1334
|
+
}
|
1335
|
+
filename = StringToFilename(cx, str);
|
1336
|
+
if (!filename) {
|
1337
|
+
ok = JS_FALSE;
|
1338
|
+
goto out;
|
1339
|
+
}
|
1340
|
+
|
1341
|
+
ok = JS_GetProperty(cx, exnObject, js_lineNumber_str, &roots[4]);
|
1342
|
+
if (!ok)
|
1343
|
+
goto out;
|
1344
|
+
lineno = js_ValueToECMAUint32 (cx, &roots[4]);
|
1345
|
+
ok = !JSVAL_IS_NULL(roots[4]);
|
1346
|
+
if (!ok)
|
1347
|
+
goto out;
|
1348
|
+
|
1349
|
+
reportp = &report;
|
1350
|
+
memset(&report, 0, sizeof report);
|
1351
|
+
report.filename = filename;
|
1352
|
+
report.lineno = (uintN) lineno;
|
1353
|
+
}
|
1354
|
+
|
1355
|
+
if (!reportp) {
|
1356
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
1357
|
+
JSMSG_UNCAUGHT_EXCEPTION, bytes);
|
1358
|
+
} else {
|
1359
|
+
/* Flag the error as an exception. */
|
1360
|
+
reportp->flags |= JSREPORT_EXCEPTION;
|
1361
|
+
|
1362
|
+
/* Pass the exception object. */
|
1363
|
+
JS_SetPendingException(cx, exn);
|
1364
|
+
js_ReportErrorAgain(cx, bytes, reportp);
|
1365
|
+
JS_ClearPendingException(cx);
|
1366
|
+
}
|
1367
|
+
|
1368
|
+
out:
|
1369
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
1370
|
+
return ok;
|
1371
|
+
}
|