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.
Files changed (269) hide show
  1. data/MANIFEST +1 -0
  2. data/Rakefile +3 -10
  3. data/bin/johnson +2 -1
  4. data/ext/spidermonkey/context.c +3 -4
  5. data/ext/spidermonkey/context.h +1 -1
  6. data/ext/spidermonkey/conversions.c +39 -33
  7. data/ext/spidermonkey/debugger.c +5 -5
  8. data/ext/spidermonkey/immutable_node.c.erb +11 -11
  9. data/ext/spidermonkey/jroot.h +4 -4
  10. data/ext/spidermonkey/js_land_proxy.c +9 -8
  11. data/ext/spidermonkey/ruby_land_proxy.c +5 -4
  12. data/ext/spidermonkey/runtime.c +1 -1
  13. data/johnson.gemspec +36 -0
  14. data/lib/hoe.rb +0 -7
  15. data/lib/johnson/cli/options.rb +10 -4
  16. data/lib/johnson/spidermonkey/runtime.rb +2 -2
  17. data/lib/johnson/version.rb +4 -2
  18. data/lib/johnson.rb +1 -0
  19. data/test/johnson/runtime_test.rb +11 -0
  20. data/test/johnson/spidermonkey/ruby_land_proxy_test.rb +6 -0
  21. data/vendor/spidermonkey/.cvsignore +9 -0
  22. data/vendor/spidermonkey/Makefile.in +462 -0
  23. data/vendor/spidermonkey/Makefile.ref +364 -0
  24. data/vendor/spidermonkey/README.html +820 -0
  25. data/vendor/spidermonkey/SpiderMonkey.rsp +12 -0
  26. data/vendor/spidermonkey/Y.js +19 -0
  27. data/vendor/spidermonkey/build.mk +43 -0
  28. data/vendor/spidermonkey/config/AIX4.1.mk +65 -0
  29. data/vendor/spidermonkey/config/AIX4.2.mk +64 -0
  30. data/vendor/spidermonkey/config/AIX4.3.mk +65 -0
  31. data/vendor/spidermonkey/config/Darwin.mk +83 -0
  32. data/vendor/spidermonkey/config/Darwin1.3.mk +81 -0
  33. data/vendor/spidermonkey/config/Darwin1.4.mk +41 -0
  34. data/vendor/spidermonkey/config/Darwin5.2.mk +81 -0
  35. data/vendor/spidermonkey/config/Darwin5.3.mk +81 -0
  36. data/vendor/spidermonkey/config/HP-UXB.10.10.mk +77 -0
  37. data/vendor/spidermonkey/config/HP-UXB.10.20.mk +77 -0
  38. data/vendor/spidermonkey/config/HP-UXB.11.00.mk +80 -0
  39. data/vendor/spidermonkey/config/IRIX.mk +87 -0
  40. data/vendor/spidermonkey/config/IRIX5.3.mk +44 -0
  41. data/vendor/spidermonkey/config/IRIX6.1.mk +44 -0
  42. data/vendor/spidermonkey/config/IRIX6.2.mk +44 -0
  43. data/vendor/spidermonkey/config/IRIX6.3.mk +44 -0
  44. data/vendor/spidermonkey/config/IRIX6.5.mk +44 -0
  45. data/vendor/spidermonkey/config/Linux_All.mk +103 -0
  46. data/vendor/spidermonkey/config/Mac_OS10.0.mk +82 -0
  47. data/vendor/spidermonkey/config/OSF1V4.0.mk +72 -0
  48. data/vendor/spidermonkey/config/OSF1V5.0.mk +69 -0
  49. data/vendor/spidermonkey/config/SunOS4.1.4.mk +101 -0
  50. data/vendor/spidermonkey/config/SunOS5.10.mk +50 -0
  51. data/vendor/spidermonkey/config/SunOS5.3.mk +91 -0
  52. data/vendor/spidermonkey/config/SunOS5.4.mk +92 -0
  53. data/vendor/spidermonkey/config/SunOS5.5.1.mk +44 -0
  54. data/vendor/spidermonkey/config/SunOS5.5.mk +87 -0
  55. data/vendor/spidermonkey/config/SunOS5.6.mk +89 -0
  56. data/vendor/spidermonkey/config/SunOS5.7.mk +44 -0
  57. data/vendor/spidermonkey/config/SunOS5.8.mk +44 -0
  58. data/vendor/spidermonkey/config/SunOS5.9.mk +44 -0
  59. data/vendor/spidermonkey/config/WINNT4.0.mk +117 -0
  60. data/vendor/spidermonkey/config/WINNT5.0.mk +117 -0
  61. data/vendor/spidermonkey/config/WINNT5.1.mk +117 -0
  62. data/vendor/spidermonkey/config/WINNT5.2.mk +117 -0
  63. data/vendor/spidermonkey/config/WINNT6.0.mk +117 -0
  64. data/vendor/spidermonkey/config/dgux.mk +64 -0
  65. data/vendor/spidermonkey/config.mk +192 -0
  66. data/vendor/spidermonkey/editline/Makefile.ref +144 -0
  67. data/vendor/spidermonkey/editline/README +83 -0
  68. data/vendor/spidermonkey/editline/editline.3 +175 -0
  69. data/vendor/spidermonkey/editline/editline.c +1369 -0
  70. data/vendor/spidermonkey/editline/editline.h +135 -0
  71. data/vendor/spidermonkey/editline/sysunix.c +182 -0
  72. data/vendor/spidermonkey/editline/unix.h +82 -0
  73. data/vendor/spidermonkey/fdlibm/.cvsignore +7 -0
  74. data/vendor/spidermonkey/fdlibm/Makefile.in +127 -0
  75. data/vendor/spidermonkey/fdlibm/Makefile.ref +192 -0
  76. data/vendor/spidermonkey/fdlibm/e_acos.c +147 -0
  77. data/vendor/spidermonkey/fdlibm/e_acosh.c +105 -0
  78. data/vendor/spidermonkey/fdlibm/e_asin.c +156 -0
  79. data/vendor/spidermonkey/fdlibm/e_atan2.c +165 -0
  80. data/vendor/spidermonkey/fdlibm/e_atanh.c +110 -0
  81. data/vendor/spidermonkey/fdlibm/e_cosh.c +133 -0
  82. data/vendor/spidermonkey/fdlibm/e_exp.c +202 -0
  83. data/vendor/spidermonkey/fdlibm/e_fmod.c +184 -0
  84. data/vendor/spidermonkey/fdlibm/e_gamma.c +71 -0
  85. data/vendor/spidermonkey/fdlibm/e_gamma_r.c +70 -0
  86. data/vendor/spidermonkey/fdlibm/e_hypot.c +173 -0
  87. data/vendor/spidermonkey/fdlibm/e_j0.c +524 -0
  88. data/vendor/spidermonkey/fdlibm/e_j1.c +523 -0
  89. data/vendor/spidermonkey/fdlibm/e_jn.c +315 -0
  90. data/vendor/spidermonkey/fdlibm/e_lgamma.c +71 -0
  91. data/vendor/spidermonkey/fdlibm/e_lgamma_r.c +347 -0
  92. data/vendor/spidermonkey/fdlibm/e_log.c +184 -0
  93. data/vendor/spidermonkey/fdlibm/e_log10.c +134 -0
  94. data/vendor/spidermonkey/fdlibm/e_pow.c +386 -0
  95. data/vendor/spidermonkey/fdlibm/e_rem_pio2.c +222 -0
  96. data/vendor/spidermonkey/fdlibm/e_remainder.c +120 -0
  97. data/vendor/spidermonkey/fdlibm/e_scalb.c +89 -0
  98. data/vendor/spidermonkey/fdlibm/e_sinh.c +122 -0
  99. data/vendor/spidermonkey/fdlibm/e_sqrt.c +497 -0
  100. data/vendor/spidermonkey/fdlibm/fdlibm.h +273 -0
  101. data/vendor/spidermonkey/fdlibm/fdlibm.mak +1453 -0
  102. data/vendor/spidermonkey/fdlibm/fdlibm.mdp +0 -0
  103. data/vendor/spidermonkey/fdlibm/k_cos.c +135 -0
  104. data/vendor/spidermonkey/fdlibm/k_rem_pio2.c +354 -0
  105. data/vendor/spidermonkey/fdlibm/k_sin.c +114 -0
  106. data/vendor/spidermonkey/fdlibm/k_standard.c +785 -0
  107. data/vendor/spidermonkey/fdlibm/k_tan.c +170 -0
  108. data/vendor/spidermonkey/fdlibm/s_asinh.c +101 -0
  109. data/vendor/spidermonkey/fdlibm/s_atan.c +175 -0
  110. data/vendor/spidermonkey/fdlibm/s_cbrt.c +133 -0
  111. data/vendor/spidermonkey/fdlibm/s_ceil.c +120 -0
  112. data/vendor/spidermonkey/fdlibm/s_copysign.c +72 -0
  113. data/vendor/spidermonkey/fdlibm/s_cos.c +118 -0
  114. data/vendor/spidermonkey/fdlibm/s_erf.c +356 -0
  115. data/vendor/spidermonkey/fdlibm/s_expm1.c +267 -0
  116. data/vendor/spidermonkey/fdlibm/s_fabs.c +70 -0
  117. data/vendor/spidermonkey/fdlibm/s_finite.c +71 -0
  118. data/vendor/spidermonkey/fdlibm/s_floor.c +121 -0
  119. data/vendor/spidermonkey/fdlibm/s_frexp.c +99 -0
  120. data/vendor/spidermonkey/fdlibm/s_ilogb.c +85 -0
  121. data/vendor/spidermonkey/fdlibm/s_isnan.c +74 -0
  122. data/vendor/spidermonkey/fdlibm/s_ldexp.c +66 -0
  123. data/vendor/spidermonkey/fdlibm/s_lib_version.c +73 -0
  124. data/vendor/spidermonkey/fdlibm/s_log1p.c +211 -0
  125. data/vendor/spidermonkey/fdlibm/s_logb.c +79 -0
  126. data/vendor/spidermonkey/fdlibm/s_matherr.c +64 -0
  127. data/vendor/spidermonkey/fdlibm/s_modf.c +132 -0
  128. data/vendor/spidermonkey/fdlibm/s_nextafter.c +124 -0
  129. data/vendor/spidermonkey/fdlibm/s_rint.c +131 -0
  130. data/vendor/spidermonkey/fdlibm/s_scalbn.c +107 -0
  131. data/vendor/spidermonkey/fdlibm/s_signgam.c +40 -0
  132. data/vendor/spidermonkey/fdlibm/s_significand.c +68 -0
  133. data/vendor/spidermonkey/fdlibm/s_sin.c +118 -0
  134. data/vendor/spidermonkey/fdlibm/s_tan.c +112 -0
  135. data/vendor/spidermonkey/fdlibm/s_tanh.c +122 -0
  136. data/vendor/spidermonkey/fdlibm/w_acos.c +78 -0
  137. data/vendor/spidermonkey/fdlibm/w_acosh.c +78 -0
  138. data/vendor/spidermonkey/fdlibm/w_asin.c +80 -0
  139. data/vendor/spidermonkey/fdlibm/w_atan2.c +79 -0
  140. data/vendor/spidermonkey/fdlibm/w_atanh.c +81 -0
  141. data/vendor/spidermonkey/fdlibm/w_cosh.c +77 -0
  142. data/vendor/spidermonkey/fdlibm/w_exp.c +88 -0
  143. data/vendor/spidermonkey/fdlibm/w_fmod.c +78 -0
  144. data/vendor/spidermonkey/fdlibm/w_gamma.c +85 -0
  145. data/vendor/spidermonkey/fdlibm/w_gamma_r.c +81 -0
  146. data/vendor/spidermonkey/fdlibm/w_hypot.c +78 -0
  147. data/vendor/spidermonkey/fdlibm/w_j0.c +105 -0
  148. data/vendor/spidermonkey/fdlibm/w_j1.c +106 -0
  149. data/vendor/spidermonkey/fdlibm/w_jn.c +128 -0
  150. data/vendor/spidermonkey/fdlibm/w_lgamma.c +85 -0
  151. data/vendor/spidermonkey/fdlibm/w_lgamma_r.c +81 -0
  152. data/vendor/spidermonkey/fdlibm/w_log.c +78 -0
  153. data/vendor/spidermonkey/fdlibm/w_log10.c +81 -0
  154. data/vendor/spidermonkey/fdlibm/w_pow.c +99 -0
  155. data/vendor/spidermonkey/fdlibm/w_remainder.c +77 -0
  156. data/vendor/spidermonkey/fdlibm/w_scalb.c +95 -0
  157. data/vendor/spidermonkey/fdlibm/w_sinh.c +77 -0
  158. data/vendor/spidermonkey/fdlibm/w_sqrt.c +77 -0
  159. data/vendor/spidermonkey/javascript-trace.d +73 -0
  160. data/vendor/spidermonkey/js.c +3951 -0
  161. data/vendor/spidermonkey/js.mak +4438 -0
  162. data/vendor/spidermonkey/js.mdp +0 -0
  163. data/vendor/spidermonkey/js.msg +307 -0
  164. data/vendor/spidermonkey/js.pkg +2 -0
  165. data/vendor/spidermonkey/js3240.rc +79 -0
  166. data/vendor/spidermonkey/jsOS240.def +654 -0
  167. data/vendor/spidermonkey/jsapi.c +5836 -0
  168. data/vendor/spidermonkey/jsapi.h +2624 -0
  169. data/vendor/spidermonkey/jsarena.c +450 -0
  170. data/vendor/spidermonkey/jsarena.h +318 -0
  171. data/vendor/spidermonkey/jsarray.c +2988 -0
  172. data/vendor/spidermonkey/jsarray.h +124 -0
  173. data/vendor/spidermonkey/jsatom.c +1045 -0
  174. data/vendor/spidermonkey/jsatom.h +442 -0
  175. data/vendor/spidermonkey/jsbit.h +253 -0
  176. data/vendor/spidermonkey/jsbool.c +176 -0
  177. data/vendor/spidermonkey/jsbool.h +73 -0
  178. data/vendor/spidermonkey/jsclist.h +139 -0
  179. data/vendor/spidermonkey/jscntxt.c +1348 -0
  180. data/vendor/spidermonkey/jscntxt.h +1120 -0
  181. data/vendor/spidermonkey/jscompat.h +57 -0
  182. data/vendor/spidermonkey/jsconfig.h +248 -0
  183. data/vendor/spidermonkey/jsconfig.mk +181 -0
  184. data/vendor/spidermonkey/jscpucfg.c +383 -0
  185. data/vendor/spidermonkey/jscpucfg.h +212 -0
  186. data/vendor/spidermonkey/jsdate.c +2398 -0
  187. data/vendor/spidermonkey/jsdate.h +124 -0
  188. data/vendor/spidermonkey/jsdbgapi.c +1799 -0
  189. data/vendor/spidermonkey/jsdbgapi.h +464 -0
  190. data/vendor/spidermonkey/jsdhash.c +868 -0
  191. data/vendor/spidermonkey/jsdhash.h +592 -0
  192. data/vendor/spidermonkey/jsdtoa.c +3167 -0
  193. data/vendor/spidermonkey/jsdtoa.h +130 -0
  194. data/vendor/spidermonkey/jsdtracef.c +317 -0
  195. data/vendor/spidermonkey/jsdtracef.h +77 -0
  196. data/vendor/spidermonkey/jsemit.c +6909 -0
  197. data/vendor/spidermonkey/jsemit.h +741 -0
  198. data/vendor/spidermonkey/jsexn.c +1371 -0
  199. data/vendor/spidermonkey/jsexn.h +96 -0
  200. data/vendor/spidermonkey/jsfile.c +2736 -0
  201. data/vendor/spidermonkey/jsfile.h +56 -0
  202. data/vendor/spidermonkey/jsfile.msg +90 -0
  203. data/vendor/spidermonkey/jsfun.c +2634 -0
  204. data/vendor/spidermonkey/jsfun.h +254 -0
  205. data/vendor/spidermonkey/jsgc.c +3554 -0
  206. data/vendor/spidermonkey/jsgc.h +403 -0
  207. data/vendor/spidermonkey/jshash.c +476 -0
  208. data/vendor/spidermonkey/jshash.h +151 -0
  209. data/vendor/spidermonkey/jsify.pl +485 -0
  210. data/vendor/spidermonkey/jsinterp.c +6981 -0
  211. data/vendor/spidermonkey/jsinterp.h +521 -0
  212. data/vendor/spidermonkey/jsinvoke.c +43 -0
  213. data/vendor/spidermonkey/jsiter.c +1067 -0
  214. data/vendor/spidermonkey/jsiter.h +122 -0
  215. data/vendor/spidermonkey/jskeyword.tbl +124 -0
  216. data/vendor/spidermonkey/jskwgen.c +460 -0
  217. data/vendor/spidermonkey/jslibmath.h +266 -0
  218. data/vendor/spidermonkey/jslock.c +1309 -0
  219. data/vendor/spidermonkey/jslock.h +313 -0
  220. data/vendor/spidermonkey/jslocko.asm +60 -0
  221. data/vendor/spidermonkey/jslog2.c +94 -0
  222. data/vendor/spidermonkey/jslong.c +264 -0
  223. data/vendor/spidermonkey/jslong.h +412 -0
  224. data/vendor/spidermonkey/jsmath.c +568 -0
  225. data/vendor/spidermonkey/jsmath.h +57 -0
  226. data/vendor/spidermonkey/jsnum.c +1228 -0
  227. data/vendor/spidermonkey/jsnum.h +283 -0
  228. data/vendor/spidermonkey/jsobj.c +5266 -0
  229. data/vendor/spidermonkey/jsobj.h +709 -0
  230. data/vendor/spidermonkey/jsopcode.c +5245 -0
  231. data/vendor/spidermonkey/jsopcode.h +394 -0
  232. data/vendor/spidermonkey/jsopcode.tbl +523 -0
  233. data/vendor/spidermonkey/jsotypes.h +202 -0
  234. data/vendor/spidermonkey/jsparse.c +6680 -0
  235. data/vendor/spidermonkey/jsparse.h +511 -0
  236. data/vendor/spidermonkey/jsprf.c +1262 -0
  237. data/vendor/spidermonkey/jsprf.h +150 -0
  238. data/vendor/spidermonkey/jsproto.tbl +128 -0
  239. data/vendor/spidermonkey/jsprvtd.h +267 -0
  240. data/vendor/spidermonkey/jspubtd.h +744 -0
  241. data/vendor/spidermonkey/jsregexp.c +4352 -0
  242. data/vendor/spidermonkey/jsregexp.h +183 -0
  243. data/vendor/spidermonkey/jsreops.tbl +145 -0
  244. data/vendor/spidermonkey/jsscan.c +2003 -0
  245. data/vendor/spidermonkey/jsscan.h +387 -0
  246. data/vendor/spidermonkey/jsscope.c +1948 -0
  247. data/vendor/spidermonkey/jsscope.h +418 -0
  248. data/vendor/spidermonkey/jsscript.c +1832 -0
  249. data/vendor/spidermonkey/jsscript.h +287 -0
  250. data/vendor/spidermonkey/jsshell.msg +50 -0
  251. data/vendor/spidermonkey/jsstddef.h +83 -0
  252. data/vendor/spidermonkey/jsstr.c +5004 -0
  253. data/vendor/spidermonkey/jsstr.h +641 -0
  254. data/vendor/spidermonkey/jstypes.h +475 -0
  255. data/vendor/spidermonkey/jsutil.c +345 -0
  256. data/vendor/spidermonkey/jsutil.h +157 -0
  257. data/vendor/spidermonkey/jsxdrapi.c +800 -0
  258. data/vendor/spidermonkey/jsxdrapi.h +218 -0
  259. data/vendor/spidermonkey/jsxml.c +8471 -0
  260. data/vendor/spidermonkey/jsxml.h +349 -0
  261. data/vendor/spidermonkey/lock_SunOS.s +119 -0
  262. data/vendor/spidermonkey/perfect.js +39 -0
  263. data/vendor/spidermonkey/plify_jsdhash.sed +36 -0
  264. data/vendor/spidermonkey/prmjtime.c +846 -0
  265. data/vendor/spidermonkey/prmjtime.h +103 -0
  266. data/vendor/spidermonkey/resource.h +15 -0
  267. data/vendor/spidermonkey/rules.mk +197 -0
  268. data/vendor/spidermonkey/win32.order +384 -0
  269. metadata +4 -3
@@ -0,0 +1,418 @@
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
+ #ifndef jsscope_h___
42
+ #define jsscope_h___
43
+ /*
44
+ * JS symbol tables.
45
+ */
46
+ #include "jstypes.h"
47
+ #include "jslock.h"
48
+ #include "jsobj.h"
49
+ #include "jsprvtd.h"
50
+ #include "jspubtd.h"
51
+
52
+ JS_BEGIN_EXTERN_C
53
+
54
+ /*
55
+ * Given P independent, non-unique properties each of size S words mapped by
56
+ * all scopes in a runtime, construct a property tree of N nodes each of size
57
+ * S+L words (L for tree linkage). A nominal L value is 2 for leftmost-child
58
+ * and right-sibling links. We hope that the N < P by enough that the space
59
+ * overhead of L, and the overhead of scope entries pointing at property tree
60
+ * nodes, is worth it.
61
+ *
62
+ * The tree construction goes as follows. If any empty scope in the runtime
63
+ * has a property X added to it, find or create a node under the tree root
64
+ * labeled X, and set scope->lastProp to point at that node. If any non-empty
65
+ * scope whose most recently added property is labeled Y has another property
66
+ * labeled Z added, find or create a node for Z under the node that was added
67
+ * for Y, and set scope->lastProp to point at that node.
68
+ *
69
+ * A property is labeled by its members' values: id, getter, setter, slot,
70
+ * attributes, tiny or short id, and a field telling for..in order. Note that
71
+ * labels are not unique in the tree, but they are unique among a node's kids
72
+ * (barring rare and benign multi-threaded race condition outcomes, see below)
73
+ * and along any ancestor line from the tree root to a given leaf node (except
74
+ * for the hard case of duplicate formal parameters to a function).
75
+ *
76
+ * Thus the root of the tree represents all empty scopes, and the first ply
77
+ * of the tree represents all scopes containing one property, etc. Each node
78
+ * in the tree can stand for any number of scopes having the same ordered set
79
+ * of properties, where that node was the last added to the scope. (We need
80
+ * not store the root of the tree as a node, and do not -- all we need are
81
+ * links to its kids.)
82
+ *
83
+ * Sidebar on for..in loop order: ECMA requires no particular order, but this
84
+ * implementation has promised and delivered property definition order, and
85
+ * compatibility is king. We could use an order number per property, which
86
+ * would require a sort in js_Enumerate, and an entry order generation number
87
+ * per scope. An order number beats a list, which should be doubly-linked for
88
+ * O(1) delete. An even better scheme is to use a parent link in the property
89
+ * tree, so that the ancestor line can be iterated from scope->lastProp when
90
+ * filling in a JSIdArray from back to front. This parent link also helps the
91
+ * GC to sweep properties iteratively.
92
+ *
93
+ * What if a property Y is deleted from a scope? If Y is the last property in
94
+ * the scope, we simply adjust the scope's lastProp member after we remove the
95
+ * scope's hash-table entry pointing at that property node. The parent link
96
+ * mentioned in the for..in sidebar above makes this adjustment O(1). But if
97
+ * Y comes between X and Z in the scope, then we might have to "fork" the tree
98
+ * at X, leaving X->Y->Z in case other scopes have those properties added in
99
+ * that order; and to finish the fork, we'd add a node labeled Z with the path
100
+ * X->Z, if it doesn't exist. This could lead to lots of extra nodes, and to
101
+ * O(n^2) growth when deleting lots of properties.
102
+ *
103
+ * Rather, for O(1) growth all around, we should share the path X->Y->Z among
104
+ * scopes having those three properties added in that order, and among scopes
105
+ * having only X->Z where Y was deleted. All such scopes have a lastProp that
106
+ * points to the Z child of Y. But a scope in which Y was deleted does not
107
+ * have a table entry for Y, and when iterating that scope by traversing the
108
+ * ancestor line from Z, we will have to test for a table entry for each node,
109
+ * skipping nodes that lack entries.
110
+ *
111
+ * What if we add Y again? X->Y->Z->Y is wrong and we'll enumerate Y twice.
112
+ * Therefore we must fork in such a case, if not earlier. Because delete is
113
+ * "bursty", we should not fork eagerly. Delaying a fork till we are at risk
114
+ * of adding Y after it was deleted already requires a flag in the JSScope, to
115
+ * wit, SCOPE_MIDDLE_DELETE.
116
+ *
117
+ * What about thread safety? If the property tree operations done by requests
118
+ * are find-node and insert-node, then the only hazard is duplicate insertion.
119
+ * This is harmless except for minor bloat. When all requests have ended or
120
+ * been suspended, the GC is free to sweep the tree after marking all nodes
121
+ * reachable from scopes, performing remove-node operations as needed.
122
+ *
123
+ * Is the property tree worth it compared to property storage in each table's
124
+ * entries? To decide, we must find the relation <> between the words used
125
+ * with a property tree and the words required without a tree.
126
+ *
127
+ * Model all scopes as one super-scope of capacity T entries (T a power of 2).
128
+ * Let alpha be the load factor of this double hash-table. With the property
129
+ * tree, each entry in the table is a word-sized pointer to a node that can be
130
+ * shared by many scopes. But all such pointers are overhead compared to the
131
+ * situation without the property tree, where the table stores property nodes
132
+ * directly, as entries each of size S words. With the property tree, we need
133
+ * L=2 extra words per node for siblings and kids pointers. Without the tree,
134
+ * (1-alpha)*S*T words are wasted on free or removed sentinel-entries required
135
+ * by double hashing.
136
+ *
137
+ * Therefore,
138
+ *
139
+ * (property tree) <> (no property tree)
140
+ * N*(S+L) + T <> S*T
141
+ * N*(S+L) + T <> P*S + (1-alpha)*S*T
142
+ * N*(S+L) + alpha*T + (1-alpha)*T <> P*S + (1-alpha)*S*T
143
+ *
144
+ * Note that P is alpha*T by definition, so
145
+ *
146
+ * N*(S+L) + P + (1-alpha)*T <> P*S + (1-alpha)*S*T
147
+ * N*(S+L) <> P*S - P + (1-alpha)*S*T - (1-alpha)*T
148
+ * N*(S+L) <> (P + (1-alpha)*T) * (S-1)
149
+ * N*(S+L) <> (P + (1-alpha)*P/alpha) * (S-1)
150
+ * N*(S+L) <> P * (1/alpha) * (S-1)
151
+ *
152
+ * Let N = P*beta for a compression ratio beta, beta <= 1:
153
+ *
154
+ * P*beta*(S+L) <> P * (1/alpha) * (S-1)
155
+ * beta*(S+L) <> (S-1)/alpha
156
+ * beta <> (S-1)/((S+L)*alpha)
157
+ *
158
+ * For S = 6 (32-bit architectures) and L = 2, the property tree wins iff
159
+ *
160
+ * beta < 5/(8*alpha)
161
+ *
162
+ * We ensure that alpha <= .75, so the property tree wins if beta < .83_. An
163
+ * average beta from recent Mozilla browser startups was around .6.
164
+ *
165
+ * Can we reduce L? Observe that the property tree degenerates into a list of
166
+ * lists if at most one property Y follows X in all scopes. In or near such a
167
+ * case, we waste a word on the right-sibling link outside of the root ply of
168
+ * the tree. Note also that the root ply tends to be large, so O(n^2) growth
169
+ * searching it is likely, indicating the need for hashing (but with increased
170
+ * thread safety costs).
171
+ *
172
+ * If only K out of N nodes in the property tree have more than one child, we
173
+ * could eliminate the sibling link and overlay a children list or hash-table
174
+ * pointer on the leftmost-child link (which would then be either null or an
175
+ * only-child link; the overlay could be tagged in the low bit of the pointer,
176
+ * or flagged elsewhere in the property tree node, although such a flag must
177
+ * not be considered when comparing node labels during tree search).
178
+ *
179
+ * For such a system, L = 1 + (K * averageChildrenTableSize) / N instead of 2.
180
+ * If K << N, L approaches 1 and the property tree wins if beta < .95.
181
+ *
182
+ * We observe that fan-out below the root ply of the property tree appears to
183
+ * have extremely low degree (see the MeterPropertyTree code that histograms
184
+ * child-counts in jsscope.c), so instead of a hash-table we use a linked list
185
+ * of child node pointer arrays ("kid chunks"). The details are isolated in
186
+ * jsscope.c; others must treat JSScopeProperty.kids as opaque. We leave it
187
+ * strongly typed for debug-ability of the common (null or one-kid) cases.
188
+ *
189
+ * One final twist (can you stand it?): the mean number of entries per scope
190
+ * in Mozilla is < 5, with a large standard deviation (~8). Instead of always
191
+ * allocating scope->table, we leave it null while initializing all the other
192
+ * scope members as if it were non-null and minimal-length. Until a property
193
+ * is added that crosses the threshold of 6 or more entries for hashing, or
194
+ * until a "middle delete" occurs, we use linear search from scope->lastProp
195
+ * to find a given id, and save on the space overhead of a hash table.
196
+ */
197
+
198
+ struct JSScope {
199
+ JSObjectMap map; /* base class state */
200
+ #ifdef JS_THREADSAFE
201
+ JSTitle title; /* lock state */
202
+ #endif
203
+ JSObject *object; /* object that owns this scope */
204
+ uint32 shape; /* property cache shape identifier */
205
+ uint8 flags; /* flags, see below */
206
+ int8 hashShift; /* multiplicative hash shift */
207
+ uint16 spare; /* reserved */
208
+ uint32 entryCount; /* number of entries in table */
209
+ uint32 removedCount; /* removed entry sentinels in table */
210
+ JSScopeProperty **table; /* table of ptrs to shared tree nodes */
211
+ JSScopeProperty *lastProp; /* pointer to last property added */
212
+ };
213
+
214
+ #ifdef JS_THREADSAFE
215
+ JS_STATIC_ASSERT(offsetof(JSScope, title) == sizeof(JSObjectMap));
216
+ #endif
217
+
218
+ #define JS_IS_SCOPE_LOCKED(cx, scope) JS_IS_TITLE_LOCKED(cx, &(scope)->title)
219
+
220
+ #define OBJ_SCOPE(obj) ((JSScope *)(obj)->map)
221
+
222
+ #define SCOPE_MAKE_UNIQUE_SHAPE(cx,scope) \
223
+ ((scope)->shape = js_GenerateShape((cx), JS_FALSE))
224
+
225
+ #define SCOPE_EXTEND_SHAPE(cx,scope,sprop) \
226
+ JS_BEGIN_MACRO \
227
+ if (!(scope)->lastProp || \
228
+ (scope)->shape == (scope)->lastProp->shape) { \
229
+ (scope)->shape = (sprop)->shape; \
230
+ } else { \
231
+ (scope)->shape = js_GenerateShape((cx), JS_FALSE); \
232
+ } \
233
+ JS_END_MACRO
234
+
235
+ /* By definition, hashShift = JS_DHASH_BITS - log2(capacity). */
236
+ #define SCOPE_CAPACITY(scope) JS_BIT(JS_DHASH_BITS-(scope)->hashShift)
237
+
238
+ /* Scope flags and some macros to hide them from other files than jsscope.c. */
239
+ #define SCOPE_MIDDLE_DELETE 0x0001
240
+ #define SCOPE_SEALED 0x0002
241
+ #define SCOPE_BRANDED 0x0004
242
+
243
+ #define SCOPE_HAD_MIDDLE_DELETE(scope) ((scope)->flags & SCOPE_MIDDLE_DELETE)
244
+ #define SCOPE_SET_MIDDLE_DELETE(scope) ((scope)->flags |= SCOPE_MIDDLE_DELETE)
245
+ #define SCOPE_CLR_MIDDLE_DELETE(scope) ((scope)->flags &= ~SCOPE_MIDDLE_DELETE)
246
+
247
+ #define SCOPE_IS_SEALED(scope) ((scope)->flags & SCOPE_SEALED)
248
+ #define SCOPE_SET_SEALED(scope) ((scope)->flags |= SCOPE_SEALED)
249
+ #if 0
250
+ /*
251
+ * Don't define this, it can't be done safely because JS_LOCK_OBJ will avoid
252
+ * taking the lock if the object owns its scope and the scope is sealed.
253
+ */
254
+ #undef SCOPE_CLR_SEALED(scope) ((scope)->flags &= ~SCOPE_SEALED)
255
+ #endif
256
+
257
+ /*
258
+ * A branded scope's object contains plain old methods (function-valued
259
+ * properties without magic getters and setters), and its scope->shape
260
+ * evolves whenever a function value changes.
261
+ */
262
+ #define SCOPE_IS_BRANDED(scope) ((scope)->flags & SCOPE_BRANDED)
263
+ #define SCOPE_SET_BRANDED(scope) ((scope)->flags |= SCOPE_BRANDED)
264
+ #define SCOPE_CLR_BRANDED(scope) ((scope)->flags &= ~SCOPE_BRANDED)
265
+
266
+ /*
267
+ * A little information hiding for scope->lastProp, in case it ever becomes
268
+ * a tagged pointer again.
269
+ */
270
+ #define SCOPE_LAST_PROP(scope) ((scope)->lastProp)
271
+ #define SCOPE_REMOVE_LAST_PROP(scope) ((scope)->lastProp = \
272
+ (scope)->lastProp->parent)
273
+
274
+ struct JSScopeProperty {
275
+ jsid id; /* int-tagged jsval/untagged JSAtom* */
276
+ JSPropertyOp getter; /* getter and setter hooks or objects */
277
+ JSPropertyOp setter;
278
+ uint32 slot; /* abstract index in object slots */
279
+ uint8 attrs; /* attributes, see jsapi.h JSPROP_* */
280
+ uint8 flags; /* flags, see below for defines */
281
+ int16 shortid; /* tinyid, or local arg/var index */
282
+ JSScopeProperty *parent; /* parent node, reverse for..in order */
283
+ JSScopeProperty *kids; /* null, single child, or a tagged ptr
284
+ to many-kids data structure */
285
+ uint32 shape; /* property cache shape identifier */
286
+ };
287
+
288
+ /* JSScopeProperty pointer tag bit indicating a collision. */
289
+ #define SPROP_COLLISION ((jsuword)1)
290
+ #define SPROP_REMOVED ((JSScopeProperty *) SPROP_COLLISION)
291
+
292
+ /* Macros to get and set sprop pointer values and collision flags. */
293
+ #define SPROP_IS_FREE(sprop) ((sprop) == NULL)
294
+ #define SPROP_IS_REMOVED(sprop) ((sprop) == SPROP_REMOVED)
295
+ #define SPROP_IS_LIVE(sprop) ((sprop) > SPROP_REMOVED)
296
+ #define SPROP_FLAG_COLLISION(spp,sprop) (*(spp) = (JSScopeProperty *) \
297
+ ((jsuword)(sprop) | SPROP_COLLISION))
298
+ #define SPROP_HAD_COLLISION(sprop) ((jsuword)(sprop) & SPROP_COLLISION)
299
+ #define SPROP_FETCH(spp) SPROP_CLEAR_COLLISION(*(spp))
300
+
301
+ #define SPROP_CLEAR_COLLISION(sprop) \
302
+ ((JSScopeProperty *) ((jsuword)(sprop) & ~SPROP_COLLISION))
303
+
304
+ #define SPROP_STORE_PRESERVING_COLLISION(spp, sprop) \
305
+ (*(spp) = (JSScopeProperty *) ((jsuword)(sprop) \
306
+ | SPROP_HAD_COLLISION(*(spp))))
307
+
308
+ /* Bits stored in sprop->flags. */
309
+ #define SPROP_MARK 0x01
310
+ #define SPROP_IS_ALIAS 0x02
311
+ #define SPROP_HAS_SHORTID 0x04
312
+ #define SPROP_FLAG_SHAPE_REGEN 0x08
313
+
314
+ /*
315
+ * If SPROP_HAS_SHORTID is set in sprop->flags, we use sprop->shortid rather
316
+ * than id when calling sprop's getter or setter.
317
+ */
318
+ #define SPROP_USERID(sprop) \
319
+ (((sprop)->flags & SPROP_HAS_SHORTID) ? INT_TO_JSVAL((sprop)->shortid) \
320
+ : ID_TO_VALUE((sprop)->id))
321
+
322
+ #define SPROP_INVALID_SLOT 0xffffffff
323
+
324
+ #define SLOT_IN_SCOPE(slot,scope) ((slot) < (scope)->map.freeslot)
325
+ #define SPROP_HAS_VALID_SLOT(sprop,scope) SLOT_IN_SCOPE((sprop)->slot, scope)
326
+
327
+ #define SPROP_HAS_STUB_GETTER(sprop) (!(sprop)->getter)
328
+ #define SPROP_HAS_STUB_SETTER(sprop) (!(sprop)->setter)
329
+
330
+ /*
331
+ * NB: SPROP_GET must not be called if SPROP_HAS_STUB_GETTER(sprop).
332
+ */
333
+ #define SPROP_GET(cx,sprop,obj,obj2,vp) \
334
+ (((sprop)->attrs & JSPROP_GETTER) \
335
+ ? js_InternalGetOrSet(cx, obj, (sprop)->id, \
336
+ OBJECT_TO_JSVAL((sprop)->getter), JSACC_READ, \
337
+ 0, 0, vp) \
338
+ : (sprop)->getter(cx, OBJ_THIS_OBJECT(cx,obj), SPROP_USERID(sprop), vp))
339
+
340
+ /*
341
+ * NB: SPROP_SET must not be called if (SPROP_HAS_STUB_SETTER(sprop) &&
342
+ * !(sprop->attrs & JSPROP_GETTER)).
343
+ */
344
+ #define SPROP_SET(cx,sprop,obj,obj2,vp) \
345
+ (((sprop)->attrs & JSPROP_SETTER) \
346
+ ? js_InternalGetOrSet(cx, obj, (sprop)->id, \
347
+ OBJECT_TO_JSVAL((sprop)->setter), JSACC_WRITE, \
348
+ 1, vp, vp) \
349
+ : ((sprop)->attrs & JSPROP_GETTER) \
350
+ ? (JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, \
351
+ JSMSG_GETTER_ONLY, NULL), JS_FALSE) \
352
+ : (sprop)->setter(cx, OBJ_THIS_OBJECT(cx,obj), SPROP_USERID(sprop), vp))
353
+
354
+ /* Macro for common expression to test for shared permanent attributes. */
355
+ #define SPROP_IS_SHARED_PERMANENT(sprop) \
356
+ ((~(sprop)->attrs & (JSPROP_SHARED | JSPROP_PERMANENT)) == 0)
357
+
358
+ extern JSScope *
359
+ js_GetMutableScope(JSContext *cx, JSObject *obj);
360
+
361
+ extern JSScope *
362
+ js_NewScope(JSContext *cx, jsrefcount nrefs, JSObjectOps *ops, JSClass *clasp,
363
+ JSObject *obj);
364
+
365
+ extern void
366
+ js_DestroyScope(JSContext *cx, JSScope *scope);
367
+
368
+ extern JS_FRIEND_API(JSScopeProperty **)
369
+ js_SearchScope(JSScope *scope, jsid id, JSBool adding);
370
+
371
+ #define SCOPE_GET_PROPERTY(scope, id) \
372
+ SPROP_FETCH(js_SearchScope(scope, id, JS_FALSE))
373
+
374
+ #define SCOPE_HAS_PROPERTY(scope, sprop) \
375
+ (SCOPE_GET_PROPERTY(scope, (sprop)->id) == (sprop))
376
+
377
+ extern JSScopeProperty *
378
+ js_AddScopeProperty(JSContext *cx, JSScope *scope, jsid id,
379
+ JSPropertyOp getter, JSPropertyOp setter, uint32 slot,
380
+ uintN attrs, uintN flags, intN shortid);
381
+
382
+ extern JSScopeProperty *
383
+ js_ChangeScopePropertyAttrs(JSContext *cx, JSScope *scope,
384
+ JSScopeProperty *sprop, uintN attrs, uintN mask,
385
+ JSPropertyOp getter, JSPropertyOp setter);
386
+
387
+ extern JSBool
388
+ js_RemoveScopeProperty(JSContext *cx, JSScope *scope, jsid id);
389
+
390
+ extern void
391
+ js_ClearScope(JSContext *cx, JSScope *scope);
392
+
393
+ /*
394
+ * These macros used to inline short code sequences, but they grew over time.
395
+ * We retain them for internal backward compatibility, and in case one or both
396
+ * ever shrink to inline-able size.
397
+ */
398
+ #define TRACE_ID(trc, id) js_TraceId(trc, id)
399
+ #define TRACE_SCOPE_PROPERTY(trc, sprop) js_TraceScopeProperty(trc, sprop)
400
+
401
+ extern void
402
+ js_TraceId(JSTracer *trc, jsid id);
403
+
404
+ extern void
405
+ js_TraceScopeProperty(JSTracer *trc, JSScopeProperty *sprop);
406
+
407
+ extern void
408
+ js_SweepScopeProperties(JSContext *cx);
409
+
410
+ extern JSBool
411
+ js_InitPropertyTree(JSRuntime *rt);
412
+
413
+ extern void
414
+ js_FinishPropertyTree(JSRuntime *rt);
415
+
416
+ JS_END_EXTERN_C
417
+
418
+ #endif /* jsscope_h___ */
@@ -0,0 +1,1832 @@
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 script operations.
43
+ */
44
+ #include "jsstddef.h"
45
+ #include <string.h>
46
+ #include "jstypes.h"
47
+ #include "jsutil.h" /* Added by JSIFY */
48
+ #include "jsprf.h"
49
+ #include "jsapi.h"
50
+ #include "jsatom.h"
51
+ #include "jscntxt.h"
52
+ #include "jsconfig.h"
53
+ #include "jsdbgapi.h"
54
+ #include "jsemit.h"
55
+ #include "jsfun.h"
56
+ #include "jsinterp.h"
57
+ #include "jslock.h"
58
+ #include "jsnum.h"
59
+ #include "jsopcode.h"
60
+ #include "jsparse.h"
61
+ #include "jsscript.h"
62
+ #if JS_HAS_XDR
63
+ #include "jsxdrapi.h"
64
+ #endif
65
+
66
+ #if JS_HAS_SCRIPT_OBJECT
67
+
68
+ static const char js_script_exec_str[] = "Script.prototype.exec";
69
+ static const char js_script_compile_str[] = "Script.prototype.compile";
70
+
71
+ /*
72
+ * This routine requires that obj has been locked previously.
73
+ */
74
+ static jsint
75
+ GetScriptExecDepth(JSContext *cx, JSObject *obj)
76
+ {
77
+ jsval v;
78
+
79
+ JS_ASSERT(JS_IS_OBJ_LOCKED(cx, obj));
80
+ v = LOCKED_OBJ_GET_SLOT(obj, JSSLOT_START(&js_ScriptClass));
81
+ return JSVAL_TO_INT(v);
82
+ }
83
+
84
+ static void
85
+ AdjustScriptExecDepth(JSContext *cx, JSObject *obj, jsint delta)
86
+ {
87
+ jsint execDepth;
88
+
89
+ JS_LOCK_OBJ(cx, obj);
90
+ execDepth = GetScriptExecDepth(cx, obj);
91
+ LOCKED_OBJ_SET_SLOT(obj, JSSLOT_START(&js_ScriptClass),
92
+ INT_TO_JSVAL(execDepth + delta));
93
+ JS_UNLOCK_OBJ(cx, obj);
94
+ }
95
+
96
+ #if JS_HAS_TOSOURCE
97
+ static JSBool
98
+ script_toSource(JSContext *cx, uintN argc, jsval *vp)
99
+ {
100
+ JSObject *obj;
101
+ uint32 indent;
102
+ JSScript *script;
103
+ size_t i, j, k, n;
104
+ char buf[16];
105
+ jschar *s, *t;
106
+ JSString *str;
107
+
108
+ obj = JS_THIS_OBJECT(cx, vp);
109
+ if (!JS_InstanceOf(cx, obj, &js_ScriptClass, vp + 2))
110
+ return JS_FALSE;
111
+
112
+ indent = 0;
113
+ if (argc != 0) {
114
+ indent = js_ValueToECMAUint32(cx, &vp[2]);
115
+ if (JSVAL_IS_NULL(vp[2]))
116
+ return JS_FALSE;
117
+ }
118
+
119
+ script = (JSScript *) JS_GetPrivate(cx, obj);
120
+
121
+ /* Let n count the source string length, j the "front porch" length. */
122
+ j = JS_snprintf(buf, sizeof buf, "(new %s(", js_ScriptClass.name);
123
+ n = j + 2;
124
+ if (!script) {
125
+ /* Let k count the constructor argument string length. */
126
+ k = 0;
127
+ s = NULL; /* quell GCC overwarning */
128
+ } else {
129
+ str = JS_DecompileScript(cx, script, "Script.prototype.toSource",
130
+ (uintN)indent);
131
+ if (!str)
132
+ return JS_FALSE;
133
+ str = js_QuoteString(cx, str, '\'');
134
+ if (!str)
135
+ return JS_FALSE;
136
+ JSSTRING_CHARS_AND_LENGTH(str, s, k);
137
+ n += k;
138
+ }
139
+
140
+ /* Allocate the source string and copy into it. */
141
+ t = (jschar *) JS_malloc(cx, (n + 1) * sizeof(jschar));
142
+ if (!t)
143
+ return JS_FALSE;
144
+ for (i = 0; i < j; i++)
145
+ t[i] = buf[i];
146
+ for (j = 0; j < k; i++, j++)
147
+ t[i] = s[j];
148
+ t[i++] = ')';
149
+ t[i++] = ')';
150
+ t[i] = 0;
151
+
152
+ /* Create and return a JS string for t. */
153
+ str = JS_NewUCString(cx, t, n);
154
+ if (!str) {
155
+ JS_free(cx, t);
156
+ return JS_FALSE;
157
+ }
158
+ *vp = STRING_TO_JSVAL(str);
159
+ return JS_TRUE;
160
+ }
161
+ #endif /* JS_HAS_TOSOURCE */
162
+
163
+ static JSBool
164
+ script_toString(JSContext *cx, uintN argc, jsval *vp)
165
+ {
166
+ uint32 indent;
167
+ JSObject *obj;
168
+ JSScript *script;
169
+ JSString *str;
170
+
171
+ indent = 0;
172
+ if (argc != 0) {
173
+ indent = js_ValueToECMAUint32(cx, &vp[2]);
174
+ if (JSVAL_IS_NULL(vp[2]))
175
+ return JS_FALSE;
176
+ }
177
+
178
+ obj = JS_THIS_OBJECT(cx, vp);
179
+ if (!JS_InstanceOf(cx, obj, &js_ScriptClass, vp + 2))
180
+ return JS_FALSE;
181
+ script = (JSScript *) JS_GetPrivate(cx, obj);
182
+ if (!script) {
183
+ *vp = STRING_TO_JSVAL(cx->runtime->emptyString);
184
+ return JS_TRUE;
185
+ }
186
+
187
+ str = JS_DecompileScript(cx, script, "Script.prototype.toString",
188
+ (uintN)indent);
189
+ if (!str)
190
+ return JS_FALSE;
191
+ *vp = STRING_TO_JSVAL(str);
192
+ return JS_TRUE;
193
+ }
194
+
195
+ static JSBool
196
+ script_compile_sub(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
197
+ jsval *rval)
198
+ {
199
+ JSString *str;
200
+ JSObject *scopeobj;
201
+ jsval v;
202
+ JSScript *script, *oldscript;
203
+ JSStackFrame *fp, *caller;
204
+ const char *file;
205
+ uintN line;
206
+ JSPrincipals *principals;
207
+ uint32 tcflags;
208
+ jsint execDepth;
209
+
210
+ /* Make sure obj is a Script object. */
211
+ if (!JS_InstanceOf(cx, obj, &js_ScriptClass, argv))
212
+ return JS_FALSE;
213
+
214
+ /* If no args, leave private undefined and return early. */
215
+ if (argc == 0)
216
+ goto out;
217
+
218
+ /* Otherwise, the first arg is the script source to compile. */
219
+ str = js_ValueToString(cx, argv[0]);
220
+ if (!str)
221
+ return JS_FALSE;
222
+ argv[0] = STRING_TO_JSVAL(str);
223
+
224
+ scopeobj = NULL;
225
+ if (argc >= 2) {
226
+ if (!js_ValueToObject(cx, argv[1], &scopeobj))
227
+ return JS_FALSE;
228
+ argv[1] = OBJECT_TO_JSVAL(scopeobj);
229
+ }
230
+
231
+ /* Compile using the caller's scope chain, which js_Invoke passes to fp. */
232
+ fp = cx->fp;
233
+ caller = JS_GetScriptedCaller(cx, fp);
234
+ JS_ASSERT(!caller || fp->scopeChain == caller->scopeChain);
235
+
236
+ if (caller) {
237
+ if (!scopeobj) {
238
+ scopeobj = js_GetScopeChain(cx, caller);
239
+ if (!scopeobj)
240
+ return JS_FALSE;
241
+ fp->scopeChain = scopeobj; /* for the compiler's benefit */
242
+ }
243
+
244
+ principals = JS_EvalFramePrincipals(cx, fp, caller);
245
+ file = js_ComputeFilename(cx, caller, principals, &line);
246
+ } else {
247
+ file = NULL;
248
+ line = 0;
249
+ principals = NULL;
250
+ }
251
+
252
+ /* Ensure we compile this script with the right (inner) principals. */
253
+ scopeobj = js_CheckScopeChainValidity(cx, scopeobj, js_script_compile_str);
254
+ if (!scopeobj)
255
+ return JS_FALSE;
256
+
257
+ /*
258
+ * Compile the new script using the caller's scope chain, a la eval().
259
+ * Unlike jsobj.c:obj_eval, however, we do not pass TCF_COMPILE_N_GO in
260
+ * tcflags, because compilation is here separated from execution, and the
261
+ * run-time scope chain may not match the compile-time. JSFRAME_EVAL is
262
+ * tested in jsemit.c and jsscan.c to optimize based on identity of run-
263
+ * and compile-time scope.
264
+ */
265
+ fp->flags |= JSFRAME_SCRIPT_OBJECT;
266
+ tcflags = 0;
267
+ script = js_CompileScript(cx, scopeobj, principals, tcflags,
268
+ JSSTRING_CHARS(str), JSSTRING_LENGTH(str),
269
+ NULL, file, line);
270
+ if (!script)
271
+ return JS_FALSE;
272
+
273
+ JS_LOCK_OBJ(cx, obj);
274
+ execDepth = GetScriptExecDepth(cx, obj);
275
+
276
+ /*
277
+ * execDepth must be 0 to allow compilation here, otherwise the JSScript
278
+ * struct can be released while running.
279
+ */
280
+ if (execDepth > 0) {
281
+ JS_UNLOCK_OBJ(cx, obj);
282
+ JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
283
+ JSMSG_COMPILE_EXECED_SCRIPT);
284
+ return JS_FALSE;
285
+ }
286
+
287
+ /* Swap script for obj's old script, if any. */
288
+ v = LOCKED_OBJ_GET_SLOT(obj, JSSLOT_PRIVATE);
289
+ oldscript = !JSVAL_IS_VOID(v) ? JSVAL_TO_PRIVATE(v) : NULL;
290
+ LOCKED_OBJ_SET_SLOT(obj, JSSLOT_PRIVATE, PRIVATE_TO_JSVAL(script));
291
+ JS_UNLOCK_OBJ(cx, obj);
292
+
293
+ if (oldscript)
294
+ js_DestroyScript(cx, oldscript);
295
+
296
+ script->object = obj;
297
+ js_CallNewScriptHook(cx, script, NULL);
298
+
299
+ out:
300
+ /* Return the object. */
301
+ *rval = OBJECT_TO_JSVAL(obj);
302
+ return JS_TRUE;
303
+ }
304
+
305
+ static JSBool
306
+ script_compile(JSContext *cx, uintN argc, jsval *vp)
307
+ {
308
+ return script_compile_sub(cx, JS_THIS_OBJECT(cx, vp), argc, vp + 2, vp);
309
+ }
310
+
311
+ static JSBool
312
+ script_exec_sub(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
313
+ jsval *rval)
314
+ {
315
+ JSObject *scopeobj, *parent;
316
+ JSStackFrame *fp, *caller;
317
+ JSPrincipals *principals;
318
+ JSScript *script;
319
+ JSBool ok;
320
+
321
+ if (!JS_InstanceOf(cx, obj, &js_ScriptClass, argv))
322
+ return JS_FALSE;
323
+
324
+ scopeobj = NULL;
325
+ if (argc != 0) {
326
+ if (!js_ValueToObject(cx, argv[0], &scopeobj))
327
+ return JS_FALSE;
328
+ argv[0] = OBJECT_TO_JSVAL(scopeobj);
329
+ }
330
+
331
+ /*
332
+ * Emulate eval() by using caller's this, var object, sharp array, etc.,
333
+ * all propagated by js_Execute via a non-null fourth (down) argument to
334
+ * js_Execute. If there is no scripted caller, js_Execute uses its second
335
+ * (chain) argument to set the exec frame's varobj, thisp, and scopeChain.
336
+ *
337
+ * Unlike eval, which the compiler detects, Script.prototype.exec may be
338
+ * called from a lightweight function, or even from native code (in which
339
+ * case fp->varobj and fp->scopeChain are null). If exec is called from
340
+ * a lightweight function, we will need to get a Call object representing
341
+ * its frame, to act as the var object and scope chain head.
342
+ */
343
+ fp = cx->fp;
344
+ caller = JS_GetScriptedCaller(cx, fp);
345
+ if (caller && !caller->varobj) {
346
+ /* Called from a lightweight function. */
347
+ JS_ASSERT(caller->fun && !JSFUN_HEAVYWEIGHT_TEST(caller->fun->flags));
348
+
349
+ /* Scope chain links from Call object to callee's parent. */
350
+ parent = OBJ_GET_PARENT(cx, caller->callee);
351
+ if (!js_GetCallObject(cx, caller, parent))
352
+ return JS_FALSE;
353
+ }
354
+
355
+ if (!scopeobj) {
356
+ /* No scope object passed in: try to use the caller's scope chain. */
357
+ if (caller) {
358
+ /*
359
+ * Load caller->scopeChain after the conditional js_GetCallObject
360
+ * call above, which resets scopeChain as well as varobj.
361
+ */
362
+ scopeobj = js_GetScopeChain(cx, caller);
363
+ if (!scopeobj)
364
+ return JS_FALSE;
365
+ } else {
366
+ /*
367
+ * Called from native code, so we don't know what scope object to
368
+ * use. We could use parent (see above), but Script.prototype.exec
369
+ * might be a shared/sealed "superglobal" method. A more general
370
+ * approach would use cx->globalObject, which will be the same as
371
+ * exec.__parent__ in the non-superglobal case. In the superglobal
372
+ * case it's the right object: the global, not the superglobal.
373
+ */
374
+ scopeobj = cx->globalObject;
375
+ }
376
+ }
377
+
378
+ scopeobj = js_CheckScopeChainValidity(cx, scopeobj, js_script_exec_str);
379
+ if (!scopeobj)
380
+ return JS_FALSE;
381
+
382
+ /* Keep track of nesting depth for the script. */
383
+ AdjustScriptExecDepth(cx, obj, 1);
384
+
385
+ /* Must get to out label after this */
386
+ script = (JSScript *) JS_GetPrivate(cx, obj);
387
+ if (!script) {
388
+ ok = JS_FALSE;
389
+ goto out;
390
+ }
391
+
392
+ /* Belt-and-braces: check that this script object has access to scopeobj. */
393
+ principals = script->principals;
394
+ ok = js_CheckPrincipalsAccess(cx, scopeobj, principals,
395
+ CLASS_ATOM(cx, Script));
396
+ if (!ok)
397
+ goto out;
398
+
399
+ ok = js_Execute(cx, scopeobj, script, caller, JSFRAME_EVAL, rval);
400
+
401
+ out:
402
+ AdjustScriptExecDepth(cx, obj, -1);
403
+ return ok;
404
+ }
405
+
406
+ static JSBool
407
+ script_exec(JSContext *cx, uintN argc, jsval *vp)
408
+ {
409
+ return script_exec_sub(cx, JS_THIS_OBJECT(cx, vp), argc, vp + 2, vp);
410
+ }
411
+
412
+ #endif /* JS_HAS_SCRIPT_OBJECT */
413
+
414
+ #if JS_HAS_XDR
415
+
416
+ JSBool
417
+ js_XDRScript(JSXDRState *xdr, JSScript **scriptp, JSBool *hasMagic)
418
+ {
419
+ JSContext *cx;
420
+ JSScript *script, *oldscript;
421
+ JSBool ok;
422
+ jsbytecode *code;
423
+ uint32 length, lineno, depth, magic;
424
+ uint32 natoms, nsrcnotes, ntrynotes, nobjects, nregexps, i;
425
+ uint32 prologLength, version;
426
+ JSTempValueRooter tvr;
427
+ JSPrincipals *principals;
428
+ uint32 encodeable;
429
+ JSBool filenameWasSaved;
430
+ jssrcnote *notes, *sn;
431
+
432
+ cx = xdr->cx;
433
+ script = *scriptp;
434
+ nsrcnotes = ntrynotes = natoms = nobjects = nregexps = 0;
435
+ filenameWasSaved = JS_FALSE;
436
+ notes = NULL;
437
+
438
+ if (xdr->mode == JSXDR_ENCODE)
439
+ magic = JSXDR_MAGIC_SCRIPT_CURRENT;
440
+ if (!JS_XDRUint32(xdr, &magic))
441
+ return JS_FALSE;
442
+ if (magic != JSXDR_MAGIC_SCRIPT_CURRENT) {
443
+ /* We do not provide binary compatibility with older scripts. */
444
+ if (!hasMagic) {
445
+ JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
446
+ JSMSG_BAD_SCRIPT_MAGIC);
447
+ return JS_FALSE;
448
+ }
449
+ *hasMagic = JS_FALSE;
450
+ return JS_TRUE;
451
+ }
452
+ if (hasMagic)
453
+ *hasMagic = JS_TRUE;
454
+
455
+ if (xdr->mode == JSXDR_ENCODE) {
456
+ length = script->length;
457
+ prologLength = PTRDIFF(script->main, script->code, jsbytecode);
458
+ JS_ASSERT((int16)script->version != JSVERSION_UNKNOWN);
459
+ version = (uint32)script->version | (script->ngvars << 16);
460
+ lineno = (uint32)script->lineno;
461
+ depth = (uint32)script->depth;
462
+ natoms = (uint32)script->atomMap.length;
463
+
464
+ /* Count the srcnotes, keeping notes pointing at the first one. */
465
+ notes = SCRIPT_NOTES(script);
466
+ for (sn = notes; !SN_IS_TERMINATOR(sn); sn = SN_NEXT(sn))
467
+ continue;
468
+ nsrcnotes = PTRDIFF(sn, notes, jssrcnote);
469
+ nsrcnotes++; /* room for the terminator */
470
+
471
+ if (script->objectsOffset != 0)
472
+ nobjects = JS_SCRIPT_OBJECTS(script)->length;
473
+ if (script->regexpsOffset != 0)
474
+ nregexps = JS_SCRIPT_REGEXPS(script)->length;
475
+ if (script->trynotesOffset != 0)
476
+ ntrynotes = JS_SCRIPT_TRYNOTES(script)->length;
477
+ }
478
+
479
+ if (!JS_XDRUint32(xdr, &length))
480
+ return JS_FALSE;
481
+ if (!JS_XDRUint32(xdr, &prologLength))
482
+ return JS_FALSE;
483
+ if (!JS_XDRUint32(xdr, &version))
484
+ return JS_FALSE;
485
+
486
+ /*
487
+ * To fuse allocations, we need srcnote, atom, objects, regexp and trynote
488
+ * counts early.
489
+ */
490
+ if (!JS_XDRUint32(xdr, &natoms))
491
+ return JS_FALSE;
492
+ if (!JS_XDRUint32(xdr, &nsrcnotes))
493
+ return JS_FALSE;
494
+ if (!JS_XDRUint32(xdr, &ntrynotes))
495
+ return JS_FALSE;
496
+ if (!JS_XDRUint32(xdr, &nobjects))
497
+ return JS_FALSE;
498
+ if (!JS_XDRUint32(xdr, &nregexps))
499
+ return JS_FALSE;
500
+
501
+ if (xdr->mode == JSXDR_DECODE) {
502
+ script = js_NewScript(cx, length, nsrcnotes, natoms, nobjects, nregexps,
503
+ ntrynotes);
504
+ if (!script)
505
+ return JS_FALSE;
506
+
507
+ script->main += prologLength;
508
+ script->version = (JSVersion) (version & 0xffff);
509
+ script->ngvars = (uint16) (version >> 16);
510
+
511
+ /* If we know nsrcnotes, we allocated space for notes in script. */
512
+ notes = SCRIPT_NOTES(script);
513
+ *scriptp = script;
514
+ JS_PUSH_TEMP_ROOT_SCRIPT(cx, script, &tvr);
515
+ }
516
+
517
+ /*
518
+ * Control hereafter must goto error on failure, in order for the
519
+ * DECODE case to destroy script.
520
+ */
521
+ oldscript = xdr->script;
522
+ code = script->code;
523
+ if (xdr->mode == JSXDR_ENCODE) {
524
+ code = js_UntrapScriptCode(cx, script);
525
+ if (!code)
526
+ goto error;
527
+ }
528
+
529
+ xdr->script = script;
530
+ ok = JS_XDRBytes(xdr, (char *) code, length * sizeof(jsbytecode));
531
+
532
+ if (code != script->code)
533
+ JS_free(cx, code);
534
+
535
+ if (!ok)
536
+ goto error;
537
+
538
+ if (!JS_XDRBytes(xdr, (char *)notes, nsrcnotes * sizeof(jssrcnote)) ||
539
+ !JS_XDRCStringOrNull(xdr, (char **)&script->filename) ||
540
+ !JS_XDRUint32(xdr, &lineno) ||
541
+ !JS_XDRUint32(xdr, &depth)) {
542
+ goto error;
543
+ }
544
+
545
+ if (xdr->mode == JSXDR_ENCODE) {
546
+ principals = script->principals;
547
+ encodeable = (cx->runtime->principalsTranscoder != NULL);
548
+ if (!JS_XDRUint32(xdr, &encodeable))
549
+ goto error;
550
+ if (encodeable &&
551
+ !cx->runtime->principalsTranscoder(xdr, &principals)) {
552
+ goto error;
553
+ }
554
+ } else {
555
+ if (!JS_XDRUint32(xdr, &encodeable))
556
+ goto error;
557
+ if (encodeable) {
558
+ if (!cx->runtime->principalsTranscoder) {
559
+ JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
560
+ JSMSG_CANT_DECODE_PRINCIPALS);
561
+ goto error;
562
+ }
563
+ if (!cx->runtime->principalsTranscoder(xdr, &principals))
564
+ goto error;
565
+ script->principals = principals;
566
+ }
567
+ }
568
+
569
+ if (xdr->mode == JSXDR_DECODE) {
570
+ const char *filename = script->filename;
571
+ if (filename) {
572
+ filename = js_SaveScriptFilename(cx, filename);
573
+ if (!filename)
574
+ goto error;
575
+ JS_free(cx, (void *) script->filename);
576
+ script->filename = filename;
577
+ filenameWasSaved = JS_TRUE;
578
+ }
579
+ script->lineno = (uintN)lineno;
580
+ script->depth = (uintN)depth;
581
+ }
582
+
583
+ for (i = 0; i != natoms; ++i) {
584
+ if (!js_XDRAtom(xdr, &script->atomMap.vector[i]))
585
+ goto error;
586
+ }
587
+
588
+ /*
589
+ * Here looping from 0-to-length to xdr objects is essential. It ensures
590
+ * that block objects from the script->objects will be written and
591
+ * restored in the outer-to-inner order. block_xdrObject uses this to
592
+ * restore the parent chain.
593
+ */
594
+ for (i = 0; i != nobjects; ++i) {
595
+ if (!js_XDRObject(xdr, &JS_SCRIPT_OBJECTS(script)->vector[i]))
596
+ goto error;
597
+ }
598
+ for (i = 0; i != nregexps; ++i) {
599
+ if (!js_XDRObject(xdr, &JS_SCRIPT_REGEXPS(script)->vector[i]))
600
+ goto error;
601
+ }
602
+
603
+ if (ntrynotes != 0) {
604
+ /*
605
+ * We combine tn->kind and tn->stackDepth when serializing as XDR is not
606
+ * efficient when serializing small integer types.
607
+ */
608
+ JSTryNote *tn, *tnfirst;
609
+ uint32 kindAndDepth;
610
+ JS_STATIC_ASSERT(sizeof(tn->kind) == sizeof(uint8));
611
+ JS_STATIC_ASSERT(sizeof(tn->stackDepth) == sizeof(uint16));
612
+
613
+ tnfirst = JS_SCRIPT_TRYNOTES(script)->vector;
614
+ JS_ASSERT(JS_SCRIPT_TRYNOTES(script)->length == ntrynotes);
615
+ tn = tnfirst + ntrynotes;
616
+ do {
617
+ --tn;
618
+ if (xdr->mode == JSXDR_ENCODE) {
619
+ kindAndDepth = ((uint32)tn->kind << 16)
620
+ | (uint32)tn->stackDepth;
621
+ }
622
+ if (!JS_XDRUint32(xdr, &kindAndDepth) ||
623
+ !JS_XDRUint32(xdr, &tn->start) ||
624
+ !JS_XDRUint32(xdr, &tn->length)) {
625
+ goto error;
626
+ }
627
+ if (xdr->mode == JSXDR_DECODE) {
628
+ tn->kind = (uint8)(kindAndDepth >> 16);
629
+ tn->stackDepth = (uint16)kindAndDepth;
630
+ }
631
+ } while (tn != tnfirst);
632
+ }
633
+
634
+ xdr->script = oldscript;
635
+ if (xdr->mode == JSXDR_DECODE)
636
+ JS_POP_TEMP_ROOT(cx, &tvr);
637
+ return JS_TRUE;
638
+
639
+ error:
640
+ if (xdr->mode == JSXDR_DECODE) {
641
+ JS_POP_TEMP_ROOT(cx, &tvr);
642
+ if (script->filename && !filenameWasSaved) {
643
+ JS_free(cx, (void *) script->filename);
644
+ script->filename = NULL;
645
+ }
646
+ js_DestroyScript(cx, script);
647
+ *scriptp = NULL;
648
+ }
649
+ xdr->script = oldscript;
650
+ return JS_FALSE;
651
+ }
652
+
653
+ #if JS_HAS_SCRIPT_OBJECT && JS_HAS_XDR_FREEZE_THAW
654
+ /*
655
+ * These cannot be exposed to web content, and chrome does not need them, so
656
+ * we take them out of the Mozilla client altogether. Fortunately, there is
657
+ * no way to serialize a native function (see fun_xdrObject in jsfun.c).
658
+ */
659
+
660
+ static JSBool
661
+ script_freeze(JSContext *cx, uintN argc, jsval *vp)
662
+ {
663
+ JSObject *obj;
664
+ JSXDRState *xdr;
665
+ JSScript *script;
666
+ JSBool ok, hasMagic;
667
+ uint32 len;
668
+ void *buf;
669
+ JSString *str;
670
+
671
+ obj = JS_THIS_OBJECT(cx, vp);
672
+ if (!JS_InstanceOf(cx, obj, &js_ScriptClass, vp + 2))
673
+ return JS_FALSE;
674
+ script = (JSScript *) JS_GetPrivate(cx, obj);
675
+ if (!script)
676
+ return JS_TRUE;
677
+
678
+ /* create new XDR */
679
+ xdr = JS_XDRNewMem(cx, JSXDR_ENCODE);
680
+ if (!xdr)
681
+ return JS_FALSE;
682
+
683
+ /* write */
684
+ ok = js_XDRScript(xdr, &script, &hasMagic);
685
+ if (!ok)
686
+ goto out;
687
+ if (!hasMagic) {
688
+ *vp = JSVAL_VOID;
689
+ goto out;
690
+ }
691
+
692
+ buf = JS_XDRMemGetData(xdr, &len);
693
+ if (!buf) {
694
+ ok = JS_FALSE;
695
+ goto out;
696
+ }
697
+
698
+ JS_ASSERT((jsword)buf % sizeof(jschar) == 0);
699
+ len /= sizeof(jschar);
700
+ #if IS_BIG_ENDIAN
701
+ {
702
+ jschar *chars;
703
+ uint32 i;
704
+
705
+ /* Swap bytes in Unichars to keep frozen strings machine-independent. */
706
+ chars = (jschar *)buf;
707
+ for (i = 0; i < len; i++)
708
+ chars[i] = JSXDR_SWAB16(chars[i]);
709
+ }
710
+ #endif
711
+ str = JS_NewUCStringCopyN(cx, (jschar *)buf, len);
712
+ if (!str) {
713
+ ok = JS_FALSE;
714
+ goto out;
715
+ }
716
+
717
+ *vp = STRING_TO_JSVAL(str);
718
+
719
+ out:
720
+ JS_XDRDestroy(xdr);
721
+ return ok;
722
+ }
723
+
724
+ static JSBool
725
+ script_thaw(JSContext *cx, uintN argc, jsval *vp)
726
+ {
727
+ JSObject *obj;
728
+ JSXDRState *xdr;
729
+ JSString *str;
730
+ void *buf;
731
+ uint32 len;
732
+ jsval v;
733
+ JSScript *script, *oldscript;
734
+ JSBool ok, hasMagic;
735
+ jsint execDepth;
736
+
737
+ obj = JS_THIS_OBJECT(cx, vp);
738
+ if (!JS_InstanceOf(cx, obj, &js_ScriptClass, vp + 2))
739
+ return JS_FALSE;
740
+
741
+ if (argc == 0)
742
+ return JS_TRUE;
743
+ str = js_ValueToString(cx, vp[2]);
744
+ if (!str)
745
+ return JS_FALSE;
746
+ vp[2] = STRING_TO_JSVAL(str);
747
+
748
+ /* create new XDR */
749
+ xdr = JS_XDRNewMem(cx, JSXDR_DECODE);
750
+ if (!xdr)
751
+ return JS_FALSE;
752
+
753
+ JSSTRING_CHARS_AND_LENGTH(str, buf, len);
754
+ #if IS_BIG_ENDIAN
755
+ {
756
+ jschar *from, *to;
757
+ uint32 i;
758
+
759
+ /* Swap bytes in Unichars to keep frozen strings machine-independent. */
760
+ from = (jschar *)buf;
761
+ to = (jschar *) JS_malloc(cx, len * sizeof(jschar));
762
+ if (!to) {
763
+ JS_XDRDestroy(xdr);
764
+ return JS_FALSE;
765
+ }
766
+ for (i = 0; i < len; i++)
767
+ to[i] = JSXDR_SWAB16(from[i]);
768
+ buf = (char *)to;
769
+ }
770
+ #endif
771
+ len *= sizeof(jschar);
772
+ JS_XDRMemSetData(xdr, buf, len);
773
+
774
+ /* XXXbe should magic mismatch be error, or false return value? */
775
+ ok = js_XDRScript(xdr, &script, &hasMagic);
776
+ if (!ok)
777
+ goto out;
778
+ if (!hasMagic) {
779
+ *vp = JSVAL_FALSE;
780
+ goto out;
781
+ }
782
+
783
+ JS_LOCK_OBJ(cx, obj);
784
+ execDepth = GetScriptExecDepth(cx, obj);
785
+
786
+ /*
787
+ * execDepth must be 0 to allow compilation here, otherwise the JSScript
788
+ * struct can be released while running.
789
+ */
790
+ if (execDepth > 0) {
791
+ JS_UNLOCK_OBJ(cx, obj);
792
+ JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
793
+ JSMSG_COMPILE_EXECED_SCRIPT);
794
+ goto out;
795
+ }
796
+
797
+ /* Swap script for obj's old script, if any. */
798
+ v = LOCKED_OBJ_GET_SLOT(obj, JSSLOT_PRIVATE);
799
+ oldscript = !JSVAL_IS_VOID(v) ? JSVAL_TO_PRIVATE(v) : NULL;
800
+ LOCKED_OBJ_SET_SLOT(obj, JSSLOT_PRIVATE, PRIVATE_TO_JSVAL(script));
801
+ JS_UNLOCK_OBJ(cx, obj);
802
+
803
+ if (oldscript)
804
+ js_DestroyScript(cx, oldscript);
805
+
806
+ script->object = obj;
807
+ js_CallNewScriptHook(cx, script, NULL);
808
+
809
+ out:
810
+ /*
811
+ * We reset the buffer to be NULL so that it doesn't free the chars
812
+ * memory owned by str (vp[2]).
813
+ */
814
+ JS_XDRMemSetData(xdr, NULL, 0);
815
+ JS_XDRDestroy(xdr);
816
+ #if IS_BIG_ENDIAN
817
+ JS_free(cx, buf);
818
+ #endif
819
+ *vp = JSVAL_TRUE;
820
+ return ok;
821
+ }
822
+
823
+ static const char js_thaw_str[] = "thaw";
824
+
825
+ #endif /* JS_HAS_SCRIPT_OBJECT && JS_HAS_XDR_FREEZE_THAW */
826
+ #endif /* JS_HAS_XDR */
827
+
828
+ #if JS_HAS_SCRIPT_OBJECT
829
+
830
+ static JSFunctionSpec script_methods[] = {
831
+ #if JS_HAS_TOSOURCE
832
+ JS_FN(js_toSource_str, script_toSource, 0,0,0),
833
+ #endif
834
+ JS_FN(js_toString_str, script_toString, 0,0,0),
835
+ JS_FN("compile", script_compile, 0,2,0),
836
+ JS_FN("exec", script_exec, 0,1,0),
837
+ #if JS_HAS_XDR_FREEZE_THAW
838
+ JS_FN("freeze", script_freeze, 0,0,0),
839
+ JS_FN(js_thaw_str, script_thaw, 0,1,0),
840
+ #endif /* JS_HAS_XDR_FREEZE_THAW */
841
+ JS_FS_END
842
+ };
843
+
844
+ #endif /* JS_HAS_SCRIPT_OBJECT */
845
+
846
+ static void
847
+ script_finalize(JSContext *cx, JSObject *obj)
848
+ {
849
+ JSScript *script;
850
+
851
+ script = (JSScript *) JS_GetPrivate(cx, obj);
852
+ if (script)
853
+ js_DestroyScript(cx, script);
854
+ }
855
+
856
+ static JSBool
857
+ script_call(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
858
+ {
859
+ #if JS_HAS_SCRIPT_OBJECT
860
+ return script_exec_sub(cx, JSVAL_TO_OBJECT(argv[-2]), argc, argv, rval);
861
+ #else
862
+ return JS_FALSE;
863
+ #endif
864
+ }
865
+
866
+ static void
867
+ script_trace(JSTracer *trc, JSObject *obj)
868
+ {
869
+ JSScript *script;
870
+
871
+ script = (JSScript *) JS_GetPrivate(trc->context, obj);
872
+ if (script)
873
+ js_TraceScript(trc, script);
874
+ }
875
+
876
+ #if !JS_HAS_SCRIPT_OBJECT
877
+ #define JSProto_Script JSProto_Object
878
+ #endif
879
+
880
+ JS_FRIEND_DATA(JSClass) js_ScriptClass = {
881
+ js_Script_str,
882
+ JSCLASS_HAS_PRIVATE | JSCLASS_HAS_RESERVED_SLOTS(1) |
883
+ JSCLASS_MARK_IS_TRACE | JSCLASS_HAS_CACHED_PROTO(JSProto_Script),
884
+ JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
885
+ JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, script_finalize,
886
+ NULL, NULL, script_call, NULL,/*XXXbe xdr*/
887
+ NULL, NULL, JS_CLASS_TRACE(script_trace), NULL
888
+ };
889
+
890
+ #if JS_HAS_SCRIPT_OBJECT
891
+
892
+ static JSBool
893
+ Script(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
894
+ {
895
+ /* If not constructing, replace obj with a new Script object. */
896
+ if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) {
897
+ obj = js_NewObject(cx, &js_ScriptClass, NULL, NULL);
898
+ if (!obj)
899
+ return JS_FALSE;
900
+
901
+ /*
902
+ * script_compile_sub does not use rval to root its temporaries so we
903
+ * can use it to root obj.
904
+ */
905
+ *rval = OBJECT_TO_JSVAL(obj);
906
+ }
907
+
908
+ if (!JS_SetReservedSlot(cx, obj, 0, INT_TO_JSVAL(0)))
909
+ return JS_FALSE;
910
+
911
+ return script_compile_sub(cx, obj, argc, argv, rval);
912
+ }
913
+
914
+ #if JS_HAS_SCRIPT_OBJECT && JS_HAS_XDR_FREEZE_THAW
915
+
916
+ static JSBool
917
+ script_static_thaw(JSContext *cx, uintN argc, jsval *vp)
918
+ {
919
+ JSObject *obj;
920
+
921
+ obj = js_NewObject(cx, &js_ScriptClass, NULL, NULL);
922
+ if (!obj)
923
+ return JS_FALSE;
924
+ vp[1] = OBJECT_TO_JSVAL(obj);
925
+ if (!script_thaw(cx, vp))
926
+ return JS_FALSE;
927
+ *vp = OBJECT_TO_JSVAL(obj);
928
+ return JS_TRUE;
929
+ }
930
+
931
+ static JSFunctionSpec script_static_methods[] = {
932
+ JS_FN(js_thaw_str, script_static_thaw, 1,1,0),
933
+ JS_FS_END
934
+ };
935
+
936
+ #else /* !JS_HAS_SCRIPT_OBJECT || !JS_HAS_XDR_FREEZE_THAW */
937
+
938
+ #define script_static_methods NULL
939
+
940
+ #endif /* !JS_HAS_SCRIPT_OBJECT || !JS_HAS_XDR_FREEZE_THAW */
941
+
942
+ JSObject *
943
+ js_InitScriptClass(JSContext *cx, JSObject *obj)
944
+ {
945
+ return JS_InitClass(cx, obj, NULL, &js_ScriptClass, Script, 1,
946
+ NULL, script_methods, NULL, script_static_methods);
947
+ }
948
+
949
+ #endif /* JS_HAS_SCRIPT_OBJECT */
950
+
951
+ /*
952
+ * Shared script filename management.
953
+ */
954
+ JS_STATIC_DLL_CALLBACK(int)
955
+ js_compare_strings(const void *k1, const void *k2)
956
+ {
957
+ return strcmp((const char *) k1, (const char *) k2) == 0;
958
+ }
959
+
960
+ /* NB: This struct overlays JSHashEntry -- see jshash.h, do not reorganize. */
961
+ typedef struct ScriptFilenameEntry {
962
+ JSHashEntry *next; /* hash chain linkage */
963
+ JSHashNumber keyHash; /* key hash function result */
964
+ const void *key; /* ptr to filename, below */
965
+ uint32 flags; /* user-defined filename prefix flags */
966
+ JSPackedBool mark; /* GC mark flag */
967
+ char filename[3]; /* two or more bytes, NUL-terminated */
968
+ } ScriptFilenameEntry;
969
+
970
+ JS_STATIC_DLL_CALLBACK(void *)
971
+ js_alloc_table_space(void *priv, size_t size)
972
+ {
973
+ return malloc(size);
974
+ }
975
+
976
+ JS_STATIC_DLL_CALLBACK(void)
977
+ js_free_table_space(void *priv, void *item)
978
+ {
979
+ free(item);
980
+ }
981
+
982
+ JS_STATIC_DLL_CALLBACK(JSHashEntry *)
983
+ js_alloc_sftbl_entry(void *priv, const void *key)
984
+ {
985
+ size_t nbytes = offsetof(ScriptFilenameEntry, filename) +
986
+ strlen((const char *) key) + 1;
987
+
988
+ return (JSHashEntry *) malloc(JS_MAX(nbytes, sizeof(JSHashEntry)));
989
+ }
990
+
991
+ JS_STATIC_DLL_CALLBACK(void)
992
+ js_free_sftbl_entry(void *priv, JSHashEntry *he, uintN flag)
993
+ {
994
+ if (flag != HT_FREE_ENTRY)
995
+ return;
996
+ free(he);
997
+ }
998
+
999
+ static JSHashAllocOps sftbl_alloc_ops = {
1000
+ js_alloc_table_space, js_free_table_space,
1001
+ js_alloc_sftbl_entry, js_free_sftbl_entry
1002
+ };
1003
+
1004
+ JSBool
1005
+ js_InitRuntimeScriptState(JSRuntime *rt)
1006
+ {
1007
+ #ifdef JS_THREADSAFE
1008
+ JS_ASSERT(!rt->scriptFilenameTableLock);
1009
+ rt->scriptFilenameTableLock = JS_NEW_LOCK();
1010
+ if (!rt->scriptFilenameTableLock)
1011
+ return JS_FALSE;
1012
+ #endif
1013
+ JS_ASSERT(!rt->scriptFilenameTable);
1014
+ rt->scriptFilenameTable =
1015
+ JS_NewHashTable(16, JS_HashString, js_compare_strings, NULL,
1016
+ &sftbl_alloc_ops, NULL);
1017
+ if (!rt->scriptFilenameTable) {
1018
+ js_FinishRuntimeScriptState(rt); /* free lock if threadsafe */
1019
+ return JS_FALSE;
1020
+ }
1021
+ JS_INIT_CLIST(&rt->scriptFilenamePrefixes);
1022
+ return JS_TRUE;
1023
+ }
1024
+
1025
+ typedef struct ScriptFilenamePrefix {
1026
+ JSCList links; /* circular list linkage for easy deletion */
1027
+ const char *name; /* pointer to pinned ScriptFilenameEntry string */
1028
+ size_t length; /* prefix string length, precomputed */
1029
+ uint32 flags; /* user-defined flags to inherit from this prefix */
1030
+ } ScriptFilenamePrefix;
1031
+
1032
+ void
1033
+ js_FinishRuntimeScriptState(JSRuntime *rt)
1034
+ {
1035
+ if (rt->scriptFilenameTable) {
1036
+ JS_HashTableDestroy(rt->scriptFilenameTable);
1037
+ rt->scriptFilenameTable = NULL;
1038
+ }
1039
+ #ifdef JS_THREADSAFE
1040
+ if (rt->scriptFilenameTableLock) {
1041
+ JS_DESTROY_LOCK(rt->scriptFilenameTableLock);
1042
+ rt->scriptFilenameTableLock = NULL;
1043
+ }
1044
+ #endif
1045
+ }
1046
+
1047
+ void
1048
+ js_FreeRuntimeScriptState(JSRuntime *rt)
1049
+ {
1050
+ ScriptFilenamePrefix *sfp;
1051
+
1052
+ if (!rt->scriptFilenameTable)
1053
+ return;
1054
+
1055
+ while (!JS_CLIST_IS_EMPTY(&rt->scriptFilenamePrefixes)) {
1056
+ sfp = (ScriptFilenamePrefix *) rt->scriptFilenamePrefixes.next;
1057
+ JS_REMOVE_LINK(&sfp->links);
1058
+ free(sfp);
1059
+ }
1060
+ js_FinishRuntimeScriptState(rt);
1061
+ }
1062
+
1063
+ #ifdef DEBUG_brendan
1064
+ #define DEBUG_SFTBL
1065
+ #endif
1066
+ #ifdef DEBUG_SFTBL
1067
+ size_t sftbl_savings = 0;
1068
+ #endif
1069
+
1070
+ static ScriptFilenameEntry *
1071
+ SaveScriptFilename(JSRuntime *rt, const char *filename, uint32 flags)
1072
+ {
1073
+ JSHashTable *table;
1074
+ JSHashNumber hash;
1075
+ JSHashEntry **hep;
1076
+ ScriptFilenameEntry *sfe;
1077
+ size_t length;
1078
+ JSCList *head, *link;
1079
+ ScriptFilenamePrefix *sfp;
1080
+
1081
+ table = rt->scriptFilenameTable;
1082
+ hash = JS_HashString(filename);
1083
+ hep = JS_HashTableRawLookup(table, hash, filename);
1084
+ sfe = (ScriptFilenameEntry *) *hep;
1085
+ #ifdef DEBUG_SFTBL
1086
+ if (sfe)
1087
+ sftbl_savings += strlen(sfe->filename);
1088
+ #endif
1089
+
1090
+ if (!sfe) {
1091
+ sfe = (ScriptFilenameEntry *)
1092
+ JS_HashTableRawAdd(table, hep, hash, filename, NULL);
1093
+ if (!sfe)
1094
+ return NULL;
1095
+ sfe->key = strcpy(sfe->filename, filename);
1096
+ sfe->flags = 0;
1097
+ sfe->mark = JS_FALSE;
1098
+ }
1099
+
1100
+ /* If saving a prefix, add it to the set in rt->scriptFilenamePrefixes. */
1101
+ if (flags != 0) {
1102
+ /* Search in case filename was saved already; we must be idempotent. */
1103
+ sfp = NULL;
1104
+ length = strlen(filename);
1105
+ for (head = link = &rt->scriptFilenamePrefixes;
1106
+ link->next != head;
1107
+ link = link->next) {
1108
+ /* Lag link behind sfp to insert in non-increasing length order. */
1109
+ sfp = (ScriptFilenamePrefix *) link->next;
1110
+ if (!strcmp(sfp->name, filename))
1111
+ break;
1112
+ if (sfp->length <= length) {
1113
+ sfp = NULL;
1114
+ break;
1115
+ }
1116
+ sfp = NULL;
1117
+ }
1118
+
1119
+ if (!sfp) {
1120
+ /* No such prefix: add one now. */
1121
+ sfp = (ScriptFilenamePrefix *) malloc(sizeof(ScriptFilenamePrefix));
1122
+ if (!sfp)
1123
+ return NULL;
1124
+ JS_INSERT_AFTER(&sfp->links, link);
1125
+ sfp->name = sfe->filename;
1126
+ sfp->length = length;
1127
+ sfp->flags = 0;
1128
+ }
1129
+
1130
+ /*
1131
+ * Accumulate flags in both sfe and sfp: sfe for later access from the
1132
+ * JS_GetScriptedCallerFilenameFlags debug-API, and sfp so that longer
1133
+ * filename entries can inherit by prefix.
1134
+ */
1135
+ sfe->flags |= flags;
1136
+ sfp->flags |= flags;
1137
+ }
1138
+
1139
+ return sfe;
1140
+ }
1141
+
1142
+ const char *
1143
+ js_SaveScriptFilename(JSContext *cx, const char *filename)
1144
+ {
1145
+ JSRuntime *rt;
1146
+ ScriptFilenameEntry *sfe;
1147
+ JSCList *head, *link;
1148
+ ScriptFilenamePrefix *sfp;
1149
+
1150
+ rt = cx->runtime;
1151
+ JS_ACQUIRE_LOCK(rt->scriptFilenameTableLock);
1152
+ sfe = SaveScriptFilename(rt, filename, 0);
1153
+ if (!sfe) {
1154
+ JS_RELEASE_LOCK(rt->scriptFilenameTableLock);
1155
+ JS_ReportOutOfMemory(cx);
1156
+ return NULL;
1157
+ }
1158
+
1159
+ /*
1160
+ * Try to inherit flags by prefix. We assume there won't be more than a
1161
+ * few (dozen! ;-) prefixes, so linear search is tolerable.
1162
+ * XXXbe every time I've assumed that in the JS engine, I've been wrong!
1163
+ */
1164
+ for (head = &rt->scriptFilenamePrefixes, link = head->next;
1165
+ link != head;
1166
+ link = link->next) {
1167
+ sfp = (ScriptFilenamePrefix *) link;
1168
+ if (!strncmp(sfp->name, filename, sfp->length)) {
1169
+ sfe->flags |= sfp->flags;
1170
+ break;
1171
+ }
1172
+ }
1173
+ JS_RELEASE_LOCK(rt->scriptFilenameTableLock);
1174
+ return sfe->filename;
1175
+ }
1176
+
1177
+ const char *
1178
+ js_SaveScriptFilenameRT(JSRuntime *rt, const char *filename, uint32 flags)
1179
+ {
1180
+ ScriptFilenameEntry *sfe;
1181
+
1182
+ /* This may be called very early, via the jsdbgapi.h entry point. */
1183
+ if (!rt->scriptFilenameTable && !js_InitRuntimeScriptState(rt))
1184
+ return NULL;
1185
+
1186
+ JS_ACQUIRE_LOCK(rt->scriptFilenameTableLock);
1187
+ sfe = SaveScriptFilename(rt, filename, flags);
1188
+ JS_RELEASE_LOCK(rt->scriptFilenameTableLock);
1189
+ if (!sfe)
1190
+ return NULL;
1191
+
1192
+ return sfe->filename;
1193
+ }
1194
+
1195
+ /*
1196
+ * Back up from a saved filename by its offset within its hash table entry.
1197
+ */
1198
+ #define FILENAME_TO_SFE(fn) \
1199
+ ((ScriptFilenameEntry *) ((fn) - offsetof(ScriptFilenameEntry, filename)))
1200
+
1201
+ /*
1202
+ * The sfe->key member, redundant given sfe->filename but required by the old
1203
+ * jshash.c code, here gives us a useful sanity check. This assertion will
1204
+ * very likely botch if someone tries to mark a string that wasn't allocated
1205
+ * as an sfe->filename.
1206
+ */
1207
+ #define ASSERT_VALID_SFE(sfe) JS_ASSERT((sfe)->key == (sfe)->filename)
1208
+
1209
+ uint32
1210
+ js_GetScriptFilenameFlags(const char *filename)
1211
+ {
1212
+ ScriptFilenameEntry *sfe;
1213
+
1214
+ sfe = FILENAME_TO_SFE(filename);
1215
+ ASSERT_VALID_SFE(sfe);
1216
+ return sfe->flags;
1217
+ }
1218
+
1219
+ void
1220
+ js_MarkScriptFilename(const char *filename)
1221
+ {
1222
+ ScriptFilenameEntry *sfe;
1223
+
1224
+ sfe = FILENAME_TO_SFE(filename);
1225
+ ASSERT_VALID_SFE(sfe);
1226
+ sfe->mark = JS_TRUE;
1227
+ }
1228
+
1229
+ JS_STATIC_DLL_CALLBACK(intN)
1230
+ js_script_filename_marker(JSHashEntry *he, intN i, void *arg)
1231
+ {
1232
+ ScriptFilenameEntry *sfe = (ScriptFilenameEntry *) he;
1233
+
1234
+ sfe->mark = JS_TRUE;
1235
+ return HT_ENUMERATE_NEXT;
1236
+ }
1237
+
1238
+ void
1239
+ js_MarkScriptFilenames(JSRuntime *rt, JSBool keepAtoms)
1240
+ {
1241
+ JSCList *head, *link;
1242
+ ScriptFilenamePrefix *sfp;
1243
+
1244
+ if (!rt->scriptFilenameTable)
1245
+ return;
1246
+
1247
+ if (keepAtoms) {
1248
+ JS_HashTableEnumerateEntries(rt->scriptFilenameTable,
1249
+ js_script_filename_marker,
1250
+ rt);
1251
+ }
1252
+ for (head = &rt->scriptFilenamePrefixes, link = head->next;
1253
+ link != head;
1254
+ link = link->next) {
1255
+ sfp = (ScriptFilenamePrefix *) link;
1256
+ js_MarkScriptFilename(sfp->name);
1257
+ }
1258
+ }
1259
+
1260
+ JS_STATIC_DLL_CALLBACK(intN)
1261
+ js_script_filename_sweeper(JSHashEntry *he, intN i, void *arg)
1262
+ {
1263
+ ScriptFilenameEntry *sfe = (ScriptFilenameEntry *) he;
1264
+
1265
+ if (!sfe->mark)
1266
+ return HT_ENUMERATE_REMOVE;
1267
+ sfe->mark = JS_FALSE;
1268
+ return HT_ENUMERATE_NEXT;
1269
+ }
1270
+
1271
+ void
1272
+ js_SweepScriptFilenames(JSRuntime *rt)
1273
+ {
1274
+ if (!rt->scriptFilenameTable)
1275
+ return;
1276
+
1277
+ JS_HashTableEnumerateEntries(rt->scriptFilenameTable,
1278
+ js_script_filename_sweeper,
1279
+ rt);
1280
+ #ifdef DEBUG_notme
1281
+ #ifdef DEBUG_SFTBL
1282
+ printf("script filename table savings so far: %u\n", sftbl_savings);
1283
+ #endif
1284
+ #endif
1285
+ }
1286
+
1287
+ /*
1288
+ * JSScript data structures memory alignment:
1289
+ *
1290
+ * JSScript
1291
+ * JSObjectArray script objects' descriptor if JSScript.objectsOffset != 0,
1292
+ * use JS_SCRIPT_OBJECTS(script) macro to access it.
1293
+ * JSObjectArray script regexps' descriptor if JSScript.regexpsOffset != 0,
1294
+ * use JS_SCRIPT_REGEXPS(script) macro to access it.
1295
+ * JSTryNoteArray script try notes' descriptor if JSScript.tryNotesOffset
1296
+ * != 0, use JS_SCRIPT_TRYNOTES(script) macro to access it.
1297
+ * JSAtom *a[] array of JSScript.atomMap.length atoms pointed by
1298
+ * JSScript.atomMap.vector if any.
1299
+ * JSObject *o[] array of JS_SCRIPT_OBJECTS(script)->length objects if any
1300
+ * pointed by JS_SCRIPT_OBJECTS(script)->vector.
1301
+ * JSObject *r[] array of JS_SCRIPT_REGEXPS(script)->length regexps if any
1302
+ * pointed by JS_SCRIPT_REGEXPS(script)->vector.
1303
+ * JSTryNote t[] array of JS_SCRIPT_TRYNOTES(script)->length try notes if any
1304
+ * pointed by JS_SCRIPT_TRYNOTES(script)->vector.
1305
+ * jsbytecode b[] script bytecode pointed by JSScript.code.
1306
+ * jssrcnote s[] script source notes, use SCRIPT_NOTES(script) to access it
1307
+ *
1308
+ * The alignment avoids gaps between entries as alignment requirement for each
1309
+ * subsequent structure or array is the same or divides the alignment
1310
+ * requirement for the previous one.
1311
+ *
1312
+ * The followings asserts checks that assuming that the alignment requirement
1313
+ * for JSObjectArray and JSTryNoteArray are sizeof(void *) and for JSTryNote
1314
+ * it is sizeof(uint32) as the structure consists of 3 uint32 fields.
1315
+ */
1316
+ JS_STATIC_ASSERT(sizeof(JSScript) % sizeof(void *) == 0);
1317
+ JS_STATIC_ASSERT(sizeof(JSObjectArray) % sizeof(void *) == 0);
1318
+ JS_STATIC_ASSERT(sizeof(JSTryNoteArray) == sizeof(JSObjectArray));
1319
+ JS_STATIC_ASSERT(sizeof(JSAtom *) == sizeof(JSObject *));
1320
+ JS_STATIC_ASSERT(sizeof(JSObject *) % sizeof(uint32) == 0);
1321
+ JS_STATIC_ASSERT(sizeof(JSTryNote) == 3 * sizeof(uint32));
1322
+ JS_STATIC_ASSERT(sizeof(uint32) % sizeof(jsbytecode) == 0);
1323
+ JS_STATIC_ASSERT(sizeof(jsbytecode) % sizeof(jssrcnote) == 0);
1324
+
1325
+ /*
1326
+ * Check that uint8 offset for object, regexp and try note arrays is sufficient.
1327
+ */
1328
+ JS_STATIC_ASSERT(sizeof(JSScript) + 2 * sizeof(JSObjectArray) < JS_BIT(8));
1329
+
1330
+ JSScript *
1331
+ js_NewScript(JSContext *cx, uint32 length, uint32 nsrcnotes, uint32 natoms,
1332
+ uint32 nobjects, uint32 nregexps, uint32 ntrynotes)
1333
+ {
1334
+ size_t size, vectorSize;
1335
+ JSScript *script;
1336
+ uint8 *cursor;
1337
+
1338
+ size = sizeof(JSScript) +
1339
+ sizeof(JSAtom *) * natoms +
1340
+ length * sizeof(jsbytecode) +
1341
+ nsrcnotes * sizeof(jssrcnote);
1342
+ if (nobjects != 0)
1343
+ size += sizeof(JSObjectArray) + nobjects * sizeof(JSObject *);
1344
+ if (nregexps != 0)
1345
+ size += sizeof(JSObjectArray) + nregexps * sizeof(JSObject *);
1346
+ if (ntrynotes != 0)
1347
+ size += sizeof(JSTryNoteArray) + ntrynotes * sizeof(JSTryNote);
1348
+
1349
+ script = (JSScript *) JS_malloc(cx, size);
1350
+ if (!script)
1351
+ return NULL;
1352
+ memset(script, 0, sizeof(JSScript));
1353
+ script->length = length;
1354
+ script->version = cx->version;
1355
+
1356
+ cursor = (uint8 *)script + sizeof(JSScript);
1357
+ if (nobjects != 0) {
1358
+ script->objectsOffset = (uint8)(cursor - (uint8 *)script);
1359
+ cursor += sizeof(JSObjectArray);
1360
+ }
1361
+ if (nregexps != 0) {
1362
+ script->regexpsOffset = (uint8)(cursor - (uint8 *)script);
1363
+ cursor += sizeof(JSObjectArray);
1364
+ }
1365
+ if (ntrynotes != 0) {
1366
+ script->trynotesOffset = (uint8)(cursor - (uint8 *)script);
1367
+ cursor += sizeof(JSTryNoteArray);
1368
+ }
1369
+
1370
+ if (natoms != 0) {
1371
+ script->atomMap.length = natoms;
1372
+ script->atomMap.vector = (JSAtom **)cursor;
1373
+ vectorSize = natoms * sizeof(script->atomMap.vector[0]);
1374
+
1375
+ /*
1376
+ * Clear object map's vector so the GC tracing can run when not yet
1377
+ * all atoms are copied to the array.
1378
+ */
1379
+ memset(cursor, 0, vectorSize);
1380
+ cursor += vectorSize;
1381
+ }
1382
+ if (nobjects != 0) {
1383
+ JS_SCRIPT_OBJECTS(script)->length = nobjects;
1384
+ JS_SCRIPT_OBJECTS(script)->vector = (JSObject **)cursor;
1385
+ vectorSize = nobjects * sizeof(JS_SCRIPT_OBJECTS(script)->vector[0]);
1386
+ memset(cursor, 0, vectorSize);
1387
+ cursor += vectorSize;
1388
+ }
1389
+ if (nregexps != 0) {
1390
+ JS_SCRIPT_REGEXPS(script)->length = nregexps;
1391
+ JS_SCRIPT_REGEXPS(script)->vector = (JSObject **)cursor;
1392
+ vectorSize = nregexps * sizeof(JS_SCRIPT_REGEXPS(script)->vector[0]);
1393
+ memset(cursor, 0, vectorSize);
1394
+ cursor += vectorSize;
1395
+ }
1396
+ if (ntrynotes != 0) {
1397
+ JS_SCRIPT_TRYNOTES(script)->length = ntrynotes;
1398
+ JS_SCRIPT_TRYNOTES(script)->vector = (JSTryNote *)cursor;
1399
+ vectorSize = ntrynotes * sizeof(JS_SCRIPT_TRYNOTES(script)->vector[0]);
1400
+ #ifdef DEBUG
1401
+ memset(cursor, 0, vectorSize);
1402
+ #endif
1403
+ cursor += vectorSize;
1404
+ }
1405
+
1406
+ script->code = script->main = (jsbytecode *)cursor;
1407
+ JS_ASSERT(cursor +
1408
+ length * sizeof(jsbytecode) +
1409
+ nsrcnotes * sizeof(jssrcnote) ==
1410
+ (uint8 *)script + size);
1411
+
1412
+ #ifdef CHECK_SCRIPT_OWNER
1413
+ script->owner = cx->thread;
1414
+ #endif
1415
+ return script;
1416
+ }
1417
+
1418
+ JSScript *
1419
+ js_NewScriptFromCG(JSContext *cx, JSCodeGenerator *cg)
1420
+ {
1421
+ uint32 mainLength, prologLength, nsrcnotes;
1422
+ JSScript *script;
1423
+ const char *filename;
1424
+ JSFunction *fun;
1425
+
1426
+ /* The counts of indexed things must be checked during code generation. */
1427
+ JS_ASSERT(cg->atomList.count <= INDEX_LIMIT);
1428
+ JS_ASSERT(cg->objectList.length <= INDEX_LIMIT);
1429
+ JS_ASSERT(cg->regexpList.length <= INDEX_LIMIT);
1430
+
1431
+ mainLength = CG_OFFSET(cg);
1432
+ prologLength = CG_PROLOG_OFFSET(cg);
1433
+ CG_COUNT_FINAL_SRCNOTES(cg, nsrcnotes);
1434
+ script = js_NewScript(cx, prologLength + mainLength, nsrcnotes,
1435
+ cg->atomList.count, cg->objectList.length,
1436
+ cg->regexpList.length, cg->ntrynotes);
1437
+ if (!script)
1438
+ return NULL;
1439
+
1440
+ /* Now that we have script, error control flow must go to label bad. */
1441
+ script->main += prologLength;
1442
+ memcpy(script->code, CG_PROLOG_BASE(cg), prologLength * sizeof(jsbytecode));
1443
+ memcpy(script->main, CG_BASE(cg), mainLength * sizeof(jsbytecode));
1444
+ script->ngvars = cg->treeContext.ngvars;
1445
+
1446
+ js_InitAtomMap(cx, &script->atomMap, &cg->atomList);
1447
+
1448
+ filename = cg->treeContext.parseContext->tokenStream.filename;
1449
+ if (filename) {
1450
+ script->filename = js_SaveScriptFilename(cx, filename);
1451
+ if (!script->filename)
1452
+ goto bad;
1453
+ }
1454
+ script->lineno = cg->firstLine;
1455
+ script->depth = cg->maxStackDepth;
1456
+ script->principals = cg->treeContext.parseContext->principals;
1457
+ if (script->principals)
1458
+ JSPRINCIPALS_HOLD(cx, script->principals);
1459
+
1460
+ if (!js_FinishTakingSrcNotes(cx, cg, SCRIPT_NOTES(script)))
1461
+ goto bad;
1462
+ if (cg->ntrynotes != 0)
1463
+ js_FinishTakingTryNotes(cg, JS_SCRIPT_TRYNOTES(script));
1464
+ if (cg->objectList.length != 0)
1465
+ FinishParsedObjects(&cg->objectList, JS_SCRIPT_OBJECTS(script));
1466
+ if (cg->regexpList.length != 0)
1467
+ FinishParsedObjects(&cg->regexpList, JS_SCRIPT_REGEXPS(script));
1468
+
1469
+ /*
1470
+ * We initialize fun->u.script to be the script constructed above
1471
+ * so that the debugger has a valid FUN_SCRIPT(fun).
1472
+ */
1473
+ fun = NULL;
1474
+ if (cg->treeContext.flags & TCF_IN_FUNCTION) {
1475
+ fun = cg->treeContext.fun;
1476
+ JS_ASSERT(FUN_INTERPRETED(fun) && !FUN_SCRIPT(fun));
1477
+ js_FreezeLocalNames(cx, fun);
1478
+ fun->u.i.script = script;
1479
+ #ifdef CHECK_SCRIPT_OWNER
1480
+ script->owner = NULL;
1481
+ #endif
1482
+ if (cg->treeContext.flags & TCF_FUN_HEAVYWEIGHT)
1483
+ fun->flags |= JSFUN_HEAVYWEIGHT;
1484
+ if (fun->flags & JSFUN_HEAVYWEIGHT)
1485
+ ++cg->treeContext.maxScopeDepth;
1486
+ }
1487
+
1488
+ #ifdef JS_SCOPE_DEPTH_METER
1489
+ JS_BASIC_STATS_ACCUM(&cx->runtime->lexicalScopeDepthStats,
1490
+ cg->treeContext.maxScopeDepth);
1491
+ #endif
1492
+
1493
+ /* Tell the debugger about this compiled script. */
1494
+ js_CallNewScriptHook(cx, script, fun);
1495
+ return script;
1496
+
1497
+ bad:
1498
+ js_DestroyScript(cx, script);
1499
+ return NULL;
1500
+ }
1501
+
1502
+ JS_FRIEND_API(void)
1503
+ js_CallNewScriptHook(JSContext *cx, JSScript *script, JSFunction *fun)
1504
+ {
1505
+ JSNewScriptHook hook;
1506
+
1507
+ hook = cx->debugHooks->newScriptHook;
1508
+ if (hook) {
1509
+ JS_KEEP_ATOMS(cx->runtime);
1510
+ hook(cx, script->filename, script->lineno, script, fun,
1511
+ cx->debugHooks->newScriptHookData);
1512
+ JS_UNKEEP_ATOMS(cx->runtime);
1513
+ }
1514
+ }
1515
+
1516
+ JS_FRIEND_API(void)
1517
+ js_CallDestroyScriptHook(JSContext *cx, JSScript *script)
1518
+ {
1519
+ JSDestroyScriptHook hook;
1520
+
1521
+ hook = cx->debugHooks->destroyScriptHook;
1522
+ if (hook)
1523
+ hook(cx, script, cx->debugHooks->destroyScriptHookData);
1524
+ }
1525
+
1526
+ void
1527
+ js_DestroyScript(JSContext *cx, JSScript *script)
1528
+ {
1529
+ js_CallDestroyScriptHook(cx, script);
1530
+ JS_ClearScriptTraps(cx, script);
1531
+
1532
+ if (script->principals)
1533
+ JSPRINCIPALS_DROP(cx, script->principals);
1534
+
1535
+ if (JS_GSN_CACHE(cx).code == script->code)
1536
+ JS_CLEAR_GSN_CACHE(cx);
1537
+
1538
+ /*
1539
+ * The GC flushes all property caches, so no need to purge just the
1540
+ * entries for this script.
1541
+ *
1542
+ * JS_THREADSAFE note: js_FlushPropertyCacheForScript flushes only the
1543
+ * current thread's property cache, so a script not owned by a function
1544
+ * or object, which hands off lifetime management for that script to the
1545
+ * GC, must be used by only one thread over its lifetime.
1546
+ *
1547
+ * This should be an API-compatible change, since a script is never safe
1548
+ * against premature GC if shared among threads without a rooted object
1549
+ * wrapping it to protect the script's mapped atoms against GC. We use
1550
+ * script->owner to enforce this requirement via assertions.
1551
+ */
1552
+ #ifdef CHECK_SCRIPT_OWNER
1553
+ JS_ASSERT_IF(cx->runtime->gcRunning, !script->owner);
1554
+ #endif
1555
+
1556
+ if (!cx->runtime->gcRunning &&
1557
+ !(cx->fp && (cx->fp->flags & JSFRAME_EVAL))) {
1558
+ #ifdef CHECK_SCRIPT_OWNER
1559
+ JS_ASSERT(script->owner == cx->thread);
1560
+ #endif
1561
+ js_FlushPropertyCacheForScript(cx, script);
1562
+ }
1563
+
1564
+ JS_free(cx, script);
1565
+ }
1566
+
1567
+ void
1568
+ js_TraceScript(JSTracer *trc, JSScript *script)
1569
+ {
1570
+ JSAtomMap *map;
1571
+ uintN i, length;
1572
+ JSAtom **vector;
1573
+ jsval v;
1574
+ JSObjectArray *objarray;
1575
+
1576
+ map = &script->atomMap;
1577
+ length = map->length;
1578
+ vector = map->vector;
1579
+ for (i = 0; i < length; i++) {
1580
+ v = ATOM_KEY(vector[i]);
1581
+ if (JSVAL_IS_TRACEABLE(v)) {
1582
+ JS_SET_TRACING_INDEX(trc, "atomMap", i);
1583
+ JS_CallTracer(trc, JSVAL_TO_TRACEABLE(v), JSVAL_TRACE_KIND(v));
1584
+ }
1585
+ }
1586
+
1587
+ if (script->objectsOffset != 0) {
1588
+ objarray = JS_SCRIPT_OBJECTS(script);
1589
+ i = objarray->length;
1590
+ do {
1591
+ --i;
1592
+ if (objarray->vector[i]) {
1593
+ JS_SET_TRACING_INDEX(trc, "objects", i);
1594
+ JS_CallTracer(trc, objarray->vector[i], JSTRACE_OBJECT);
1595
+ }
1596
+ } while (i != 0);
1597
+ }
1598
+
1599
+ if (script->regexpsOffset != 0) {
1600
+ objarray = JS_SCRIPT_REGEXPS(script);
1601
+ i = objarray->length;
1602
+ do {
1603
+ --i;
1604
+ if (objarray->vector[i]) {
1605
+ JS_SET_TRACING_INDEX(trc, "regexps", i);
1606
+ JS_CallTracer(trc, objarray->vector[i], JSTRACE_OBJECT);
1607
+ }
1608
+ } while (i != 0);
1609
+ }
1610
+
1611
+ if (IS_GC_MARKING_TRACER(trc) && script->filename)
1612
+ js_MarkScriptFilename(script->filename);
1613
+ }
1614
+
1615
+ typedef struct GSNCacheEntry {
1616
+ JSDHashEntryHdr hdr;
1617
+ jsbytecode *pc;
1618
+ jssrcnote *sn;
1619
+ } GSNCacheEntry;
1620
+
1621
+ #define GSN_CACHE_THRESHOLD 100
1622
+
1623
+ jssrcnote *
1624
+ js_GetSrcNoteCached(JSContext *cx, JSScript *script, jsbytecode *pc)
1625
+ {
1626
+ ptrdiff_t target, offset;
1627
+ GSNCacheEntry *entry;
1628
+ jssrcnote *sn, *result;
1629
+ uintN nsrcnotes;
1630
+
1631
+
1632
+ target = PTRDIFF(pc, script->code, jsbytecode);
1633
+ if ((uint32)target >= script->length)
1634
+ return NULL;
1635
+
1636
+ if (JS_GSN_CACHE(cx).code == script->code) {
1637
+ JS_METER_GSN_CACHE(cx, hits);
1638
+ entry = (GSNCacheEntry *)
1639
+ JS_DHashTableOperate(&JS_GSN_CACHE(cx).table, pc,
1640
+ JS_DHASH_LOOKUP);
1641
+ return entry->sn;
1642
+ }
1643
+
1644
+ JS_METER_GSN_CACHE(cx, misses);
1645
+ offset = 0;
1646
+ for (sn = SCRIPT_NOTES(script); ; sn = SN_NEXT(sn)) {
1647
+ if (SN_IS_TERMINATOR(sn)) {
1648
+ result = NULL;
1649
+ break;
1650
+ }
1651
+ offset += SN_DELTA(sn);
1652
+ if (offset == target && SN_IS_GETTABLE(sn)) {
1653
+ result = sn;
1654
+ break;
1655
+ }
1656
+ }
1657
+
1658
+ if (JS_GSN_CACHE(cx).code != script->code &&
1659
+ script->length >= GSN_CACHE_THRESHOLD) {
1660
+ JS_CLEAR_GSN_CACHE(cx);
1661
+ nsrcnotes = 0;
1662
+ for (sn = SCRIPT_NOTES(script); !SN_IS_TERMINATOR(sn);
1663
+ sn = SN_NEXT(sn)) {
1664
+ if (SN_IS_GETTABLE(sn))
1665
+ ++nsrcnotes;
1666
+ }
1667
+ if (!JS_DHashTableInit(&JS_GSN_CACHE(cx).table, JS_DHashGetStubOps(),
1668
+ NULL, sizeof(GSNCacheEntry),
1669
+ JS_DHASH_DEFAULT_CAPACITY(nsrcnotes))) {
1670
+ JS_GSN_CACHE(cx).table.ops = NULL;
1671
+ } else {
1672
+ pc = script->code;
1673
+ for (sn = SCRIPT_NOTES(script); !SN_IS_TERMINATOR(sn);
1674
+ sn = SN_NEXT(sn)) {
1675
+ pc += SN_DELTA(sn);
1676
+ if (SN_IS_GETTABLE(sn)) {
1677
+ entry = (GSNCacheEntry *)
1678
+ JS_DHashTableOperate(&JS_GSN_CACHE(cx).table, pc,
1679
+ JS_DHASH_ADD);
1680
+ entry->pc = pc;
1681
+ entry->sn = sn;
1682
+ }
1683
+ }
1684
+ JS_GSN_CACHE(cx).code = script->code;
1685
+ JS_METER_GSN_CACHE(cx, fills);
1686
+ }
1687
+ }
1688
+
1689
+ return result;
1690
+ }
1691
+
1692
+ uintN
1693
+ js_PCToLineNumber(JSContext *cx, JSScript *script, jsbytecode *pc)
1694
+ {
1695
+ JSFunction *fun;
1696
+ uintN lineno;
1697
+ ptrdiff_t offset, target;
1698
+ jssrcnote *sn;
1699
+ JSSrcNoteType type;
1700
+
1701
+ /* Cope with JSStackFrame.pc value prior to entering js_Interpret. */
1702
+ if (!pc)
1703
+ return 0;
1704
+
1705
+ /*
1706
+ * Special case: function definition needs no line number note because
1707
+ * the function's script contains its starting line number.
1708
+ */
1709
+ if (js_CodeSpec[*pc].format & JOF_INDEXBASE)
1710
+ pc += js_CodeSpec[*pc].length;
1711
+ if (*pc == JSOP_DEFFUN) {
1712
+ GET_FUNCTION_FROM_BYTECODE(script, pc, 0, fun);
1713
+ return fun->u.i.script->lineno;
1714
+ }
1715
+
1716
+ /*
1717
+ * General case: walk through source notes accumulating their deltas,
1718
+ * keeping track of line-number notes, until we pass the note for pc's
1719
+ * offset within script->code.
1720
+ */
1721
+ lineno = script->lineno;
1722
+ offset = 0;
1723
+ target = PTRDIFF(pc, script->code, jsbytecode);
1724
+ for (sn = SCRIPT_NOTES(script); !SN_IS_TERMINATOR(sn); sn = SN_NEXT(sn)) {
1725
+ offset += SN_DELTA(sn);
1726
+ type = (JSSrcNoteType) SN_TYPE(sn);
1727
+ if (type == SRC_SETLINE) {
1728
+ if (offset <= target)
1729
+ lineno = (uintN) js_GetSrcNoteOffset(sn, 0);
1730
+ } else if (type == SRC_NEWLINE) {
1731
+ if (offset <= target)
1732
+ lineno++;
1733
+ }
1734
+ if (offset > target)
1735
+ break;
1736
+ }
1737
+ return lineno;
1738
+ }
1739
+
1740
+ /* The line number limit is the same as the jssrcnote offset limit. */
1741
+ #define SN_LINE_LIMIT (SN_3BYTE_OFFSET_FLAG << 16)
1742
+
1743
+ jsbytecode *
1744
+ js_LineNumberToPC(JSScript *script, uintN target)
1745
+ {
1746
+ ptrdiff_t offset, best;
1747
+ uintN lineno, bestdiff, diff;
1748
+ jssrcnote *sn;
1749
+ JSSrcNoteType type;
1750
+
1751
+ offset = 0;
1752
+ best = -1;
1753
+ lineno = script->lineno;
1754
+ bestdiff = SN_LINE_LIMIT;
1755
+ for (sn = SCRIPT_NOTES(script); !SN_IS_TERMINATOR(sn); sn = SN_NEXT(sn)) {
1756
+ /*
1757
+ * Exact-match only if offset is not in the prolog; otherwise use
1758
+ * nearest greater-or-equal line number match.
1759
+ */
1760
+ if (lineno == target && script->code + offset >= script->main)
1761
+ goto out;
1762
+ if (lineno >= target) {
1763
+ diff = lineno - target;
1764
+ if (diff < bestdiff) {
1765
+ bestdiff = diff;
1766
+ best = offset;
1767
+ }
1768
+ }
1769
+ offset += SN_DELTA(sn);
1770
+ type = (JSSrcNoteType) SN_TYPE(sn);
1771
+ if (type == SRC_SETLINE) {
1772
+ lineno = (uintN) js_GetSrcNoteOffset(sn, 0);
1773
+ } else if (type == SRC_NEWLINE) {
1774
+ lineno++;
1775
+ }
1776
+ }
1777
+ if (best >= 0)
1778
+ offset = best;
1779
+ out:
1780
+ return script->code + offset;
1781
+ }
1782
+
1783
+ JS_FRIEND_API(uintN)
1784
+ js_GetScriptLineExtent(JSScript *script)
1785
+ {
1786
+ uintN lineno;
1787
+ jssrcnote *sn;
1788
+ JSSrcNoteType type;
1789
+
1790
+ lineno = script->lineno;
1791
+ for (sn = SCRIPT_NOTES(script); !SN_IS_TERMINATOR(sn); sn = SN_NEXT(sn)) {
1792
+ type = (JSSrcNoteType) SN_TYPE(sn);
1793
+ if (type == SRC_SETLINE) {
1794
+ lineno = (uintN) js_GetSrcNoteOffset(sn, 0);
1795
+ } else if (type == SRC_NEWLINE) {
1796
+ lineno++;
1797
+ }
1798
+ }
1799
+ return 1 + lineno - script->lineno;
1800
+ }
1801
+
1802
+ #if JS_HAS_GENERATORS
1803
+
1804
+ JSBool
1805
+ js_IsInsideTryWithFinally(JSScript *script, jsbytecode *pc)
1806
+ {
1807
+ JSTryNoteArray *tarray;
1808
+ JSTryNote *tn, *tnlimit;
1809
+ uint32 off;
1810
+
1811
+ JS_ASSERT(script->code <= pc);
1812
+ JS_ASSERT(pc < script->code + script->length);
1813
+
1814
+ if (!script->trynotesOffset != 0)
1815
+ return JS_FALSE;
1816
+ tarray = JS_SCRIPT_TRYNOTES(script);
1817
+ JS_ASSERT(tarray->length != 0);
1818
+
1819
+ tn = tarray->vector;
1820
+ tnlimit = tn + tarray->length;
1821
+ off = (uint32)(pc - script->main);
1822
+ do {
1823
+ if (off - tn->start < tn->length) {
1824
+ if (tn->kind == JSTN_FINALLY)
1825
+ return JS_TRUE;
1826
+ JS_ASSERT(tn->kind == JSTN_CATCH);
1827
+ }
1828
+ } while (++tn != tnlimit);
1829
+ return JS_FALSE;
1830
+ }
1831
+
1832
+ #endif