jbarnette-johnson 1.0.0.200806240111 → 1.0.0.200807291507

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