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,124 @@
|
|
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 Date class interface.
|
42
|
+
*/
|
43
|
+
|
44
|
+
#ifndef jsdate_h___
|
45
|
+
#define jsdate_h___
|
46
|
+
|
47
|
+
JS_BEGIN_EXTERN_C
|
48
|
+
|
49
|
+
extern JSClass js_DateClass;
|
50
|
+
|
51
|
+
extern JSObject *
|
52
|
+
js_InitDateClass(JSContext *cx, JSObject *obj);
|
53
|
+
|
54
|
+
/*
|
55
|
+
* These functions provide a C interface to the date/time object
|
56
|
+
*/
|
57
|
+
|
58
|
+
/*
|
59
|
+
* Construct a new Date Object from a time value given in milliseconds UTC
|
60
|
+
* since the epoch.
|
61
|
+
*/
|
62
|
+
extern JS_FRIEND_API(JSObject*)
|
63
|
+
js_NewDateObjectMsec(JSContext* cx, jsdouble msec_time);
|
64
|
+
|
65
|
+
/*
|
66
|
+
* Construct a new Date Object from an exploded local time value.
|
67
|
+
*
|
68
|
+
* Assert that mon < 12 to help catch off-by-one user errors, which are common
|
69
|
+
* due to the 0-based month numbering copied into JS from Java (java.util.Date
|
70
|
+
* in 1995). js_DateSetMonth (below) likewise asserts month < 12.
|
71
|
+
*/
|
72
|
+
extern JS_FRIEND_API(JSObject*)
|
73
|
+
js_NewDateObject(JSContext* cx, int year, int mon, int mday,
|
74
|
+
int hour, int min, int sec);
|
75
|
+
|
76
|
+
/*
|
77
|
+
* Detect whether the internal date value is NaN. (Because failure is
|
78
|
+
* out-of-band for js_DateGet*)
|
79
|
+
*/
|
80
|
+
extern JS_FRIEND_API(JSBool)
|
81
|
+
js_DateIsValid(JSContext *cx, JSObject* obj);
|
82
|
+
|
83
|
+
extern JS_FRIEND_API(int)
|
84
|
+
js_DateGetYear(JSContext *cx, JSObject* obj);
|
85
|
+
|
86
|
+
extern JS_FRIEND_API(int)
|
87
|
+
js_DateGetMonth(JSContext *cx, JSObject* obj);
|
88
|
+
|
89
|
+
extern JS_FRIEND_API(int)
|
90
|
+
js_DateGetDate(JSContext *cx, JSObject* obj);
|
91
|
+
|
92
|
+
extern JS_FRIEND_API(int)
|
93
|
+
js_DateGetHours(JSContext *cx, JSObject* obj);
|
94
|
+
|
95
|
+
extern JS_FRIEND_API(int)
|
96
|
+
js_DateGetMinutes(JSContext *cx, JSObject* obj);
|
97
|
+
|
98
|
+
extern JS_FRIEND_API(int)
|
99
|
+
js_DateGetSeconds(JSContext *cx, JSObject* obj);
|
100
|
+
|
101
|
+
extern JS_FRIEND_API(void)
|
102
|
+
js_DateSetYear(JSContext *cx, JSObject *obj, int year);
|
103
|
+
|
104
|
+
extern JS_FRIEND_API(void)
|
105
|
+
js_DateSetMonth(JSContext *cx, JSObject *obj, int month);
|
106
|
+
|
107
|
+
extern JS_FRIEND_API(void)
|
108
|
+
js_DateSetDate(JSContext *cx, JSObject *obj, int date);
|
109
|
+
|
110
|
+
extern JS_FRIEND_API(void)
|
111
|
+
js_DateSetHours(JSContext *cx, JSObject *obj, int hours);
|
112
|
+
|
113
|
+
extern JS_FRIEND_API(void)
|
114
|
+
js_DateSetMinutes(JSContext *cx, JSObject *obj, int minutes);
|
115
|
+
|
116
|
+
extern JS_FRIEND_API(void)
|
117
|
+
js_DateSetSeconds(JSContext *cx, JSObject *obj, int seconds);
|
118
|
+
|
119
|
+
extern JS_FRIEND_API(jsdouble)
|
120
|
+
js_DateGetMsecSinceEpoch(JSContext *cx, JSObject *obj);
|
121
|
+
|
122
|
+
JS_END_EXTERN_C
|
123
|
+
|
124
|
+
#endif /* jsdate_h___ */
|
@@ -0,0 +1,1799 @@
|
|
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 debugging API.
|
43
|
+
*/
|
44
|
+
#include "jsstddef.h"
|
45
|
+
#include <string.h>
|
46
|
+
#include "jstypes.h"
|
47
|
+
#include "jsutil.h" /* Added by JSIFY */
|
48
|
+
#include "jsclist.h"
|
49
|
+
#include "jsapi.h"
|
50
|
+
#include "jscntxt.h"
|
51
|
+
#include "jsconfig.h"
|
52
|
+
#include "jsdbgapi.h"
|
53
|
+
#include "jsemit.h"
|
54
|
+
#include "jsfun.h"
|
55
|
+
#include "jsgc.h"
|
56
|
+
#include "jsinterp.h"
|
57
|
+
#include "jslock.h"
|
58
|
+
#include "jsobj.h"
|
59
|
+
#include "jsopcode.h"
|
60
|
+
#include "jsparse.h"
|
61
|
+
#include "jsscope.h"
|
62
|
+
#include "jsscript.h"
|
63
|
+
#include "jsstr.h"
|
64
|
+
|
65
|
+
typedef struct JSTrap {
|
66
|
+
JSCList links;
|
67
|
+
JSScript *script;
|
68
|
+
jsbytecode *pc;
|
69
|
+
JSOp op;
|
70
|
+
JSTrapHandler handler;
|
71
|
+
void *closure;
|
72
|
+
} JSTrap;
|
73
|
+
|
74
|
+
#define DBG_LOCK(rt) JS_ACQUIRE_LOCK((rt)->debuggerLock)
|
75
|
+
#define DBG_UNLOCK(rt) JS_RELEASE_LOCK((rt)->debuggerLock)
|
76
|
+
#define DBG_LOCK_EVAL(rt,expr) (DBG_LOCK(rt), (expr), DBG_UNLOCK(rt))
|
77
|
+
|
78
|
+
/*
|
79
|
+
* NB: FindTrap must be called with rt->debuggerLock acquired.
|
80
|
+
*/
|
81
|
+
static JSTrap *
|
82
|
+
FindTrap(JSRuntime *rt, JSScript *script, jsbytecode *pc)
|
83
|
+
{
|
84
|
+
JSTrap *trap;
|
85
|
+
|
86
|
+
for (trap = (JSTrap *)rt->trapList.next;
|
87
|
+
trap != (JSTrap *)&rt->trapList;
|
88
|
+
trap = (JSTrap *)trap->links.next) {
|
89
|
+
if (trap->script == script && trap->pc == pc)
|
90
|
+
return trap;
|
91
|
+
}
|
92
|
+
return NULL;
|
93
|
+
}
|
94
|
+
|
95
|
+
jsbytecode *
|
96
|
+
js_UntrapScriptCode(JSContext *cx, JSScript *script)
|
97
|
+
{
|
98
|
+
jsbytecode *code;
|
99
|
+
JSRuntime *rt;
|
100
|
+
JSTrap *trap;
|
101
|
+
|
102
|
+
code = script->code;
|
103
|
+
rt = cx->runtime;
|
104
|
+
DBG_LOCK(rt);
|
105
|
+
for (trap = (JSTrap *)rt->trapList.next;
|
106
|
+
trap != (JSTrap *)&rt->trapList;
|
107
|
+
trap = (JSTrap *)trap->links.next) {
|
108
|
+
if (trap->script == script &&
|
109
|
+
(size_t)(trap->pc - script->code) < script->length) {
|
110
|
+
if (code == script->code) {
|
111
|
+
jssrcnote *sn, *notes;
|
112
|
+
size_t nbytes;
|
113
|
+
|
114
|
+
nbytes = script->length * sizeof(jsbytecode);
|
115
|
+
notes = SCRIPT_NOTES(script);
|
116
|
+
for (sn = notes; !SN_IS_TERMINATOR(sn); sn = SN_NEXT(sn))
|
117
|
+
continue;
|
118
|
+
nbytes += (sn - notes + 1) * sizeof *sn;
|
119
|
+
|
120
|
+
code = (jsbytecode *) JS_malloc(cx, nbytes);
|
121
|
+
if (!code)
|
122
|
+
break;
|
123
|
+
memcpy(code, script->code, nbytes);
|
124
|
+
JS_CLEAR_GSN_CACHE(cx);
|
125
|
+
}
|
126
|
+
code[trap->pc - script->code] = trap->op;
|
127
|
+
}
|
128
|
+
}
|
129
|
+
DBG_UNLOCK(rt);
|
130
|
+
return code;
|
131
|
+
}
|
132
|
+
|
133
|
+
JS_PUBLIC_API(JSBool)
|
134
|
+
JS_SetTrap(JSContext *cx, JSScript *script, jsbytecode *pc,
|
135
|
+
JSTrapHandler handler, void *closure)
|
136
|
+
{
|
137
|
+
JSTrap *junk, *trap, *twin;
|
138
|
+
JSRuntime *rt;
|
139
|
+
uint32 sample;
|
140
|
+
|
141
|
+
JS_ASSERT((JSOp) *pc != JSOP_TRAP);
|
142
|
+
junk = NULL;
|
143
|
+
rt = cx->runtime;
|
144
|
+
DBG_LOCK(rt);
|
145
|
+
trap = FindTrap(rt, script, pc);
|
146
|
+
if (trap) {
|
147
|
+
JS_ASSERT(trap->script == script && trap->pc == pc);
|
148
|
+
JS_ASSERT(*pc == JSOP_TRAP);
|
149
|
+
} else {
|
150
|
+
sample = rt->debuggerMutations;
|
151
|
+
DBG_UNLOCK(rt);
|
152
|
+
trap = (JSTrap *) JS_malloc(cx, sizeof *trap);
|
153
|
+
if (!trap)
|
154
|
+
return JS_FALSE;
|
155
|
+
trap->closure = NULL;
|
156
|
+
if(!js_AddRoot(cx, &trap->closure, "trap->closure")) {
|
157
|
+
JS_free(cx, trap);
|
158
|
+
return JS_FALSE;
|
159
|
+
}
|
160
|
+
DBG_LOCK(rt);
|
161
|
+
twin = (rt->debuggerMutations != sample)
|
162
|
+
? FindTrap(rt, script, pc)
|
163
|
+
: NULL;
|
164
|
+
if (twin) {
|
165
|
+
junk = trap;
|
166
|
+
trap = twin;
|
167
|
+
} else {
|
168
|
+
JS_APPEND_LINK(&trap->links, &rt->trapList);
|
169
|
+
++rt->debuggerMutations;
|
170
|
+
trap->script = script;
|
171
|
+
trap->pc = pc;
|
172
|
+
trap->op = (JSOp)*pc;
|
173
|
+
*pc = JSOP_TRAP;
|
174
|
+
}
|
175
|
+
}
|
176
|
+
trap->handler = handler;
|
177
|
+
trap->closure = closure;
|
178
|
+
DBG_UNLOCK(rt);
|
179
|
+
if (junk) {
|
180
|
+
js_RemoveRoot(rt, &junk->closure);
|
181
|
+
JS_free(cx, junk);
|
182
|
+
}
|
183
|
+
return JS_TRUE;
|
184
|
+
}
|
185
|
+
|
186
|
+
JS_PUBLIC_API(JSOp)
|
187
|
+
JS_GetTrapOpcode(JSContext *cx, JSScript *script, jsbytecode *pc)
|
188
|
+
{
|
189
|
+
JSRuntime *rt;
|
190
|
+
JSTrap *trap;
|
191
|
+
JSOp op;
|
192
|
+
|
193
|
+
rt = cx->runtime;
|
194
|
+
DBG_LOCK(rt);
|
195
|
+
trap = FindTrap(rt, script, pc);
|
196
|
+
op = trap ? trap->op : (JSOp) *pc;
|
197
|
+
DBG_UNLOCK(rt);
|
198
|
+
return op;
|
199
|
+
}
|
200
|
+
|
201
|
+
static void
|
202
|
+
DestroyTrapAndUnlock(JSContext *cx, JSTrap *trap)
|
203
|
+
{
|
204
|
+
++cx->runtime->debuggerMutations;
|
205
|
+
JS_REMOVE_LINK(&trap->links);
|
206
|
+
*trap->pc = (jsbytecode)trap->op;
|
207
|
+
DBG_UNLOCK(cx->runtime);
|
208
|
+
|
209
|
+
js_RemoveRoot(cx->runtime, &trap->closure);
|
210
|
+
JS_free(cx, trap);
|
211
|
+
}
|
212
|
+
|
213
|
+
JS_PUBLIC_API(void)
|
214
|
+
JS_ClearTrap(JSContext *cx, JSScript *script, jsbytecode *pc,
|
215
|
+
JSTrapHandler *handlerp, void **closurep)
|
216
|
+
{
|
217
|
+
JSTrap *trap;
|
218
|
+
|
219
|
+
DBG_LOCK(cx->runtime);
|
220
|
+
trap = FindTrap(cx->runtime, script, pc);
|
221
|
+
if (handlerp)
|
222
|
+
*handlerp = trap ? trap->handler : NULL;
|
223
|
+
if (closurep)
|
224
|
+
*closurep = trap ? trap->closure : NULL;
|
225
|
+
if (trap)
|
226
|
+
DestroyTrapAndUnlock(cx, trap);
|
227
|
+
else
|
228
|
+
DBG_UNLOCK(cx->runtime);
|
229
|
+
}
|
230
|
+
|
231
|
+
JS_PUBLIC_API(void)
|
232
|
+
JS_ClearScriptTraps(JSContext *cx, JSScript *script)
|
233
|
+
{
|
234
|
+
JSRuntime *rt;
|
235
|
+
JSTrap *trap, *next;
|
236
|
+
uint32 sample;
|
237
|
+
|
238
|
+
rt = cx->runtime;
|
239
|
+
DBG_LOCK(rt);
|
240
|
+
for (trap = (JSTrap *)rt->trapList.next;
|
241
|
+
trap != (JSTrap *)&rt->trapList;
|
242
|
+
trap = next) {
|
243
|
+
next = (JSTrap *)trap->links.next;
|
244
|
+
if (trap->script == script) {
|
245
|
+
sample = rt->debuggerMutations;
|
246
|
+
DestroyTrapAndUnlock(cx, trap);
|
247
|
+
DBG_LOCK(rt);
|
248
|
+
if (rt->debuggerMutations != sample + 1)
|
249
|
+
next = (JSTrap *)rt->trapList.next;
|
250
|
+
}
|
251
|
+
}
|
252
|
+
DBG_UNLOCK(rt);
|
253
|
+
}
|
254
|
+
|
255
|
+
JS_PUBLIC_API(void)
|
256
|
+
JS_ClearAllTraps(JSContext *cx)
|
257
|
+
{
|
258
|
+
JSRuntime *rt;
|
259
|
+
JSTrap *trap, *next;
|
260
|
+
uint32 sample;
|
261
|
+
|
262
|
+
rt = cx->runtime;
|
263
|
+
DBG_LOCK(rt);
|
264
|
+
for (trap = (JSTrap *)rt->trapList.next;
|
265
|
+
trap != (JSTrap *)&rt->trapList;
|
266
|
+
trap = next) {
|
267
|
+
next = (JSTrap *)trap->links.next;
|
268
|
+
sample = rt->debuggerMutations;
|
269
|
+
DestroyTrapAndUnlock(cx, trap);
|
270
|
+
DBG_LOCK(rt);
|
271
|
+
if (rt->debuggerMutations != sample + 1)
|
272
|
+
next = (JSTrap *)rt->trapList.next;
|
273
|
+
}
|
274
|
+
DBG_UNLOCK(rt);
|
275
|
+
}
|
276
|
+
|
277
|
+
JS_PUBLIC_API(JSTrapStatus)
|
278
|
+
JS_HandleTrap(JSContext *cx, JSScript *script, jsbytecode *pc, jsval *rval)
|
279
|
+
{
|
280
|
+
JSTrap *trap;
|
281
|
+
jsint op;
|
282
|
+
JSTrapStatus status;
|
283
|
+
|
284
|
+
DBG_LOCK(cx->runtime);
|
285
|
+
trap = FindTrap(cx->runtime, script, pc);
|
286
|
+
JS_ASSERT(!trap || trap->handler);
|
287
|
+
if (!trap) {
|
288
|
+
op = (JSOp) *pc;
|
289
|
+
DBG_UNLOCK(cx->runtime);
|
290
|
+
|
291
|
+
/* Defend against "pc for wrong script" API usage error. */
|
292
|
+
JS_ASSERT(op != JSOP_TRAP);
|
293
|
+
|
294
|
+
#ifdef JS_THREADSAFE
|
295
|
+
/* If the API was abused, we must fail for want of the real op. */
|
296
|
+
if (op == JSOP_TRAP)
|
297
|
+
return JSTRAP_ERROR;
|
298
|
+
|
299
|
+
/* Assume a race with a debugger thread and try to carry on. */
|
300
|
+
*rval = INT_TO_JSVAL(op);
|
301
|
+
return JSTRAP_CONTINUE;
|
302
|
+
#else
|
303
|
+
/* Always fail if single-threaded (must be an API usage error). */
|
304
|
+
return JSTRAP_ERROR;
|
305
|
+
#endif
|
306
|
+
}
|
307
|
+
DBG_UNLOCK(cx->runtime);
|
308
|
+
|
309
|
+
/*
|
310
|
+
* It's important that we not use 'trap->' after calling the callback --
|
311
|
+
* the callback might remove the trap!
|
312
|
+
*/
|
313
|
+
op = (jsint)trap->op;
|
314
|
+
status = trap->handler(cx, script, pc, rval, trap->closure);
|
315
|
+
if (status == JSTRAP_CONTINUE) {
|
316
|
+
/* By convention, return the true op to the interpreter in rval. */
|
317
|
+
*rval = INT_TO_JSVAL(op);
|
318
|
+
}
|
319
|
+
return status;
|
320
|
+
}
|
321
|
+
|
322
|
+
JS_PUBLIC_API(JSBool)
|
323
|
+
JS_SetInterrupt(JSRuntime *rt, JSTrapHandler handler, void *closure)
|
324
|
+
{
|
325
|
+
rt->globalDebugHooks.interruptHandler = handler;
|
326
|
+
rt->globalDebugHooks.interruptHandlerData = closure;
|
327
|
+
return JS_TRUE;
|
328
|
+
}
|
329
|
+
|
330
|
+
JS_PUBLIC_API(JSBool)
|
331
|
+
JS_ClearInterrupt(JSRuntime *rt, JSTrapHandler *handlerp, void **closurep)
|
332
|
+
{
|
333
|
+
if (handlerp)
|
334
|
+
*handlerp = (JSTrapHandler)rt->globalDebugHooks.interruptHandler;
|
335
|
+
if (closurep)
|
336
|
+
*closurep = rt->globalDebugHooks.interruptHandlerData;
|
337
|
+
rt->globalDebugHooks.interruptHandler = 0;
|
338
|
+
rt->globalDebugHooks.interruptHandlerData = 0;
|
339
|
+
return JS_TRUE;
|
340
|
+
}
|
341
|
+
|
342
|
+
/************************************************************************/
|
343
|
+
|
344
|
+
typedef struct JSWatchPoint {
|
345
|
+
JSCList links;
|
346
|
+
JSObject *object; /* weak link, see js_FinalizeObject */
|
347
|
+
JSScopeProperty *sprop;
|
348
|
+
JSPropertyOp setter;
|
349
|
+
JSWatchPointHandler handler;
|
350
|
+
void *closure;
|
351
|
+
uintN flags;
|
352
|
+
} JSWatchPoint;
|
353
|
+
|
354
|
+
#define JSWP_LIVE 0x1 /* live because set and not cleared */
|
355
|
+
#define JSWP_HELD 0x2 /* held while running handler/setter */
|
356
|
+
|
357
|
+
/*
|
358
|
+
* NB: DropWatchPointAndUnlock releases cx->runtime->debuggerLock in all cases.
|
359
|
+
*/
|
360
|
+
static JSBool
|
361
|
+
DropWatchPointAndUnlock(JSContext *cx, JSWatchPoint *wp, uintN flag)
|
362
|
+
{
|
363
|
+
JSBool ok, found;
|
364
|
+
JSScopeProperty *sprop;
|
365
|
+
JSScope *scope;
|
366
|
+
JSPropertyOp setter;
|
367
|
+
|
368
|
+
ok = JS_TRUE;
|
369
|
+
wp->flags &= ~flag;
|
370
|
+
if (wp->flags != 0) {
|
371
|
+
DBG_UNLOCK(cx->runtime);
|
372
|
+
return ok;
|
373
|
+
}
|
374
|
+
|
375
|
+
/*
|
376
|
+
* Remove wp from the list, then if there are no other watchpoints for
|
377
|
+
* wp->sprop in any scope, restore wp->sprop->setter from wp.
|
378
|
+
*/
|
379
|
+
++cx->runtime->debuggerMutations;
|
380
|
+
JS_REMOVE_LINK(&wp->links);
|
381
|
+
sprop = wp->sprop;
|
382
|
+
|
383
|
+
/*
|
384
|
+
* Passing null for the scope parameter tells js_GetWatchedSetter to find
|
385
|
+
* any watch point for sprop, and not to lock or unlock rt->debuggerLock.
|
386
|
+
* If js_ChangeNativePropertyAttrs fails, propagate failure after removing
|
387
|
+
* wp->closure's root and freeing wp.
|
388
|
+
*/
|
389
|
+
setter = js_GetWatchedSetter(cx->runtime, NULL, sprop);
|
390
|
+
DBG_UNLOCK(cx->runtime);
|
391
|
+
if (!setter) {
|
392
|
+
JS_LOCK_OBJ(cx, wp->object);
|
393
|
+
scope = OBJ_SCOPE(wp->object);
|
394
|
+
found = (scope->object == wp->object &&
|
395
|
+
SCOPE_GET_PROPERTY(scope, sprop->id));
|
396
|
+
JS_UNLOCK_SCOPE(cx, scope);
|
397
|
+
|
398
|
+
/*
|
399
|
+
* If the property wasn't found on wp->object or didn't exist, then
|
400
|
+
* someone else has dealt with this sprop, and we don't need to change
|
401
|
+
* the property attributes.
|
402
|
+
*/
|
403
|
+
if (found) {
|
404
|
+
sprop = js_ChangeScopePropertyAttrs(cx, scope, sprop,
|
405
|
+
0, sprop->attrs,
|
406
|
+
sprop->getter,
|
407
|
+
wp->setter);
|
408
|
+
if (!sprop)
|
409
|
+
ok = JS_FALSE;
|
410
|
+
}
|
411
|
+
}
|
412
|
+
|
413
|
+
JS_free(cx, wp);
|
414
|
+
return ok;
|
415
|
+
}
|
416
|
+
|
417
|
+
/*
|
418
|
+
* NB: js_TraceWatchPoints does not acquire cx->runtime->debuggerLock, since
|
419
|
+
* the debugger should never be racing with the GC (i.e., the debugger must
|
420
|
+
* respect the request model).
|
421
|
+
*/
|
422
|
+
void
|
423
|
+
js_TraceWatchPoints(JSTracer *trc, JSObject *obj)
|
424
|
+
{
|
425
|
+
JSRuntime *rt;
|
426
|
+
JSWatchPoint *wp;
|
427
|
+
|
428
|
+
rt = trc->context->runtime;
|
429
|
+
|
430
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
431
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
432
|
+
wp = (JSWatchPoint *)wp->links.next) {
|
433
|
+
if (wp->object == obj) {
|
434
|
+
TRACE_SCOPE_PROPERTY(trc, wp->sprop);
|
435
|
+
if ((wp->sprop->attrs & JSPROP_SETTER) && wp->setter) {
|
436
|
+
JS_CALL_OBJECT_TRACER(trc, (JSObject *)wp->setter,
|
437
|
+
"wp->setter");
|
438
|
+
}
|
439
|
+
JS_SET_TRACING_NAME(trc, "wp->closure");
|
440
|
+
js_CallValueTracerIfGCThing(trc, (jsval) wp->closure);
|
441
|
+
}
|
442
|
+
}
|
443
|
+
}
|
444
|
+
|
445
|
+
void
|
446
|
+
js_SweepWatchPoints(JSContext *cx)
|
447
|
+
{
|
448
|
+
JSRuntime *rt;
|
449
|
+
JSWatchPoint *wp, *next;
|
450
|
+
uint32 sample;
|
451
|
+
|
452
|
+
rt = cx->runtime;
|
453
|
+
DBG_LOCK(rt);
|
454
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
455
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
456
|
+
wp = next) {
|
457
|
+
next = (JSWatchPoint *)wp->links.next;
|
458
|
+
if (js_IsAboutToBeFinalized(cx, wp->object)) {
|
459
|
+
sample = rt->debuggerMutations;
|
460
|
+
|
461
|
+
/* Ignore failures. */
|
462
|
+
DropWatchPointAndUnlock(cx, wp, JSWP_LIVE);
|
463
|
+
DBG_LOCK(rt);
|
464
|
+
if (rt->debuggerMutations != sample + 1)
|
465
|
+
next = (JSWatchPoint *)rt->watchPointList.next;
|
466
|
+
}
|
467
|
+
}
|
468
|
+
DBG_UNLOCK(rt);
|
469
|
+
}
|
470
|
+
|
471
|
+
|
472
|
+
|
473
|
+
/*
|
474
|
+
* NB: FindWatchPoint must be called with rt->debuggerLock acquired.
|
475
|
+
*/
|
476
|
+
static JSWatchPoint *
|
477
|
+
FindWatchPoint(JSRuntime *rt, JSScope *scope, jsid id)
|
478
|
+
{
|
479
|
+
JSWatchPoint *wp;
|
480
|
+
|
481
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
482
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
483
|
+
wp = (JSWatchPoint *)wp->links.next) {
|
484
|
+
if (wp->object == scope->object && wp->sprop->id == id)
|
485
|
+
return wp;
|
486
|
+
}
|
487
|
+
return NULL;
|
488
|
+
}
|
489
|
+
|
490
|
+
JSScopeProperty *
|
491
|
+
js_FindWatchPoint(JSRuntime *rt, JSScope *scope, jsid id)
|
492
|
+
{
|
493
|
+
JSWatchPoint *wp;
|
494
|
+
JSScopeProperty *sprop;
|
495
|
+
|
496
|
+
DBG_LOCK(rt);
|
497
|
+
wp = FindWatchPoint(rt, scope, id);
|
498
|
+
sprop = wp ? wp->sprop : NULL;
|
499
|
+
DBG_UNLOCK(rt);
|
500
|
+
return sprop;
|
501
|
+
}
|
502
|
+
|
503
|
+
/*
|
504
|
+
* Secret handshake with DropWatchPointAndUnlock: if (!scope), we know our
|
505
|
+
* caller has acquired rt->debuggerLock, so we don't have to.
|
506
|
+
*/
|
507
|
+
JSPropertyOp
|
508
|
+
js_GetWatchedSetter(JSRuntime *rt, JSScope *scope,
|
509
|
+
const JSScopeProperty *sprop)
|
510
|
+
{
|
511
|
+
JSPropertyOp setter;
|
512
|
+
JSWatchPoint *wp;
|
513
|
+
|
514
|
+
setter = NULL;
|
515
|
+
if (scope)
|
516
|
+
DBG_LOCK(rt);
|
517
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
518
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
519
|
+
wp = (JSWatchPoint *)wp->links.next) {
|
520
|
+
if ((!scope || wp->object == scope->object) && wp->sprop == sprop) {
|
521
|
+
setter = wp->setter;
|
522
|
+
break;
|
523
|
+
}
|
524
|
+
}
|
525
|
+
if (scope)
|
526
|
+
DBG_UNLOCK(rt);
|
527
|
+
return setter;
|
528
|
+
}
|
529
|
+
|
530
|
+
JSBool JS_DLL_CALLBACK
|
531
|
+
js_watch_set(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
532
|
+
{
|
533
|
+
JSRuntime *rt;
|
534
|
+
JSWatchPoint *wp;
|
535
|
+
JSScopeProperty *sprop;
|
536
|
+
jsval propid, userid;
|
537
|
+
JSScope *scope;
|
538
|
+
JSBool ok;
|
539
|
+
|
540
|
+
rt = cx->runtime;
|
541
|
+
DBG_LOCK(rt);
|
542
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
543
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
544
|
+
wp = (JSWatchPoint *)wp->links.next) {
|
545
|
+
sprop = wp->sprop;
|
546
|
+
if (wp->object == obj && SPROP_USERID(sprop) == id &&
|
547
|
+
!(wp->flags & JSWP_HELD)) {
|
548
|
+
wp->flags |= JSWP_HELD;
|
549
|
+
DBG_UNLOCK(rt);
|
550
|
+
|
551
|
+
JS_LOCK_OBJ(cx, obj);
|
552
|
+
propid = ID_TO_VALUE(sprop->id);
|
553
|
+
userid = (sprop->flags & SPROP_HAS_SHORTID)
|
554
|
+
? INT_TO_JSVAL(sprop->shortid)
|
555
|
+
: propid;
|
556
|
+
scope = OBJ_SCOPE(obj);
|
557
|
+
JS_UNLOCK_OBJ(cx, obj);
|
558
|
+
|
559
|
+
/* NB: wp is held, so we can safely dereference it still. */
|
560
|
+
ok = wp->handler(cx, obj, propid,
|
561
|
+
SPROP_HAS_VALID_SLOT(sprop, scope)
|
562
|
+
? OBJ_GET_SLOT(cx, obj, sprop->slot)
|
563
|
+
: JSVAL_VOID,
|
564
|
+
vp, wp->closure);
|
565
|
+
if (ok) {
|
566
|
+
/*
|
567
|
+
* Create a pseudo-frame for the setter invocation so that any
|
568
|
+
* stack-walking security code under the setter will correctly
|
569
|
+
* identify the guilty party. So that the watcher appears to
|
570
|
+
* be active to obj_eval and other such code, point frame.pc
|
571
|
+
* at the JSOP_STOP at the end of the script.
|
572
|
+
*
|
573
|
+
* The pseudo-frame is not created for fast natives as they
|
574
|
+
* are treated as interpreter frame extensions and always
|
575
|
+
* trusted.
|
576
|
+
*/
|
577
|
+
JSObject *closure;
|
578
|
+
JSClass *clasp;
|
579
|
+
JSFunction *fun;
|
580
|
+
JSScript *script;
|
581
|
+
JSBool injectFrame;
|
582
|
+
uintN nslots;
|
583
|
+
jsval smallv[5];
|
584
|
+
jsval *argv;
|
585
|
+
JSStackFrame frame;
|
586
|
+
JSFrameRegs regs;
|
587
|
+
|
588
|
+
closure = (JSObject *) wp->closure;
|
589
|
+
clasp = OBJ_GET_CLASS(cx, closure);
|
590
|
+
if (clasp == &js_FunctionClass) {
|
591
|
+
fun = GET_FUNCTION_PRIVATE(cx, closure);
|
592
|
+
script = FUN_SCRIPT(fun);
|
593
|
+
} else if (clasp == &js_ScriptClass) {
|
594
|
+
fun = NULL;
|
595
|
+
script = (JSScript *) JS_GetPrivate(cx, closure);
|
596
|
+
} else {
|
597
|
+
fun = NULL;
|
598
|
+
script = NULL;
|
599
|
+
}
|
600
|
+
|
601
|
+
nslots = 2;
|
602
|
+
injectFrame = JS_TRUE;
|
603
|
+
if (fun) {
|
604
|
+
nslots += FUN_MINARGS(fun);
|
605
|
+
if (!FUN_INTERPRETED(fun)) {
|
606
|
+
nslots += fun->u.n.extra;
|
607
|
+
injectFrame = !(fun->flags & JSFUN_FAST_NATIVE);
|
608
|
+
}
|
609
|
+
}
|
610
|
+
|
611
|
+
if (injectFrame) {
|
612
|
+
if (nslots <= JS_ARRAY_LENGTH(smallv)) {
|
613
|
+
argv = smallv;
|
614
|
+
} else {
|
615
|
+
argv = (jsval *) JS_malloc(cx, nslots * sizeof(jsval));
|
616
|
+
if (!argv) {
|
617
|
+
DBG_LOCK(rt);
|
618
|
+
DropWatchPointAndUnlock(cx, wp, JSWP_HELD);
|
619
|
+
return JS_FALSE;
|
620
|
+
}
|
621
|
+
}
|
622
|
+
|
623
|
+
argv[0] = OBJECT_TO_JSVAL(closure);
|
624
|
+
argv[1] = JSVAL_NULL;
|
625
|
+
memset(argv + 2, 0, (nslots - 2) * sizeof(jsval));
|
626
|
+
|
627
|
+
memset(&frame, 0, sizeof(frame));
|
628
|
+
frame.script = script;
|
629
|
+
frame.regs = NULL;
|
630
|
+
if (script) {
|
631
|
+
JS_ASSERT(script->length >= JSOP_STOP_LENGTH);
|
632
|
+
regs.pc = script->code + script->length
|
633
|
+
- JSOP_STOP_LENGTH;
|
634
|
+
regs.sp = NULL;
|
635
|
+
frame.regs = ®s;
|
636
|
+
}
|
637
|
+
frame.callee = closure;
|
638
|
+
frame.fun = fun;
|
639
|
+
frame.argv = argv + 2;
|
640
|
+
frame.down = cx->fp;
|
641
|
+
frame.scopeChain = OBJ_GET_PARENT(cx, closure);
|
642
|
+
|
643
|
+
cx->fp = &frame;
|
644
|
+
}
|
645
|
+
#ifdef __GNUC__
|
646
|
+
else
|
647
|
+
argv = NULL; /* suppress bogus gcc warnings */
|
648
|
+
#endif
|
649
|
+
ok = !wp->setter ||
|
650
|
+
((sprop->attrs & JSPROP_SETTER)
|
651
|
+
? js_InternalCall(cx, obj, OBJECT_TO_JSVAL(wp->setter),
|
652
|
+
1, vp, vp)
|
653
|
+
: wp->setter(cx, OBJ_THIS_OBJECT(cx, obj), userid, vp));
|
654
|
+
if (injectFrame) {
|
655
|
+
/* Evil code can cause us to have an arguments object. */
|
656
|
+
if (frame.callobj)
|
657
|
+
ok &= js_PutCallObject(cx, &frame);
|
658
|
+
if (frame.argsobj)
|
659
|
+
ok &= js_PutArgsObject(cx, &frame);
|
660
|
+
|
661
|
+
cx->fp = frame.down;
|
662
|
+
if (argv != smallv)
|
663
|
+
JS_free(cx, argv);
|
664
|
+
}
|
665
|
+
}
|
666
|
+
DBG_LOCK(rt);
|
667
|
+
return DropWatchPointAndUnlock(cx, wp, JSWP_HELD) && ok;
|
668
|
+
}
|
669
|
+
}
|
670
|
+
DBG_UNLOCK(rt);
|
671
|
+
return JS_TRUE;
|
672
|
+
}
|
673
|
+
|
674
|
+
JSBool JS_DLL_CALLBACK
|
675
|
+
js_watch_set_wrapper(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
|
676
|
+
jsval *rval)
|
677
|
+
{
|
678
|
+
JSObject *funobj;
|
679
|
+
JSFunction *wrapper;
|
680
|
+
jsval userid;
|
681
|
+
|
682
|
+
funobj = JSVAL_TO_OBJECT(argv[-2]);
|
683
|
+
JS_ASSERT(OBJ_GET_CLASS(cx, funobj) == &js_FunctionClass);
|
684
|
+
wrapper = GET_FUNCTION_PRIVATE(cx, funobj);
|
685
|
+
userid = ATOM_KEY(wrapper->atom);
|
686
|
+
*rval = argv[0];
|
687
|
+
return js_watch_set(cx, obj, userid, rval);
|
688
|
+
}
|
689
|
+
|
690
|
+
JSPropertyOp
|
691
|
+
js_WrapWatchedSetter(JSContext *cx, jsid id, uintN attrs, JSPropertyOp setter)
|
692
|
+
{
|
693
|
+
JSAtom *atom;
|
694
|
+
JSFunction *wrapper;
|
695
|
+
|
696
|
+
if (!(attrs & JSPROP_SETTER))
|
697
|
+
return &js_watch_set; /* & to silence schoolmarmish MSVC */
|
698
|
+
|
699
|
+
if (JSID_IS_ATOM(id)) {
|
700
|
+
atom = JSID_TO_ATOM(id);
|
701
|
+
} else if (JSID_IS_INT(id)) {
|
702
|
+
if (!js_ValueToStringId(cx, INT_JSID_TO_JSVAL(id), &id))
|
703
|
+
return NULL;
|
704
|
+
atom = JSID_TO_ATOM(id);
|
705
|
+
} else {
|
706
|
+
atom = NULL;
|
707
|
+
}
|
708
|
+
wrapper = js_NewFunction(cx, NULL, js_watch_set_wrapper, 1, 0,
|
709
|
+
OBJ_GET_PARENT(cx, (JSObject *)setter),
|
710
|
+
atom);
|
711
|
+
if (!wrapper)
|
712
|
+
return NULL;
|
713
|
+
return (JSPropertyOp) FUN_OBJECT(wrapper);
|
714
|
+
}
|
715
|
+
|
716
|
+
JS_PUBLIC_API(JSBool)
|
717
|
+
JS_SetWatchPoint(JSContext *cx, JSObject *obj, jsval idval,
|
718
|
+
JSWatchPointHandler handler, void *closure)
|
719
|
+
{
|
720
|
+
jsid propid;
|
721
|
+
JSObject *pobj;
|
722
|
+
JSProperty *prop;
|
723
|
+
JSScopeProperty *sprop;
|
724
|
+
JSRuntime *rt;
|
725
|
+
JSBool ok;
|
726
|
+
JSWatchPoint *wp;
|
727
|
+
JSPropertyOp watcher;
|
728
|
+
|
729
|
+
if (!OBJ_IS_NATIVE(obj)) {
|
730
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_CANT_WATCH,
|
731
|
+
OBJ_GET_CLASS(cx, obj)->name);
|
732
|
+
return JS_FALSE;
|
733
|
+
}
|
734
|
+
|
735
|
+
if (JSVAL_IS_INT(idval))
|
736
|
+
propid = INT_JSVAL_TO_JSID(idval);
|
737
|
+
else if (!js_ValueToStringId(cx, idval, &propid))
|
738
|
+
return JS_FALSE;
|
739
|
+
|
740
|
+
if (!js_LookupProperty(cx, obj, propid, &pobj, &prop))
|
741
|
+
return JS_FALSE;
|
742
|
+
sprop = (JSScopeProperty *) prop;
|
743
|
+
rt = cx->runtime;
|
744
|
+
if (!sprop) {
|
745
|
+
/* Check for a deleted symbol watchpoint, which holds its property. */
|
746
|
+
sprop = js_FindWatchPoint(rt, OBJ_SCOPE(obj), propid);
|
747
|
+
if (!sprop) {
|
748
|
+
/* Make a new property in obj so we can watch for the first set. */
|
749
|
+
if (!js_DefineProperty(cx, obj, propid, JSVAL_VOID,
|
750
|
+
NULL, NULL, JSPROP_ENUMERATE,
|
751
|
+
&prop)) {
|
752
|
+
return JS_FALSE;
|
753
|
+
}
|
754
|
+
sprop = (JSScopeProperty *) prop;
|
755
|
+
}
|
756
|
+
} else if (pobj != obj) {
|
757
|
+
/* Clone the prototype property so we can watch the right object. */
|
758
|
+
jsval value;
|
759
|
+
JSPropertyOp getter, setter;
|
760
|
+
uintN attrs, flags;
|
761
|
+
intN shortid;
|
762
|
+
|
763
|
+
if (OBJ_IS_NATIVE(pobj)) {
|
764
|
+
value = SPROP_HAS_VALID_SLOT(sprop, OBJ_SCOPE(pobj))
|
765
|
+
? LOCKED_OBJ_GET_SLOT(pobj, sprop->slot)
|
766
|
+
: JSVAL_VOID;
|
767
|
+
getter = sprop->getter;
|
768
|
+
setter = sprop->setter;
|
769
|
+
attrs = sprop->attrs;
|
770
|
+
flags = sprop->flags;
|
771
|
+
shortid = sprop->shortid;
|
772
|
+
} else {
|
773
|
+
if (!OBJ_GET_PROPERTY(cx, pobj, propid, &value) ||
|
774
|
+
!OBJ_GET_ATTRIBUTES(cx, pobj, propid, prop, &attrs)) {
|
775
|
+
OBJ_DROP_PROPERTY(cx, pobj, prop);
|
776
|
+
return JS_FALSE;
|
777
|
+
}
|
778
|
+
getter = setter = NULL;
|
779
|
+
flags = 0;
|
780
|
+
shortid = 0;
|
781
|
+
}
|
782
|
+
OBJ_DROP_PROPERTY(cx, pobj, prop);
|
783
|
+
|
784
|
+
/* Recall that obj is native, whether or not pobj is native. */
|
785
|
+
if (!js_DefineNativeProperty(cx, obj, propid, value, getter, setter,
|
786
|
+
attrs, flags, shortid, &prop)) {
|
787
|
+
return JS_FALSE;
|
788
|
+
}
|
789
|
+
sprop = (JSScopeProperty *) prop;
|
790
|
+
}
|
791
|
+
|
792
|
+
/*
|
793
|
+
* At this point, prop/sprop exists in obj, obj is locked, and we must
|
794
|
+
* OBJ_DROP_PROPERTY(cx, obj, prop) before returning.
|
795
|
+
*/
|
796
|
+
ok = JS_TRUE;
|
797
|
+
DBG_LOCK(rt);
|
798
|
+
wp = FindWatchPoint(rt, OBJ_SCOPE(obj), propid);
|
799
|
+
if (!wp) {
|
800
|
+
DBG_UNLOCK(rt);
|
801
|
+
watcher = js_WrapWatchedSetter(cx, propid, sprop->attrs, sprop->setter);
|
802
|
+
if (!watcher) {
|
803
|
+
ok = JS_FALSE;
|
804
|
+
goto out;
|
805
|
+
}
|
806
|
+
|
807
|
+
wp = (JSWatchPoint *) JS_malloc(cx, sizeof *wp);
|
808
|
+
if (!wp) {
|
809
|
+
ok = JS_FALSE;
|
810
|
+
goto out;
|
811
|
+
}
|
812
|
+
wp->handler = NULL;
|
813
|
+
wp->closure = NULL;
|
814
|
+
wp->object = obj;
|
815
|
+
JS_ASSERT(sprop->setter != js_watch_set || pobj != obj);
|
816
|
+
wp->setter = sprop->setter;
|
817
|
+
wp->flags = JSWP_LIVE;
|
818
|
+
|
819
|
+
/* XXXbe nest in obj lock here */
|
820
|
+
sprop = js_ChangeNativePropertyAttrs(cx, obj, sprop, 0, sprop->attrs,
|
821
|
+
sprop->getter, watcher);
|
822
|
+
if (!sprop) {
|
823
|
+
/* Self-link so DropWatchPointAndUnlock can JS_REMOVE_LINK it. */
|
824
|
+
JS_INIT_CLIST(&wp->links);
|
825
|
+
DBG_LOCK(rt);
|
826
|
+
DropWatchPointAndUnlock(cx, wp, JSWP_LIVE);
|
827
|
+
ok = JS_FALSE;
|
828
|
+
goto out;
|
829
|
+
}
|
830
|
+
wp->sprop = sprop;
|
831
|
+
|
832
|
+
/*
|
833
|
+
* Now that wp is fully initialized, append it to rt's wp list.
|
834
|
+
* Because obj is locked we know that no other thread could have added
|
835
|
+
* a watchpoint for (obj, propid).
|
836
|
+
*/
|
837
|
+
DBG_LOCK(rt);
|
838
|
+
JS_ASSERT(!FindWatchPoint(rt, OBJ_SCOPE(obj), propid));
|
839
|
+
JS_APPEND_LINK(&wp->links, &rt->watchPointList);
|
840
|
+
++rt->debuggerMutations;
|
841
|
+
}
|
842
|
+
wp->handler = handler;
|
843
|
+
wp->closure = closure;
|
844
|
+
DBG_UNLOCK(rt);
|
845
|
+
|
846
|
+
out:
|
847
|
+
OBJ_DROP_PROPERTY(cx, obj, prop);
|
848
|
+
return ok;
|
849
|
+
}
|
850
|
+
|
851
|
+
JS_PUBLIC_API(JSBool)
|
852
|
+
JS_ClearWatchPoint(JSContext *cx, JSObject *obj, jsval id,
|
853
|
+
JSWatchPointHandler *handlerp, void **closurep)
|
854
|
+
{
|
855
|
+
JSRuntime *rt;
|
856
|
+
JSWatchPoint *wp;
|
857
|
+
|
858
|
+
rt = cx->runtime;
|
859
|
+
DBG_LOCK(rt);
|
860
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
861
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
862
|
+
wp = (JSWatchPoint *)wp->links.next) {
|
863
|
+
if (wp->object == obj && SPROP_USERID(wp->sprop) == id) {
|
864
|
+
if (handlerp)
|
865
|
+
*handlerp = wp->handler;
|
866
|
+
if (closurep)
|
867
|
+
*closurep = wp->closure;
|
868
|
+
return DropWatchPointAndUnlock(cx, wp, JSWP_LIVE);
|
869
|
+
}
|
870
|
+
}
|
871
|
+
DBG_UNLOCK(rt);
|
872
|
+
if (handlerp)
|
873
|
+
*handlerp = NULL;
|
874
|
+
if (closurep)
|
875
|
+
*closurep = NULL;
|
876
|
+
return JS_TRUE;
|
877
|
+
}
|
878
|
+
|
879
|
+
JS_PUBLIC_API(JSBool)
|
880
|
+
JS_ClearWatchPointsForObject(JSContext *cx, JSObject *obj)
|
881
|
+
{
|
882
|
+
JSRuntime *rt;
|
883
|
+
JSWatchPoint *wp, *next;
|
884
|
+
uint32 sample;
|
885
|
+
|
886
|
+
rt = cx->runtime;
|
887
|
+
DBG_LOCK(rt);
|
888
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
889
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
890
|
+
wp = next) {
|
891
|
+
next = (JSWatchPoint *)wp->links.next;
|
892
|
+
if (wp->object == obj) {
|
893
|
+
sample = rt->debuggerMutations;
|
894
|
+
if (!DropWatchPointAndUnlock(cx, wp, JSWP_LIVE))
|
895
|
+
return JS_FALSE;
|
896
|
+
DBG_LOCK(rt);
|
897
|
+
if (rt->debuggerMutations != sample + 1)
|
898
|
+
next = (JSWatchPoint *)rt->watchPointList.next;
|
899
|
+
}
|
900
|
+
}
|
901
|
+
DBG_UNLOCK(rt);
|
902
|
+
return JS_TRUE;
|
903
|
+
}
|
904
|
+
|
905
|
+
JS_PUBLIC_API(JSBool)
|
906
|
+
JS_ClearAllWatchPoints(JSContext *cx)
|
907
|
+
{
|
908
|
+
JSRuntime *rt;
|
909
|
+
JSWatchPoint *wp, *next;
|
910
|
+
uint32 sample;
|
911
|
+
|
912
|
+
rt = cx->runtime;
|
913
|
+
DBG_LOCK(rt);
|
914
|
+
for (wp = (JSWatchPoint *)rt->watchPointList.next;
|
915
|
+
wp != (JSWatchPoint *)&rt->watchPointList;
|
916
|
+
wp = next) {
|
917
|
+
next = (JSWatchPoint *)wp->links.next;
|
918
|
+
sample = rt->debuggerMutations;
|
919
|
+
if (!DropWatchPointAndUnlock(cx, wp, JSWP_LIVE))
|
920
|
+
return JS_FALSE;
|
921
|
+
DBG_LOCK(rt);
|
922
|
+
if (rt->debuggerMutations != sample + 1)
|
923
|
+
next = (JSWatchPoint *)rt->watchPointList.next;
|
924
|
+
}
|
925
|
+
DBG_UNLOCK(rt);
|
926
|
+
return JS_TRUE;
|
927
|
+
}
|
928
|
+
|
929
|
+
/************************************************************************/
|
930
|
+
|
931
|
+
JS_PUBLIC_API(uintN)
|
932
|
+
JS_PCToLineNumber(JSContext *cx, JSScript *script, jsbytecode *pc)
|
933
|
+
{
|
934
|
+
return js_PCToLineNumber(cx, script, pc);
|
935
|
+
}
|
936
|
+
|
937
|
+
JS_PUBLIC_API(jsbytecode *)
|
938
|
+
JS_LineNumberToPC(JSContext *cx, JSScript *script, uintN lineno)
|
939
|
+
{
|
940
|
+
return js_LineNumberToPC(script, lineno);
|
941
|
+
}
|
942
|
+
|
943
|
+
JS_PUBLIC_API(JSScript *)
|
944
|
+
JS_GetFunctionScript(JSContext *cx, JSFunction *fun)
|
945
|
+
{
|
946
|
+
return FUN_SCRIPT(fun);
|
947
|
+
}
|
948
|
+
|
949
|
+
JS_PUBLIC_API(JSNative)
|
950
|
+
JS_GetFunctionNative(JSContext *cx, JSFunction *fun)
|
951
|
+
{
|
952
|
+
return FUN_NATIVE(fun);
|
953
|
+
}
|
954
|
+
|
955
|
+
JS_PUBLIC_API(JSFastNative)
|
956
|
+
JS_GetFunctionFastNative(JSContext *cx, JSFunction *fun)
|
957
|
+
{
|
958
|
+
return FUN_FAST_NATIVE(fun);
|
959
|
+
}
|
960
|
+
|
961
|
+
JS_PUBLIC_API(JSPrincipals *)
|
962
|
+
JS_GetScriptPrincipals(JSContext *cx, JSScript *script)
|
963
|
+
{
|
964
|
+
return script->principals;
|
965
|
+
}
|
966
|
+
|
967
|
+
/************************************************************************/
|
968
|
+
|
969
|
+
/*
|
970
|
+
* Stack Frame Iterator
|
971
|
+
*/
|
972
|
+
JS_PUBLIC_API(JSStackFrame *)
|
973
|
+
JS_FrameIterator(JSContext *cx, JSStackFrame **iteratorp)
|
974
|
+
{
|
975
|
+
*iteratorp = (*iteratorp == NULL) ? cx->fp : (*iteratorp)->down;
|
976
|
+
return *iteratorp;
|
977
|
+
}
|
978
|
+
|
979
|
+
JS_PUBLIC_API(JSScript *)
|
980
|
+
JS_GetFrameScript(JSContext *cx, JSStackFrame *fp)
|
981
|
+
{
|
982
|
+
return fp->script;
|
983
|
+
}
|
984
|
+
|
985
|
+
JS_PUBLIC_API(jsbytecode *)
|
986
|
+
JS_GetFramePC(JSContext *cx, JSStackFrame *fp)
|
987
|
+
{
|
988
|
+
return fp->regs ? fp->regs->pc : NULL;
|
989
|
+
}
|
990
|
+
|
991
|
+
JS_PUBLIC_API(JSStackFrame *)
|
992
|
+
JS_GetScriptedCaller(JSContext *cx, JSStackFrame *fp)
|
993
|
+
{
|
994
|
+
if (!fp)
|
995
|
+
fp = cx->fp;
|
996
|
+
while (fp) {
|
997
|
+
if (fp->script)
|
998
|
+
return fp;
|
999
|
+
fp = fp->down;
|
1000
|
+
}
|
1001
|
+
return NULL;
|
1002
|
+
}
|
1003
|
+
|
1004
|
+
JS_PUBLIC_API(JSPrincipals *)
|
1005
|
+
JS_StackFramePrincipals(JSContext *cx, JSStackFrame *fp)
|
1006
|
+
{
|
1007
|
+
if (fp->fun) {
|
1008
|
+
JSRuntime *rt = cx->runtime;
|
1009
|
+
|
1010
|
+
if (rt->findObjectPrincipals) {
|
1011
|
+
if (FUN_OBJECT(fp->fun) != fp->callee)
|
1012
|
+
return rt->findObjectPrincipals(cx, fp->callee);
|
1013
|
+
/* FALL THROUGH */
|
1014
|
+
}
|
1015
|
+
}
|
1016
|
+
if (fp->script)
|
1017
|
+
return fp->script->principals;
|
1018
|
+
return NULL;
|
1019
|
+
}
|
1020
|
+
|
1021
|
+
JS_PUBLIC_API(JSPrincipals *)
|
1022
|
+
JS_EvalFramePrincipals(JSContext *cx, JSStackFrame *fp, JSStackFrame *caller)
|
1023
|
+
{
|
1024
|
+
JSRuntime *rt;
|
1025
|
+
JSPrincipals *principals, *callerPrincipals;
|
1026
|
+
|
1027
|
+
rt = cx->runtime;
|
1028
|
+
if (rt->findObjectPrincipals) {
|
1029
|
+
principals = rt->findObjectPrincipals(cx, fp->callee);
|
1030
|
+
} else {
|
1031
|
+
principals = NULL;
|
1032
|
+
}
|
1033
|
+
if (!caller)
|
1034
|
+
return principals;
|
1035
|
+
callerPrincipals = JS_StackFramePrincipals(cx, caller);
|
1036
|
+
return (callerPrincipals && principals &&
|
1037
|
+
callerPrincipals->subsume(callerPrincipals, principals))
|
1038
|
+
? principals
|
1039
|
+
: callerPrincipals;
|
1040
|
+
}
|
1041
|
+
|
1042
|
+
JS_PUBLIC_API(void *)
|
1043
|
+
JS_GetFrameAnnotation(JSContext *cx, JSStackFrame *fp)
|
1044
|
+
{
|
1045
|
+
if (fp->annotation && fp->script) {
|
1046
|
+
JSPrincipals *principals = JS_StackFramePrincipals(cx, fp);
|
1047
|
+
|
1048
|
+
if (principals && principals->globalPrivilegesEnabled(cx, principals)) {
|
1049
|
+
/*
|
1050
|
+
* Give out an annotation only if privileges have not been revoked
|
1051
|
+
* or disabled globally.
|
1052
|
+
*/
|
1053
|
+
return fp->annotation;
|
1054
|
+
}
|
1055
|
+
}
|
1056
|
+
|
1057
|
+
return NULL;
|
1058
|
+
}
|
1059
|
+
|
1060
|
+
JS_PUBLIC_API(void)
|
1061
|
+
JS_SetFrameAnnotation(JSContext *cx, JSStackFrame *fp, void *annotation)
|
1062
|
+
{
|
1063
|
+
fp->annotation = annotation;
|
1064
|
+
}
|
1065
|
+
|
1066
|
+
JS_PUBLIC_API(void *)
|
1067
|
+
JS_GetFramePrincipalArray(JSContext *cx, JSStackFrame *fp)
|
1068
|
+
{
|
1069
|
+
JSPrincipals *principals;
|
1070
|
+
|
1071
|
+
principals = JS_StackFramePrincipals(cx, fp);
|
1072
|
+
if (!principals)
|
1073
|
+
return NULL;
|
1074
|
+
return principals->getPrincipalArray(cx, principals);
|
1075
|
+
}
|
1076
|
+
|
1077
|
+
JS_PUBLIC_API(JSBool)
|
1078
|
+
JS_IsNativeFrame(JSContext *cx, JSStackFrame *fp)
|
1079
|
+
{
|
1080
|
+
return !fp->script;
|
1081
|
+
}
|
1082
|
+
|
1083
|
+
/* this is deprecated, use JS_GetFrameScopeChain instead */
|
1084
|
+
JS_PUBLIC_API(JSObject *)
|
1085
|
+
JS_GetFrameObject(JSContext *cx, JSStackFrame *fp)
|
1086
|
+
{
|
1087
|
+
return fp->scopeChain;
|
1088
|
+
}
|
1089
|
+
|
1090
|
+
JS_PUBLIC_API(JSObject *)
|
1091
|
+
JS_GetFrameScopeChain(JSContext *cx, JSStackFrame *fp)
|
1092
|
+
{
|
1093
|
+
/* Force creation of argument and call objects if not yet created */
|
1094
|
+
(void) JS_GetFrameCallObject(cx, fp);
|
1095
|
+
return js_GetScopeChain(cx, fp);
|
1096
|
+
}
|
1097
|
+
|
1098
|
+
JS_PUBLIC_API(JSObject *)
|
1099
|
+
JS_GetFrameCallObject(JSContext *cx, JSStackFrame *fp)
|
1100
|
+
{
|
1101
|
+
if (! fp->fun)
|
1102
|
+
return NULL;
|
1103
|
+
|
1104
|
+
/* Force creation of argument object if not yet created */
|
1105
|
+
(void) js_GetArgsObject(cx, fp);
|
1106
|
+
|
1107
|
+
/*
|
1108
|
+
* XXX ill-defined: null return here means error was reported, unlike a
|
1109
|
+
* null returned above or in the #else
|
1110
|
+
*/
|
1111
|
+
return js_GetCallObject(cx, fp, NULL);
|
1112
|
+
}
|
1113
|
+
|
1114
|
+
JS_PUBLIC_API(JSObject *)
|
1115
|
+
JS_GetFrameThis(JSContext *cx, JSStackFrame *fp)
|
1116
|
+
{
|
1117
|
+
JSStackFrame *afp;
|
1118
|
+
|
1119
|
+
if (fp->flags & JSFRAME_COMPUTED_THIS)
|
1120
|
+
return fp->thisp;
|
1121
|
+
|
1122
|
+
/* js_ComputeThis gets confused if fp != cx->fp, so set it aside. */
|
1123
|
+
if (cx->fp != fp) {
|
1124
|
+
afp = cx->fp;
|
1125
|
+
if (afp) {
|
1126
|
+
afp->dormantNext = cx->dormantFrameChain;
|
1127
|
+
cx->dormantFrameChain = afp;
|
1128
|
+
cx->fp = fp;
|
1129
|
+
}
|
1130
|
+
} else {
|
1131
|
+
afp = NULL;
|
1132
|
+
}
|
1133
|
+
|
1134
|
+
if (!fp->thisp && fp->argv)
|
1135
|
+
fp->thisp = js_ComputeThis(cx, JS_TRUE, fp->argv);
|
1136
|
+
|
1137
|
+
if (afp) {
|
1138
|
+
cx->fp = afp;
|
1139
|
+
cx->dormantFrameChain = afp->dormantNext;
|
1140
|
+
afp->dormantNext = NULL;
|
1141
|
+
}
|
1142
|
+
|
1143
|
+
return fp->thisp;
|
1144
|
+
}
|
1145
|
+
|
1146
|
+
JS_PUBLIC_API(JSFunction *)
|
1147
|
+
JS_GetFrameFunction(JSContext *cx, JSStackFrame *fp)
|
1148
|
+
{
|
1149
|
+
return fp->fun;
|
1150
|
+
}
|
1151
|
+
|
1152
|
+
JS_PUBLIC_API(JSObject *)
|
1153
|
+
JS_GetFrameFunctionObject(JSContext *cx, JSStackFrame *fp)
|
1154
|
+
{
|
1155
|
+
/*
|
1156
|
+
* Test both fp->fun and fp->argv to defend against any control flow from
|
1157
|
+
* the compiler reaching this API entry point, where fp is a frame pushed
|
1158
|
+
* by the compiler that has non-null fun but null argv.
|
1159
|
+
*/
|
1160
|
+
if (fp->fun && fp->argv) {
|
1161
|
+
JSObject *obj = fp->callee;
|
1162
|
+
|
1163
|
+
JS_ASSERT(OBJ_GET_CLASS(cx, obj) == &js_FunctionClass);
|
1164
|
+
JS_ASSERT(OBJ_GET_PRIVATE(cx, obj) == fp->fun);
|
1165
|
+
return obj;
|
1166
|
+
}
|
1167
|
+
return NULL;
|
1168
|
+
}
|
1169
|
+
|
1170
|
+
JS_PUBLIC_API(JSBool)
|
1171
|
+
JS_IsConstructorFrame(JSContext *cx, JSStackFrame *fp)
|
1172
|
+
{
|
1173
|
+
return (fp->flags & JSFRAME_CONSTRUCTING) != 0;
|
1174
|
+
}
|
1175
|
+
|
1176
|
+
JS_PUBLIC_API(JSObject *)
|
1177
|
+
JS_GetFrameCalleeObject(JSContext *cx, JSStackFrame *fp)
|
1178
|
+
{
|
1179
|
+
return fp->callee;
|
1180
|
+
}
|
1181
|
+
|
1182
|
+
JS_PUBLIC_API(JSBool)
|
1183
|
+
JS_IsDebuggerFrame(JSContext *cx, JSStackFrame *fp)
|
1184
|
+
{
|
1185
|
+
return (fp->flags & JSFRAME_DEBUGGER) != 0;
|
1186
|
+
}
|
1187
|
+
|
1188
|
+
JS_PUBLIC_API(jsval)
|
1189
|
+
JS_GetFrameReturnValue(JSContext *cx, JSStackFrame *fp)
|
1190
|
+
{
|
1191
|
+
return fp->rval;
|
1192
|
+
}
|
1193
|
+
|
1194
|
+
JS_PUBLIC_API(void)
|
1195
|
+
JS_SetFrameReturnValue(JSContext *cx, JSStackFrame *fp, jsval rval)
|
1196
|
+
{
|
1197
|
+
fp->rval = rval;
|
1198
|
+
}
|
1199
|
+
|
1200
|
+
/************************************************************************/
|
1201
|
+
|
1202
|
+
JS_PUBLIC_API(const char *)
|
1203
|
+
JS_GetScriptFilename(JSContext *cx, JSScript *script)
|
1204
|
+
{
|
1205
|
+
return script->filename;
|
1206
|
+
}
|
1207
|
+
|
1208
|
+
JS_PUBLIC_API(uintN)
|
1209
|
+
JS_GetScriptBaseLineNumber(JSContext *cx, JSScript *script)
|
1210
|
+
{
|
1211
|
+
return script->lineno;
|
1212
|
+
}
|
1213
|
+
|
1214
|
+
JS_PUBLIC_API(uintN)
|
1215
|
+
JS_GetScriptLineExtent(JSContext *cx, JSScript *script)
|
1216
|
+
{
|
1217
|
+
return js_GetScriptLineExtent(script);
|
1218
|
+
}
|
1219
|
+
|
1220
|
+
JS_PUBLIC_API(JSVersion)
|
1221
|
+
JS_GetScriptVersion(JSContext *cx, JSScript *script)
|
1222
|
+
{
|
1223
|
+
return (JSVersion) (script->version & JSVERSION_MASK);
|
1224
|
+
}
|
1225
|
+
|
1226
|
+
/***************************************************************************/
|
1227
|
+
|
1228
|
+
JS_PUBLIC_API(void)
|
1229
|
+
JS_SetNewScriptHook(JSRuntime *rt, JSNewScriptHook hook, void *callerdata)
|
1230
|
+
{
|
1231
|
+
rt->globalDebugHooks.newScriptHook = hook;
|
1232
|
+
rt->globalDebugHooks.newScriptHookData = callerdata;
|
1233
|
+
}
|
1234
|
+
|
1235
|
+
JS_PUBLIC_API(void)
|
1236
|
+
JS_SetDestroyScriptHook(JSRuntime *rt, JSDestroyScriptHook hook,
|
1237
|
+
void *callerdata)
|
1238
|
+
{
|
1239
|
+
rt->globalDebugHooks.destroyScriptHook = hook;
|
1240
|
+
rt->globalDebugHooks.destroyScriptHookData = callerdata;
|
1241
|
+
}
|
1242
|
+
|
1243
|
+
/***************************************************************************/
|
1244
|
+
|
1245
|
+
JS_PUBLIC_API(JSBool)
|
1246
|
+
JS_EvaluateUCInStackFrame(JSContext *cx, JSStackFrame *fp,
|
1247
|
+
const jschar *chars, uintN length,
|
1248
|
+
const char *filename, uintN lineno,
|
1249
|
+
jsval *rval)
|
1250
|
+
{
|
1251
|
+
JSObject *scobj;
|
1252
|
+
uint32 flags;
|
1253
|
+
JSScript *script;
|
1254
|
+
JSBool ok;
|
1255
|
+
|
1256
|
+
scobj = JS_GetFrameScopeChain(cx, fp);
|
1257
|
+
if (!scobj)
|
1258
|
+
return JS_FALSE;
|
1259
|
+
|
1260
|
+
/*
|
1261
|
+
* XXX Hack around ancient compiler API to propagate the JSFRAME_SPECIAL
|
1262
|
+
* flags to the code generator.
|
1263
|
+
*/
|
1264
|
+
flags = fp->flags;
|
1265
|
+
fp->flags |= JSFRAME_DEBUGGER | JSFRAME_EVAL;
|
1266
|
+
script = js_CompileScript(cx, scobj, JS_StackFramePrincipals(cx, fp),
|
1267
|
+
TCF_COMPILE_N_GO, chars, length, NULL,
|
1268
|
+
filename, lineno);
|
1269
|
+
fp->flags = flags;
|
1270
|
+
if (!script)
|
1271
|
+
return JS_FALSE;
|
1272
|
+
|
1273
|
+
ok = js_Execute(cx, scobj, script, fp, JSFRAME_DEBUGGER | JSFRAME_EVAL,
|
1274
|
+
rval);
|
1275
|
+
js_DestroyScript(cx, script);
|
1276
|
+
return ok;
|
1277
|
+
}
|
1278
|
+
|
1279
|
+
JS_PUBLIC_API(JSBool)
|
1280
|
+
JS_EvaluateInStackFrame(JSContext *cx, JSStackFrame *fp,
|
1281
|
+
const char *bytes, uintN length,
|
1282
|
+
const char *filename, uintN lineno,
|
1283
|
+
jsval *rval)
|
1284
|
+
{
|
1285
|
+
jschar *chars;
|
1286
|
+
JSBool ok;
|
1287
|
+
size_t len = length;
|
1288
|
+
|
1289
|
+
chars = js_InflateString(cx, bytes, &len);
|
1290
|
+
if (!chars)
|
1291
|
+
return JS_FALSE;
|
1292
|
+
length = (uintN) len;
|
1293
|
+
ok = JS_EvaluateUCInStackFrame(cx, fp, chars, length, filename, lineno,
|
1294
|
+
rval);
|
1295
|
+
JS_free(cx, chars);
|
1296
|
+
|
1297
|
+
return ok;
|
1298
|
+
}
|
1299
|
+
|
1300
|
+
/************************************************************************/
|
1301
|
+
|
1302
|
+
/* XXXbe this all needs to be reworked to avoid requiring JSScope types. */
|
1303
|
+
|
1304
|
+
JS_PUBLIC_API(JSScopeProperty *)
|
1305
|
+
JS_PropertyIterator(JSObject *obj, JSScopeProperty **iteratorp)
|
1306
|
+
{
|
1307
|
+
JSScopeProperty *sprop;
|
1308
|
+
JSScope *scope;
|
1309
|
+
|
1310
|
+
sprop = *iteratorp;
|
1311
|
+
scope = OBJ_SCOPE(obj);
|
1312
|
+
|
1313
|
+
/* XXXbe minor(?) incompatibility: iterate in reverse definition order */
|
1314
|
+
if (!sprop) {
|
1315
|
+
sprop = SCOPE_LAST_PROP(scope);
|
1316
|
+
} else {
|
1317
|
+
while ((sprop = sprop->parent) != NULL) {
|
1318
|
+
if (!SCOPE_HAD_MIDDLE_DELETE(scope))
|
1319
|
+
break;
|
1320
|
+
if (SCOPE_HAS_PROPERTY(scope, sprop))
|
1321
|
+
break;
|
1322
|
+
}
|
1323
|
+
}
|
1324
|
+
*iteratorp = sprop;
|
1325
|
+
return sprop;
|
1326
|
+
}
|
1327
|
+
|
1328
|
+
JS_PUBLIC_API(JSBool)
|
1329
|
+
JS_GetPropertyDesc(JSContext *cx, JSObject *obj, JSScopeProperty *sprop,
|
1330
|
+
JSPropertyDesc *pd)
|
1331
|
+
{
|
1332
|
+
JSScope *scope;
|
1333
|
+
JSScopeProperty *aprop;
|
1334
|
+
jsval lastException;
|
1335
|
+
JSBool wasThrowing;
|
1336
|
+
|
1337
|
+
pd->id = ID_TO_VALUE(sprop->id);
|
1338
|
+
|
1339
|
+
wasThrowing = cx->throwing;
|
1340
|
+
if (wasThrowing) {
|
1341
|
+
lastException = cx->exception;
|
1342
|
+
if (JSVAL_IS_GCTHING(lastException) &&
|
1343
|
+
!js_AddRoot(cx, &lastException, "lastException")) {
|
1344
|
+
return JS_FALSE;
|
1345
|
+
}
|
1346
|
+
cx->throwing = JS_FALSE;
|
1347
|
+
}
|
1348
|
+
|
1349
|
+
if (!js_GetProperty(cx, obj, sprop->id, &pd->value)) {
|
1350
|
+
if (!cx->throwing) {
|
1351
|
+
pd->flags = JSPD_ERROR;
|
1352
|
+
pd->value = JSVAL_VOID;
|
1353
|
+
} else {
|
1354
|
+
pd->flags = JSPD_EXCEPTION;
|
1355
|
+
pd->value = cx->exception;
|
1356
|
+
}
|
1357
|
+
} else {
|
1358
|
+
pd->flags = 0;
|
1359
|
+
}
|
1360
|
+
|
1361
|
+
cx->throwing = wasThrowing;
|
1362
|
+
if (wasThrowing) {
|
1363
|
+
cx->exception = lastException;
|
1364
|
+
if (JSVAL_IS_GCTHING(lastException))
|
1365
|
+
js_RemoveRoot(cx->runtime, &lastException);
|
1366
|
+
}
|
1367
|
+
|
1368
|
+
pd->flags |= ((sprop->attrs & JSPROP_ENUMERATE) ? JSPD_ENUMERATE : 0)
|
1369
|
+
| ((sprop->attrs & JSPROP_READONLY) ? JSPD_READONLY : 0)
|
1370
|
+
| ((sprop->attrs & JSPROP_PERMANENT) ? JSPD_PERMANENT : 0);
|
1371
|
+
pd->spare = 0;
|
1372
|
+
if (sprop->getter == js_GetCallArg) {
|
1373
|
+
pd->slot = sprop->shortid;
|
1374
|
+
pd->flags |= JSPD_ARGUMENT;
|
1375
|
+
} else if (sprop->getter == js_GetCallVar) {
|
1376
|
+
pd->slot = sprop->shortid;
|
1377
|
+
pd->flags |= JSPD_VARIABLE;
|
1378
|
+
} else {
|
1379
|
+
pd->slot = 0;
|
1380
|
+
}
|
1381
|
+
pd->alias = JSVAL_VOID;
|
1382
|
+
scope = OBJ_SCOPE(obj);
|
1383
|
+
if (SPROP_HAS_VALID_SLOT(sprop, scope)) {
|
1384
|
+
for (aprop = SCOPE_LAST_PROP(scope); aprop; aprop = aprop->parent) {
|
1385
|
+
if (aprop != sprop && aprop->slot == sprop->slot) {
|
1386
|
+
pd->alias = ID_TO_VALUE(aprop->id);
|
1387
|
+
break;
|
1388
|
+
}
|
1389
|
+
}
|
1390
|
+
}
|
1391
|
+
return JS_TRUE;
|
1392
|
+
}
|
1393
|
+
|
1394
|
+
JS_PUBLIC_API(JSBool)
|
1395
|
+
JS_GetPropertyDescArray(JSContext *cx, JSObject *obj, JSPropertyDescArray *pda)
|
1396
|
+
{
|
1397
|
+
JSClass *clasp;
|
1398
|
+
JSScope *scope;
|
1399
|
+
uint32 i, n;
|
1400
|
+
JSPropertyDesc *pd;
|
1401
|
+
JSScopeProperty *sprop;
|
1402
|
+
|
1403
|
+
clasp = OBJ_GET_CLASS(cx, obj);
|
1404
|
+
if (!OBJ_IS_NATIVE(obj) || (clasp->flags & JSCLASS_NEW_ENUMERATE)) {
|
1405
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
1406
|
+
JSMSG_CANT_DESCRIBE_PROPS, clasp->name);
|
1407
|
+
return JS_FALSE;
|
1408
|
+
}
|
1409
|
+
if (!clasp->enumerate(cx, obj))
|
1410
|
+
return JS_FALSE;
|
1411
|
+
|
1412
|
+
/* have no props, or object's scope has not mutated from that of proto */
|
1413
|
+
scope = OBJ_SCOPE(obj);
|
1414
|
+
if (scope->object != obj || scope->entryCount == 0) {
|
1415
|
+
pda->length = 0;
|
1416
|
+
pda->array = NULL;
|
1417
|
+
return JS_TRUE;
|
1418
|
+
}
|
1419
|
+
|
1420
|
+
n = STOBJ_NSLOTS(obj);
|
1421
|
+
if (n > scope->entryCount)
|
1422
|
+
n = scope->entryCount;
|
1423
|
+
pd = (JSPropertyDesc *) JS_malloc(cx, (size_t)n * sizeof(JSPropertyDesc));
|
1424
|
+
if (!pd)
|
1425
|
+
return JS_FALSE;
|
1426
|
+
i = 0;
|
1427
|
+
for (sprop = SCOPE_LAST_PROP(scope); sprop; sprop = sprop->parent) {
|
1428
|
+
if (SCOPE_HAD_MIDDLE_DELETE(scope) && !SCOPE_HAS_PROPERTY(scope, sprop))
|
1429
|
+
continue;
|
1430
|
+
if (!js_AddRoot(cx, &pd[i].id, NULL))
|
1431
|
+
goto bad;
|
1432
|
+
if (!js_AddRoot(cx, &pd[i].value, NULL))
|
1433
|
+
goto bad;
|
1434
|
+
if (!JS_GetPropertyDesc(cx, obj, sprop, &pd[i]))
|
1435
|
+
goto bad;
|
1436
|
+
if ((pd[i].flags & JSPD_ALIAS) && !js_AddRoot(cx, &pd[i].alias, NULL))
|
1437
|
+
goto bad;
|
1438
|
+
if (++i == n)
|
1439
|
+
break;
|
1440
|
+
}
|
1441
|
+
pda->length = i;
|
1442
|
+
pda->array = pd;
|
1443
|
+
return JS_TRUE;
|
1444
|
+
|
1445
|
+
bad:
|
1446
|
+
pda->length = i + 1;
|
1447
|
+
pda->array = pd;
|
1448
|
+
JS_PutPropertyDescArray(cx, pda);
|
1449
|
+
return JS_FALSE;
|
1450
|
+
}
|
1451
|
+
|
1452
|
+
JS_PUBLIC_API(void)
|
1453
|
+
JS_PutPropertyDescArray(JSContext *cx, JSPropertyDescArray *pda)
|
1454
|
+
{
|
1455
|
+
JSPropertyDesc *pd;
|
1456
|
+
uint32 i;
|
1457
|
+
|
1458
|
+
pd = pda->array;
|
1459
|
+
for (i = 0; i < pda->length; i++) {
|
1460
|
+
js_RemoveRoot(cx->runtime, &pd[i].id);
|
1461
|
+
js_RemoveRoot(cx->runtime, &pd[i].value);
|
1462
|
+
if (pd[i].flags & JSPD_ALIAS)
|
1463
|
+
js_RemoveRoot(cx->runtime, &pd[i].alias);
|
1464
|
+
}
|
1465
|
+
JS_free(cx, pd);
|
1466
|
+
}
|
1467
|
+
|
1468
|
+
/************************************************************************/
|
1469
|
+
|
1470
|
+
JS_PUBLIC_API(JSBool)
|
1471
|
+
JS_SetDebuggerHandler(JSRuntime *rt, JSTrapHandler handler, void *closure)
|
1472
|
+
{
|
1473
|
+
rt->globalDebugHooks.debuggerHandler = handler;
|
1474
|
+
rt->globalDebugHooks.debuggerHandlerData = closure;
|
1475
|
+
return JS_TRUE;
|
1476
|
+
}
|
1477
|
+
|
1478
|
+
JS_PUBLIC_API(JSBool)
|
1479
|
+
JS_SetSourceHandler(JSRuntime *rt, JSSourceHandler handler, void *closure)
|
1480
|
+
{
|
1481
|
+
rt->globalDebugHooks.sourceHandler = handler;
|
1482
|
+
rt->globalDebugHooks.sourceHandlerData = closure;
|
1483
|
+
return JS_TRUE;
|
1484
|
+
}
|
1485
|
+
|
1486
|
+
JS_PUBLIC_API(JSBool)
|
1487
|
+
JS_SetExecuteHook(JSRuntime *rt, JSInterpreterHook hook, void *closure)
|
1488
|
+
{
|
1489
|
+
rt->globalDebugHooks.executeHook = hook;
|
1490
|
+
rt->globalDebugHooks.executeHookData = closure;
|
1491
|
+
return JS_TRUE;
|
1492
|
+
}
|
1493
|
+
|
1494
|
+
JS_PUBLIC_API(JSBool)
|
1495
|
+
JS_SetCallHook(JSRuntime *rt, JSInterpreterHook hook, void *closure)
|
1496
|
+
{
|
1497
|
+
rt->globalDebugHooks.callHook = hook;
|
1498
|
+
rt->globalDebugHooks.callHookData = closure;
|
1499
|
+
return JS_TRUE;
|
1500
|
+
}
|
1501
|
+
|
1502
|
+
JS_PUBLIC_API(JSBool)
|
1503
|
+
JS_SetObjectHook(JSRuntime *rt, JSObjectHook hook, void *closure)
|
1504
|
+
{
|
1505
|
+
rt->globalDebugHooks.objectHook = hook;
|
1506
|
+
rt->globalDebugHooks.objectHookData = closure;
|
1507
|
+
return JS_TRUE;
|
1508
|
+
}
|
1509
|
+
|
1510
|
+
JS_PUBLIC_API(JSBool)
|
1511
|
+
JS_SetThrowHook(JSRuntime *rt, JSTrapHandler hook, void *closure)
|
1512
|
+
{
|
1513
|
+
rt->globalDebugHooks.throwHook = hook;
|
1514
|
+
rt->globalDebugHooks.throwHookData = closure;
|
1515
|
+
return JS_TRUE;
|
1516
|
+
}
|
1517
|
+
|
1518
|
+
JS_PUBLIC_API(JSBool)
|
1519
|
+
JS_SetDebugErrorHook(JSRuntime *rt, JSDebugErrorHook hook, void *closure)
|
1520
|
+
{
|
1521
|
+
rt->globalDebugHooks.debugErrorHook = hook;
|
1522
|
+
rt->globalDebugHooks.debugErrorHookData = closure;
|
1523
|
+
return JS_TRUE;
|
1524
|
+
}
|
1525
|
+
|
1526
|
+
/************************************************************************/
|
1527
|
+
|
1528
|
+
JS_PUBLIC_API(size_t)
|
1529
|
+
JS_GetObjectTotalSize(JSContext *cx, JSObject *obj)
|
1530
|
+
{
|
1531
|
+
size_t nbytes;
|
1532
|
+
JSScope *scope;
|
1533
|
+
|
1534
|
+
nbytes = sizeof *obj;
|
1535
|
+
if (obj->dslots) {
|
1536
|
+
nbytes += ((uint32)obj->dslots[-1] - JS_INITIAL_NSLOTS + 1)
|
1537
|
+
* sizeof obj->dslots[0];
|
1538
|
+
}
|
1539
|
+
if (OBJ_IS_NATIVE(obj)) {
|
1540
|
+
scope = OBJ_SCOPE(obj);
|
1541
|
+
if (scope->object == obj) {
|
1542
|
+
nbytes += sizeof *scope;
|
1543
|
+
nbytes += SCOPE_CAPACITY(scope) * sizeof(JSScopeProperty *);
|
1544
|
+
}
|
1545
|
+
}
|
1546
|
+
return nbytes;
|
1547
|
+
}
|
1548
|
+
|
1549
|
+
static size_t
|
1550
|
+
GetAtomTotalSize(JSContext *cx, JSAtom *atom)
|
1551
|
+
{
|
1552
|
+
size_t nbytes;
|
1553
|
+
|
1554
|
+
nbytes = sizeof(JSAtom *) + sizeof(JSDHashEntryStub);
|
1555
|
+
if (ATOM_IS_STRING(atom)) {
|
1556
|
+
nbytes += sizeof(JSString);
|
1557
|
+
nbytes += (JSFLATSTR_LENGTH(ATOM_TO_STRING(atom)) + 1) * sizeof(jschar);
|
1558
|
+
} else if (ATOM_IS_DOUBLE(atom)) {
|
1559
|
+
nbytes += sizeof(jsdouble);
|
1560
|
+
}
|
1561
|
+
return nbytes;
|
1562
|
+
}
|
1563
|
+
|
1564
|
+
JS_PUBLIC_API(size_t)
|
1565
|
+
JS_GetFunctionTotalSize(JSContext *cx, JSFunction *fun)
|
1566
|
+
{
|
1567
|
+
size_t nbytes;
|
1568
|
+
|
1569
|
+
nbytes = sizeof *fun;
|
1570
|
+
nbytes += JS_GetObjectTotalSize(cx, FUN_OBJECT(fun));
|
1571
|
+
if (FUN_INTERPRETED(fun))
|
1572
|
+
nbytes += JS_GetScriptTotalSize(cx, fun->u.i.script);
|
1573
|
+
if (fun->atom)
|
1574
|
+
nbytes += GetAtomTotalSize(cx, fun->atom);
|
1575
|
+
return nbytes;
|
1576
|
+
}
|
1577
|
+
|
1578
|
+
#include "jsemit.h"
|
1579
|
+
|
1580
|
+
JS_PUBLIC_API(size_t)
|
1581
|
+
JS_GetScriptTotalSize(JSContext *cx, JSScript *script)
|
1582
|
+
{
|
1583
|
+
size_t nbytes, pbytes;
|
1584
|
+
jsatomid i;
|
1585
|
+
jssrcnote *sn, *notes;
|
1586
|
+
JSObjectArray *objarray;
|
1587
|
+
JSPrincipals *principals;
|
1588
|
+
|
1589
|
+
nbytes = sizeof *script;
|
1590
|
+
if (script->object)
|
1591
|
+
nbytes += JS_GetObjectTotalSize(cx, script->object);
|
1592
|
+
|
1593
|
+
nbytes += script->length * sizeof script->code[0];
|
1594
|
+
nbytes += script->atomMap.length * sizeof script->atomMap.vector[0];
|
1595
|
+
for (i = 0; i < script->atomMap.length; i++)
|
1596
|
+
nbytes += GetAtomTotalSize(cx, script->atomMap.vector[i]);
|
1597
|
+
|
1598
|
+
if (script->filename)
|
1599
|
+
nbytes += strlen(script->filename) + 1;
|
1600
|
+
|
1601
|
+
notes = SCRIPT_NOTES(script);
|
1602
|
+
for (sn = notes; !SN_IS_TERMINATOR(sn); sn = SN_NEXT(sn))
|
1603
|
+
continue;
|
1604
|
+
nbytes += (sn - notes + 1) * sizeof *sn;
|
1605
|
+
|
1606
|
+
if (script->objectsOffset != 0) {
|
1607
|
+
objarray = JS_SCRIPT_OBJECTS(script);
|
1608
|
+
i = objarray->length;
|
1609
|
+
nbytes += sizeof *objarray + i * sizeof objarray->vector[0];
|
1610
|
+
do {
|
1611
|
+
nbytes += JS_GetObjectTotalSize(cx, objarray->vector[--i]);
|
1612
|
+
} while (i != 0);
|
1613
|
+
}
|
1614
|
+
|
1615
|
+
if (script->regexpsOffset != 0) {
|
1616
|
+
objarray = JS_SCRIPT_REGEXPS(script);
|
1617
|
+
i = objarray->length;
|
1618
|
+
nbytes += sizeof *objarray + i * sizeof objarray->vector[0];
|
1619
|
+
do {
|
1620
|
+
nbytes += JS_GetObjectTotalSize(cx, objarray->vector[--i]);
|
1621
|
+
} while (i != 0);
|
1622
|
+
}
|
1623
|
+
|
1624
|
+
if (script->trynotesOffset != 0) {
|
1625
|
+
nbytes += sizeof(JSTryNoteArray) +
|
1626
|
+
JS_SCRIPT_TRYNOTES(script)->length * sizeof(JSTryNote);
|
1627
|
+
}
|
1628
|
+
|
1629
|
+
principals = script->principals;
|
1630
|
+
if (principals) {
|
1631
|
+
JS_ASSERT(principals->refcount);
|
1632
|
+
pbytes = sizeof *principals;
|
1633
|
+
if (principals->refcount > 1)
|
1634
|
+
pbytes = JS_HOWMANY(pbytes, principals->refcount);
|
1635
|
+
nbytes += pbytes;
|
1636
|
+
}
|
1637
|
+
|
1638
|
+
return nbytes;
|
1639
|
+
}
|
1640
|
+
|
1641
|
+
JS_PUBLIC_API(uint32)
|
1642
|
+
JS_GetTopScriptFilenameFlags(JSContext *cx, JSStackFrame *fp)
|
1643
|
+
{
|
1644
|
+
if (!fp)
|
1645
|
+
fp = cx->fp;
|
1646
|
+
while (fp) {
|
1647
|
+
if (fp->script)
|
1648
|
+
return JS_GetScriptFilenameFlags(fp->script);
|
1649
|
+
fp = fp->down;
|
1650
|
+
}
|
1651
|
+
return 0;
|
1652
|
+
}
|
1653
|
+
|
1654
|
+
JS_PUBLIC_API(uint32)
|
1655
|
+
JS_GetScriptFilenameFlags(JSScript *script)
|
1656
|
+
{
|
1657
|
+
JS_ASSERT(script);
|
1658
|
+
if (!script->filename)
|
1659
|
+
return JSFILENAME_NULL;
|
1660
|
+
return js_GetScriptFilenameFlags(script->filename);
|
1661
|
+
}
|
1662
|
+
|
1663
|
+
JS_PUBLIC_API(JSBool)
|
1664
|
+
JS_FlagScriptFilenamePrefix(JSRuntime *rt, const char *prefix, uint32 flags)
|
1665
|
+
{
|
1666
|
+
if (!js_SaveScriptFilenameRT(rt, prefix, flags))
|
1667
|
+
return JS_FALSE;
|
1668
|
+
return JS_TRUE;
|
1669
|
+
}
|
1670
|
+
|
1671
|
+
JS_PUBLIC_API(JSBool)
|
1672
|
+
JS_IsSystemObject(JSContext *cx, JSObject *obj)
|
1673
|
+
{
|
1674
|
+
return STOBJ_IS_SYSTEM(obj);
|
1675
|
+
}
|
1676
|
+
|
1677
|
+
JS_PUBLIC_API(JSObject *)
|
1678
|
+
JS_NewSystemObject(JSContext *cx, JSClass *clasp, JSObject *proto,
|
1679
|
+
JSObject *parent, JSBool system)
|
1680
|
+
{
|
1681
|
+
JSObject *obj;
|
1682
|
+
|
1683
|
+
obj = js_NewObject(cx, clasp, proto, parent, 0);
|
1684
|
+
if (obj && system)
|
1685
|
+
STOBJ_SET_SYSTEM(obj);
|
1686
|
+
return obj;
|
1687
|
+
}
|
1688
|
+
|
1689
|
+
/************************************************************************/
|
1690
|
+
|
1691
|
+
JS_PUBLIC_API(JSDebugHooks *)
|
1692
|
+
JS_GetGlobalDebugHooks(JSRuntime *rt)
|
1693
|
+
{
|
1694
|
+
return &rt->globalDebugHooks;
|
1695
|
+
}
|
1696
|
+
|
1697
|
+
JS_PUBLIC_API(JSDebugHooks *)
|
1698
|
+
JS_SetContextDebugHooks(JSContext *cx, JSDebugHooks *hooks)
|
1699
|
+
{
|
1700
|
+
JSDebugHooks *old;
|
1701
|
+
|
1702
|
+
JS_ASSERT(hooks);
|
1703
|
+
old = cx->debugHooks;
|
1704
|
+
cx->debugHooks = hooks;
|
1705
|
+
return old;
|
1706
|
+
}
|
1707
|
+
|
1708
|
+
#ifdef MOZ_SHARK
|
1709
|
+
|
1710
|
+
#include <CHUD/CHUD.h>
|
1711
|
+
|
1712
|
+
JS_PUBLIC_API(JSBool)
|
1713
|
+
JS_StartChudRemote()
|
1714
|
+
{
|
1715
|
+
if (chudIsRemoteAccessAcquired() &&
|
1716
|
+
(chudStartRemotePerfMonitor("Mozilla") == chudSuccess)) {
|
1717
|
+
return JS_TRUE;
|
1718
|
+
}
|
1719
|
+
|
1720
|
+
return JS_FALSE;
|
1721
|
+
}
|
1722
|
+
|
1723
|
+
JS_PUBLIC_API(JSBool)
|
1724
|
+
JS_StopChudRemote()
|
1725
|
+
{
|
1726
|
+
if (chudIsRemoteAccessAcquired() &&
|
1727
|
+
(chudStopRemotePerfMonitor() == chudSuccess)) {
|
1728
|
+
return JS_TRUE;
|
1729
|
+
}
|
1730
|
+
|
1731
|
+
return JS_FALSE;
|
1732
|
+
}
|
1733
|
+
|
1734
|
+
JS_PUBLIC_API(JSBool)
|
1735
|
+
JS_ConnectShark()
|
1736
|
+
{
|
1737
|
+
if (!chudIsInitialized() && (chudInitialize() != chudSuccess))
|
1738
|
+
return JS_FALSE;
|
1739
|
+
|
1740
|
+
if (chudAcquireRemoteAccess() != chudSuccess)
|
1741
|
+
return JS_FALSE;
|
1742
|
+
|
1743
|
+
return JS_TRUE;
|
1744
|
+
}
|
1745
|
+
|
1746
|
+
JS_PUBLIC_API(JSBool)
|
1747
|
+
JS_DisconnectShark()
|
1748
|
+
{
|
1749
|
+
if (chudIsRemoteAccessAcquired() && (chudReleaseRemoteAccess() != chudSuccess))
|
1750
|
+
return JS_FALSE;
|
1751
|
+
|
1752
|
+
return JS_TRUE;
|
1753
|
+
}
|
1754
|
+
|
1755
|
+
JS_FRIEND_API(JSBool)
|
1756
|
+
js_StartShark(JSContext *cx, JSObject *obj,
|
1757
|
+
uintN argc, jsval *argv, jsval *rval)
|
1758
|
+
{
|
1759
|
+
if (!JS_StartChudRemote()) {
|
1760
|
+
JS_ReportError(cx, "Error starting CHUD.");
|
1761
|
+
}
|
1762
|
+
|
1763
|
+
return JS_TRUE;
|
1764
|
+
}
|
1765
|
+
|
1766
|
+
JS_FRIEND_API(JSBool)
|
1767
|
+
js_StopShark(JSContext *cx, JSObject *obj,
|
1768
|
+
uintN argc, jsval *argv, jsval *rval)
|
1769
|
+
{
|
1770
|
+
if (!JS_StopChudRemote()) {
|
1771
|
+
JS_ReportError(cx, "Error stopping CHUD.");
|
1772
|
+
}
|
1773
|
+
|
1774
|
+
return JS_TRUE;
|
1775
|
+
}
|
1776
|
+
|
1777
|
+
JS_FRIEND_API(JSBool)
|
1778
|
+
js_ConnectShark(JSContext *cx, JSObject *obj,
|
1779
|
+
uintN argc, jsval *argv, jsval *rval)
|
1780
|
+
{
|
1781
|
+
if (!JS_ConnectShark()) {
|
1782
|
+
JS_ReportError(cx, "Error connecting to Shark.");
|
1783
|
+
}
|
1784
|
+
|
1785
|
+
return JS_TRUE;
|
1786
|
+
}
|
1787
|
+
|
1788
|
+
JS_FRIEND_API(JSBool)
|
1789
|
+
js_DisconnectShark(JSContext *cx, JSObject *obj,
|
1790
|
+
uintN argc, jsval *argv, jsval *rval)
|
1791
|
+
{
|
1792
|
+
if (!JS_DisconnectShark()) {
|
1793
|
+
JS_ReportError(cx, "Error disconnecting from Shark.");
|
1794
|
+
}
|
1795
|
+
|
1796
|
+
return JS_TRUE;
|
1797
|
+
}
|
1798
|
+
|
1799
|
+
#endif /* MOZ_SHARK */
|