johnson 1.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.autotest +14 -0
- data/CHANGELOG.rdoc +11 -0
- data/Manifest.txt +370 -0
- data/README.rdoc +60 -0
- data/Rakefile +42 -0
- data/bin/johnson +108 -0
- data/docs/MINGW32.mk +124 -0
- data/docs/cross-compile.txt +38 -0
- data/ext/spidermonkey/context.c +115 -0
- data/ext/spidermonkey/context.h +19 -0
- data/ext/spidermonkey/conversions.c +320 -0
- data/ext/spidermonkey/conversions.h +18 -0
- data/ext/spidermonkey/debugger.c +226 -0
- data/ext/spidermonkey/debugger.h +9 -0
- data/ext/spidermonkey/extconf.rb +30 -0
- data/ext/spidermonkey/extensions.c +37 -0
- data/ext/spidermonkey/extensions.h +12 -0
- data/ext/spidermonkey/global.c +40 -0
- data/ext/spidermonkey/global.h +11 -0
- data/ext/spidermonkey/idhash.c +16 -0
- data/ext/spidermonkey/idhash.h +8 -0
- data/ext/spidermonkey/immutable_node.c +1153 -0
- data/ext/spidermonkey/immutable_node.c.erb +523 -0
- data/ext/spidermonkey/immutable_node.h +22 -0
- data/ext/spidermonkey/jroot.h +187 -0
- data/ext/spidermonkey/js_land_proxy.c +610 -0
- data/ext/spidermonkey/js_land_proxy.h +20 -0
- data/ext/spidermonkey/ruby_land_proxy.c +543 -0
- data/ext/spidermonkey/ruby_land_proxy.h +17 -0
- data/ext/spidermonkey/runtime.c +330 -0
- data/ext/spidermonkey/runtime.h +25 -0
- data/ext/spidermonkey/spidermonkey.c +20 -0
- data/ext/spidermonkey/spidermonkey.h +29 -0
- data/johnson.gemspec +44 -0
- data/js/johnson/cli.js +30 -0
- data/js/johnson/prelude.js +80 -0
- data/lib/johnson.rb +55 -0
- data/lib/johnson/cli.rb +7 -0
- data/lib/johnson/cli/options.rb +67 -0
- data/lib/johnson/error.rb +4 -0
- data/lib/johnson/nodes.rb +7 -0
- data/lib/johnson/nodes/binary_node.rb +65 -0
- data/lib/johnson/nodes/for.rb +14 -0
- data/lib/johnson/nodes/for_in.rb +12 -0
- data/lib/johnson/nodes/function.rb +13 -0
- data/lib/johnson/nodes/list.rb +28 -0
- data/lib/johnson/nodes/node.rb +68 -0
- data/lib/johnson/nodes/ternary_node.rb +20 -0
- data/lib/johnson/parser.rb +21 -0
- data/lib/johnson/parser/syntax_error.rb +13 -0
- data/lib/johnson/runtime.rb +63 -0
- data/lib/johnson/spidermonkey/context.rb +10 -0
- data/lib/johnson/spidermonkey/debugger.rb +67 -0
- data/lib/johnson/spidermonkey/immutable_node.rb +282 -0
- data/lib/johnson/spidermonkey/js_land_proxy.rb +62 -0
- data/lib/johnson/spidermonkey/mutable_tree_visitor.rb +242 -0
- data/lib/johnson/spidermonkey/ruby_land_proxy.rb +54 -0
- data/lib/johnson/spidermonkey/runtime.rb +103 -0
- data/lib/johnson/version.rb +3 -0
- data/lib/johnson/visitable.rb +16 -0
- data/lib/johnson/visitors.rb +5 -0
- data/lib/johnson/visitors/dot_visitor.rb +169 -0
- data/lib/johnson/visitors/ecma_visitor.rb +323 -0
- data/lib/johnson/visitors/enumerating_visitor.rb +15 -0
- data/lib/johnson/visitors/sexp_visitor.rb +174 -0
- data/lib/johnson/visitors/visitor.rb +91 -0
- data/lib/rails/init.rb +37 -0
- data/lib/tasks/gem.rake +9 -0
- data/lib/tasks/parsing.rake +37 -0
- data/lib/tasks/testing.rake +36 -0
- data/lib/tasks/vendor.rake +20 -0
- data/test/helper.rb +55 -0
- data/test/johnson/browser_test.rb +43 -0
- data/test/johnson/conversions/array_test.rb +32 -0
- data/test/johnson/conversions/boolean_test.rb +17 -0
- data/test/johnson/conversions/callable_test.rb +34 -0
- data/test/johnson/conversions/file_test.rb +15 -0
- data/test/johnson/conversions/nil_test.rb +20 -0
- data/test/johnson/conversions/number_test.rb +34 -0
- data/test/johnson/conversions/regexp_test.rb +24 -0
- data/test/johnson/conversions/string_test.rb +26 -0
- data/test/johnson/conversions/struct_test.rb +15 -0
- data/test/johnson/conversions/symbol_test.rb +19 -0
- data/test/johnson/conversions/thread_test.rb +24 -0
- data/test/johnson/error_test.rb +9 -0
- data/test/johnson/extensions_test.rb +56 -0
- data/test/johnson/nodes/array_literal_test.rb +57 -0
- data/test/johnson/nodes/array_node_test.rb +26 -0
- data/test/johnson/nodes/binary_node_test.rb +61 -0
- data/test/johnson/nodes/bracket_access_test.rb +16 -0
- data/test/johnson/nodes/delete_test.rb +11 -0
- data/test/johnson/nodes/do_while_test.rb +12 -0
- data/test/johnson/nodes/dot_accessor_test.rb +15 -0
- data/test/johnson/nodes/export_test.rb +9 -0
- data/test/johnson/nodes/for_test.rb +54 -0
- data/test/johnson/nodes/function_test.rb +71 -0
- data/test/johnson/nodes/if_test.rb +41 -0
- data/test/johnson/nodes/import_test.rb +13 -0
- data/test/johnson/nodes/label_test.rb +19 -0
- data/test/johnson/nodes/let_test.rb +31 -0
- data/test/johnson/nodes/object_literal_test.rb +110 -0
- data/test/johnson/nodes/return_test.rb +16 -0
- data/test/johnson/nodes/semi_test.rb +8 -0
- data/test/johnson/nodes/switch_test.rb +55 -0
- data/test/johnson/nodes/ternary_test.rb +25 -0
- data/test/johnson/nodes/throw_test.rb +9 -0
- data/test/johnson/nodes/try_node_test.rb +59 -0
- data/test/johnson/nodes/typeof_test.rb +11 -0
- data/test/johnson/nodes/unary_node_test.rb +23 -0
- data/test/johnson/nodes/void_test.rb +11 -0
- data/test/johnson/nodes/while_test.rb +26 -0
- data/test/johnson/nodes/with_test.rb +10 -0
- data/test/johnson/prelude_test.rb +56 -0
- data/test/johnson/runtime_test.rb +64 -0
- data/test/johnson/spidermonkey/context_test.rb +21 -0
- data/test/johnson/spidermonkey/immutable_node_test.rb +34 -0
- data/test/johnson/spidermonkey/js_land_proxy_test.rb +236 -0
- data/test/johnson/spidermonkey/ruby_land_proxy_test.rb +240 -0
- data/test/johnson/spidermonkey/runtime_test.rb +17 -0
- data/test/johnson/version_test.rb +13 -0
- data/test/johnson/visitors/dot_visitor_test.rb +39 -0
- data/test/johnson/visitors/enumerating_visitor_test.rb +12 -0
- data/test/johnson_test.rb +16 -0
- data/test/parser_test.rb +276 -0
- data/vendor/spidermonkey/.cvsignore +9 -0
- data/vendor/spidermonkey/Makefile.in +449 -0
- data/vendor/spidermonkey/Makefile.ref +365 -0
- data/vendor/spidermonkey/README.html +820 -0
- data/vendor/spidermonkey/SpiderMonkey.rsp +12 -0
- data/vendor/spidermonkey/Y.js +19 -0
- data/vendor/spidermonkey/build.mk +43 -0
- data/vendor/spidermonkey/config.mk +192 -0
- data/vendor/spidermonkey/config/AIX4.1.mk +65 -0
- data/vendor/spidermonkey/config/AIX4.2.mk +64 -0
- data/vendor/spidermonkey/config/AIX4.3.mk +65 -0
- data/vendor/spidermonkey/config/Darwin.mk +83 -0
- data/vendor/spidermonkey/config/Darwin1.3.mk +81 -0
- data/vendor/spidermonkey/config/Darwin1.4.mk +41 -0
- data/vendor/spidermonkey/config/Darwin5.2.mk +81 -0
- data/vendor/spidermonkey/config/Darwin5.3.mk +81 -0
- data/vendor/spidermonkey/config/HP-UXB.10.10.mk +77 -0
- data/vendor/spidermonkey/config/HP-UXB.10.20.mk +77 -0
- data/vendor/spidermonkey/config/HP-UXB.11.00.mk +80 -0
- data/vendor/spidermonkey/config/IRIX.mk +87 -0
- data/vendor/spidermonkey/config/IRIX5.3.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.1.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.2.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.3.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.5.mk +44 -0
- data/vendor/spidermonkey/config/Linux_All.mk +103 -0
- data/vendor/spidermonkey/config/Mac_OS10.0.mk +82 -0
- data/vendor/spidermonkey/config/OSF1V4.0.mk +72 -0
- data/vendor/spidermonkey/config/OSF1V5.0.mk +69 -0
- data/vendor/spidermonkey/config/SunOS4.1.4.mk +101 -0
- data/vendor/spidermonkey/config/SunOS5.10.mk +50 -0
- data/vendor/spidermonkey/config/SunOS5.3.mk +91 -0
- data/vendor/spidermonkey/config/SunOS5.4.mk +92 -0
- data/vendor/spidermonkey/config/SunOS5.5.1.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.5.mk +87 -0
- data/vendor/spidermonkey/config/SunOS5.6.mk +89 -0
- data/vendor/spidermonkey/config/SunOS5.7.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.8.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.9.mk +44 -0
- data/vendor/spidermonkey/config/WINNT4.0.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.0.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.1.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.2.mk +117 -0
- data/vendor/spidermonkey/config/WINNT6.0.mk +117 -0
- data/vendor/spidermonkey/config/dgux.mk +64 -0
- data/vendor/spidermonkey/editline/Makefile.ref +144 -0
- data/vendor/spidermonkey/editline/README +83 -0
- data/vendor/spidermonkey/editline/editline.3 +175 -0
- data/vendor/spidermonkey/editline/editline.c +1369 -0
- data/vendor/spidermonkey/editline/editline.h +135 -0
- data/vendor/spidermonkey/editline/sysunix.c +182 -0
- data/vendor/spidermonkey/editline/unix.h +82 -0
- data/vendor/spidermonkey/fdlibm/.cvsignore +7 -0
- data/vendor/spidermonkey/fdlibm/Makefile.in +127 -0
- data/vendor/spidermonkey/fdlibm/Makefile.ref +192 -0
- data/vendor/spidermonkey/fdlibm/e_acos.c +147 -0
- data/vendor/spidermonkey/fdlibm/e_acosh.c +105 -0
- data/vendor/spidermonkey/fdlibm/e_asin.c +156 -0
- data/vendor/spidermonkey/fdlibm/e_atan2.c +165 -0
- data/vendor/spidermonkey/fdlibm/e_atanh.c +110 -0
- data/vendor/spidermonkey/fdlibm/e_cosh.c +133 -0
- data/vendor/spidermonkey/fdlibm/e_exp.c +202 -0
- data/vendor/spidermonkey/fdlibm/e_fmod.c +184 -0
- data/vendor/spidermonkey/fdlibm/e_gamma.c +71 -0
- data/vendor/spidermonkey/fdlibm/e_gamma_r.c +70 -0
- data/vendor/spidermonkey/fdlibm/e_hypot.c +173 -0
- data/vendor/spidermonkey/fdlibm/e_j0.c +524 -0
- data/vendor/spidermonkey/fdlibm/e_j1.c +523 -0
- data/vendor/spidermonkey/fdlibm/e_jn.c +315 -0
- data/vendor/spidermonkey/fdlibm/e_lgamma.c +71 -0
- data/vendor/spidermonkey/fdlibm/e_lgamma_r.c +347 -0
- data/vendor/spidermonkey/fdlibm/e_log.c +184 -0
- data/vendor/spidermonkey/fdlibm/e_log10.c +134 -0
- data/vendor/spidermonkey/fdlibm/e_pow.c +386 -0
- data/vendor/spidermonkey/fdlibm/e_rem_pio2.c +222 -0
- data/vendor/spidermonkey/fdlibm/e_remainder.c +120 -0
- data/vendor/spidermonkey/fdlibm/e_scalb.c +89 -0
- data/vendor/spidermonkey/fdlibm/e_sinh.c +122 -0
- data/vendor/spidermonkey/fdlibm/e_sqrt.c +497 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.h +273 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.mak +1453 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.mdp +0 -0
- data/vendor/spidermonkey/fdlibm/k_cos.c +135 -0
- data/vendor/spidermonkey/fdlibm/k_rem_pio2.c +354 -0
- data/vendor/spidermonkey/fdlibm/k_sin.c +114 -0
- data/vendor/spidermonkey/fdlibm/k_standard.c +785 -0
- data/vendor/spidermonkey/fdlibm/k_tan.c +170 -0
- data/vendor/spidermonkey/fdlibm/s_asinh.c +101 -0
- data/vendor/spidermonkey/fdlibm/s_atan.c +175 -0
- data/vendor/spidermonkey/fdlibm/s_cbrt.c +133 -0
- data/vendor/spidermonkey/fdlibm/s_ceil.c +120 -0
- data/vendor/spidermonkey/fdlibm/s_copysign.c +72 -0
- data/vendor/spidermonkey/fdlibm/s_cos.c +118 -0
- data/vendor/spidermonkey/fdlibm/s_erf.c +356 -0
- data/vendor/spidermonkey/fdlibm/s_expm1.c +267 -0
- data/vendor/spidermonkey/fdlibm/s_fabs.c +70 -0
- data/vendor/spidermonkey/fdlibm/s_finite.c +71 -0
- data/vendor/spidermonkey/fdlibm/s_floor.c +121 -0
- data/vendor/spidermonkey/fdlibm/s_frexp.c +99 -0
- data/vendor/spidermonkey/fdlibm/s_ilogb.c +85 -0
- data/vendor/spidermonkey/fdlibm/s_isnan.c +74 -0
- data/vendor/spidermonkey/fdlibm/s_ldexp.c +66 -0
- data/vendor/spidermonkey/fdlibm/s_lib_version.c +73 -0
- data/vendor/spidermonkey/fdlibm/s_log1p.c +211 -0
- data/vendor/spidermonkey/fdlibm/s_logb.c +79 -0
- data/vendor/spidermonkey/fdlibm/s_matherr.c +64 -0
- data/vendor/spidermonkey/fdlibm/s_modf.c +132 -0
- data/vendor/spidermonkey/fdlibm/s_nextafter.c +124 -0
- data/vendor/spidermonkey/fdlibm/s_rint.c +131 -0
- data/vendor/spidermonkey/fdlibm/s_scalbn.c +107 -0
- data/vendor/spidermonkey/fdlibm/s_signgam.c +40 -0
- data/vendor/spidermonkey/fdlibm/s_significand.c +68 -0
- data/vendor/spidermonkey/fdlibm/s_sin.c +118 -0
- data/vendor/spidermonkey/fdlibm/s_tan.c +112 -0
- data/vendor/spidermonkey/fdlibm/s_tanh.c +122 -0
- data/vendor/spidermonkey/fdlibm/w_acos.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_acosh.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_asin.c +80 -0
- data/vendor/spidermonkey/fdlibm/w_atan2.c +79 -0
- data/vendor/spidermonkey/fdlibm/w_atanh.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_cosh.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_exp.c +88 -0
- data/vendor/spidermonkey/fdlibm/w_fmod.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_gamma.c +85 -0
- data/vendor/spidermonkey/fdlibm/w_gamma_r.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_hypot.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_j0.c +105 -0
- data/vendor/spidermonkey/fdlibm/w_j1.c +106 -0
- data/vendor/spidermonkey/fdlibm/w_jn.c +128 -0
- data/vendor/spidermonkey/fdlibm/w_lgamma.c +85 -0
- data/vendor/spidermonkey/fdlibm/w_lgamma_r.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_log.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_log10.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_pow.c +99 -0
- data/vendor/spidermonkey/fdlibm/w_remainder.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_scalb.c +95 -0
- data/vendor/spidermonkey/fdlibm/w_sinh.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_sqrt.c +77 -0
- data/vendor/spidermonkey/javascript-trace.d +73 -0
- data/vendor/spidermonkey/js.c +3951 -0
- data/vendor/spidermonkey/js.mdp +0 -0
- data/vendor/spidermonkey/js.msg +308 -0
- data/vendor/spidermonkey/js3240.rc +79 -0
- data/vendor/spidermonkey/jsOS240.def +654 -0
- data/vendor/spidermonkey/jsapi.c +5836 -0
- data/vendor/spidermonkey/jsapi.h +2624 -0
- data/vendor/spidermonkey/jsarena.c +450 -0
- data/vendor/spidermonkey/jsarena.h +318 -0
- data/vendor/spidermonkey/jsarray.c +2996 -0
- data/vendor/spidermonkey/jsarray.h +127 -0
- data/vendor/spidermonkey/jsatom.c +1045 -0
- data/vendor/spidermonkey/jsatom.h +442 -0
- data/vendor/spidermonkey/jsbit.h +253 -0
- data/vendor/spidermonkey/jsbool.c +176 -0
- data/vendor/spidermonkey/jsbool.h +73 -0
- data/vendor/spidermonkey/jsclist.h +139 -0
- data/vendor/spidermonkey/jscntxt.c +1348 -0
- data/vendor/spidermonkey/jscntxt.h +1120 -0
- data/vendor/spidermonkey/jscompat.h +57 -0
- data/vendor/spidermonkey/jsconfig.h +248 -0
- data/vendor/spidermonkey/jsconfig.mk +181 -0
- data/vendor/spidermonkey/jscpucfg.c +396 -0
- data/vendor/spidermonkey/jscpucfg.h +212 -0
- data/vendor/spidermonkey/jsdate.c +2390 -0
- data/vendor/spidermonkey/jsdate.h +124 -0
- data/vendor/spidermonkey/jsdbgapi.c +1802 -0
- data/vendor/spidermonkey/jsdbgapi.h +464 -0
- data/vendor/spidermonkey/jsdhash.c +868 -0
- data/vendor/spidermonkey/jsdhash.h +592 -0
- data/vendor/spidermonkey/jsdtoa.c +3167 -0
- data/vendor/spidermonkey/jsdtoa.h +130 -0
- data/vendor/spidermonkey/jsdtracef.c +317 -0
- data/vendor/spidermonkey/jsdtracef.h +77 -0
- data/vendor/spidermonkey/jsemit.c +6909 -0
- data/vendor/spidermonkey/jsemit.h +741 -0
- data/vendor/spidermonkey/jsexn.c +1371 -0
- data/vendor/spidermonkey/jsexn.h +96 -0
- data/vendor/spidermonkey/jsfile.c +2736 -0
- data/vendor/spidermonkey/jsfile.h +56 -0
- data/vendor/spidermonkey/jsfile.msg +90 -0
- data/vendor/spidermonkey/jsfun.c +2634 -0
- data/vendor/spidermonkey/jsfun.h +254 -0
- data/vendor/spidermonkey/jsgc.c +3562 -0
- data/vendor/spidermonkey/jsgc.h +403 -0
- data/vendor/spidermonkey/jshash.c +476 -0
- data/vendor/spidermonkey/jshash.h +151 -0
- data/vendor/spidermonkey/jsify.pl +485 -0
- data/vendor/spidermonkey/jsinterp.c +7007 -0
- data/vendor/spidermonkey/jsinterp.h +525 -0
- data/vendor/spidermonkey/jsinvoke.c +43 -0
- data/vendor/spidermonkey/jsiter.c +1067 -0
- data/vendor/spidermonkey/jsiter.h +122 -0
- data/vendor/spidermonkey/jskeyword.tbl +124 -0
- data/vendor/spidermonkey/jskwgen.c +460 -0
- data/vendor/spidermonkey/jslibmath.h +266 -0
- data/vendor/spidermonkey/jslock.c +1309 -0
- data/vendor/spidermonkey/jslock.h +313 -0
- data/vendor/spidermonkey/jslocko.asm +60 -0
- data/vendor/spidermonkey/jslog2.c +94 -0
- data/vendor/spidermonkey/jslong.c +264 -0
- data/vendor/spidermonkey/jslong.h +412 -0
- data/vendor/spidermonkey/jsmath.c +567 -0
- data/vendor/spidermonkey/jsmath.h +57 -0
- data/vendor/spidermonkey/jsnum.c +1239 -0
- data/vendor/spidermonkey/jsnum.h +283 -0
- data/vendor/spidermonkey/jsobj.c +5282 -0
- data/vendor/spidermonkey/jsobj.h +709 -0
- data/vendor/spidermonkey/jsopcode.c +5245 -0
- data/vendor/spidermonkey/jsopcode.h +394 -0
- data/vendor/spidermonkey/jsopcode.tbl +523 -0
- data/vendor/spidermonkey/jsotypes.h +202 -0
- data/vendor/spidermonkey/jsparse.c +6704 -0
- data/vendor/spidermonkey/jsparse.h +511 -0
- data/vendor/spidermonkey/jsprf.c +1262 -0
- data/vendor/spidermonkey/jsprf.h +150 -0
- data/vendor/spidermonkey/jsproto.tbl +128 -0
- data/vendor/spidermonkey/jsprvtd.h +267 -0
- data/vendor/spidermonkey/jspubtd.h +744 -0
- data/vendor/spidermonkey/jsregexp.c +4364 -0
- data/vendor/spidermonkey/jsregexp.h +183 -0
- data/vendor/spidermonkey/jsreops.tbl +145 -0
- data/vendor/spidermonkey/jsscan.c +2012 -0
- data/vendor/spidermonkey/jsscan.h +387 -0
- data/vendor/spidermonkey/jsscope.c +1957 -0
- data/vendor/spidermonkey/jsscope.h +418 -0
- data/vendor/spidermonkey/jsscript.c +1832 -0
- data/vendor/spidermonkey/jsscript.h +287 -0
- data/vendor/spidermonkey/jsshell.msg +50 -0
- data/vendor/spidermonkey/jsstddef.h +83 -0
- data/vendor/spidermonkey/jsstr.c +5005 -0
- data/vendor/spidermonkey/jsstr.h +641 -0
- data/vendor/spidermonkey/jstypes.h +475 -0
- data/vendor/spidermonkey/jsutil.c +345 -0
- data/vendor/spidermonkey/jsutil.h +157 -0
- data/vendor/spidermonkey/jsxdrapi.c +800 -0
- data/vendor/spidermonkey/jsxdrapi.h +218 -0
- data/vendor/spidermonkey/jsxml.c +8476 -0
- data/vendor/spidermonkey/jsxml.h +349 -0
- data/vendor/spidermonkey/lock_SunOS.s +119 -0
- data/vendor/spidermonkey/perfect.js +39 -0
- data/vendor/spidermonkey/plify_jsdhash.sed +36 -0
- data/vendor/spidermonkey/prmjtime.c +846 -0
- data/vendor/spidermonkey/prmjtime.h +103 -0
- data/vendor/spidermonkey/resource.h +15 -0
- data/vendor/spidermonkey/rules.mk +197 -0
- data/vendor/spidermonkey/win32.order +384 -0
- metadata +513 -0
@@ -0,0 +1,741 @@
|
|
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 jsemit_h___
|
42
|
+
#define jsemit_h___
|
43
|
+
/*
|
44
|
+
* JS bytecode generation.
|
45
|
+
*/
|
46
|
+
|
47
|
+
#include "jsstddef.h"
|
48
|
+
#include "jstypes.h"
|
49
|
+
#include "jsatom.h"
|
50
|
+
#include "jsopcode.h"
|
51
|
+
#include "jsscript.h"
|
52
|
+
#include "jsprvtd.h"
|
53
|
+
#include "jspubtd.h"
|
54
|
+
|
55
|
+
JS_BEGIN_EXTERN_C
|
56
|
+
|
57
|
+
/*
|
58
|
+
* NB: If you add enumerators for scope statements, add them between STMT_WITH
|
59
|
+
* and STMT_CATCH, or you will break the STMT_TYPE_IS_SCOPE macro. If you add
|
60
|
+
* non-looping statement enumerators, add them before STMT_DO_LOOP or you will
|
61
|
+
* break the STMT_TYPE_IS_LOOP macro.
|
62
|
+
*
|
63
|
+
* Also remember to keep the statementName array in jsemit.c in sync.
|
64
|
+
*/
|
65
|
+
typedef enum JSStmtType {
|
66
|
+
STMT_LABEL, /* labeled statement: L: s */
|
67
|
+
STMT_IF, /* if (then) statement */
|
68
|
+
STMT_ELSE, /* else clause of if statement */
|
69
|
+
STMT_BODY, /* synthetic body of function with
|
70
|
+
destructuring formal parameters */
|
71
|
+
STMT_BLOCK, /* compound statement: { s1[;... sN] } */
|
72
|
+
STMT_SWITCH, /* switch statement */
|
73
|
+
STMT_WITH, /* with statement */
|
74
|
+
STMT_CATCH, /* catch block */
|
75
|
+
STMT_TRY, /* try block */
|
76
|
+
STMT_FINALLY, /* finally block */
|
77
|
+
STMT_SUBROUTINE, /* gosub-target subroutine body */
|
78
|
+
STMT_DO_LOOP, /* do/while loop statement */
|
79
|
+
STMT_FOR_LOOP, /* for loop statement */
|
80
|
+
STMT_FOR_IN_LOOP, /* for/in loop statement */
|
81
|
+
STMT_WHILE_LOOP, /* while loop statement */
|
82
|
+
STMT_LIMIT
|
83
|
+
} JSStmtType;
|
84
|
+
|
85
|
+
#define STMT_TYPE_IN_RANGE(t,b,e) ((uint)((t) - (b)) <= (uintN)((e) - (b)))
|
86
|
+
|
87
|
+
/*
|
88
|
+
* A comment on the encoding of the JSStmtType enum and type-testing macros:
|
89
|
+
*
|
90
|
+
* STMT_TYPE_MAYBE_SCOPE tells whether a statement type is always, or may
|
91
|
+
* become, a lexical scope. It therefore includes block and switch (the two
|
92
|
+
* low-numbered "maybe" scope types) and excludes with (with has dynamic scope
|
93
|
+
* pending the "reformed with" in ES4/JS2). It includes all try-catch-finally
|
94
|
+
* types, which are high-numbered maybe-scope types.
|
95
|
+
*
|
96
|
+
* STMT_TYPE_LINKS_SCOPE tells whether a JSStmtInfo of the given type eagerly
|
97
|
+
* links to other scoping statement info records. It excludes the two early
|
98
|
+
* "maybe" types, block and switch, as well as the try and both finally types,
|
99
|
+
* since try and the other trailing maybe-scope types don't need block scope
|
100
|
+
* unless they contain let declarations.
|
101
|
+
*
|
102
|
+
* We treat WITH as a static scope because it prevents lexical binding from
|
103
|
+
* continuing further up the static scope chain. With the "reformed with"
|
104
|
+
* proposal for JS2, we'll be able to model it statically, too.
|
105
|
+
*/
|
106
|
+
#define STMT_TYPE_MAYBE_SCOPE(type) \
|
107
|
+
(type != STMT_WITH && \
|
108
|
+
STMT_TYPE_IN_RANGE(type, STMT_BLOCK, STMT_SUBROUTINE))
|
109
|
+
|
110
|
+
#define STMT_TYPE_LINKS_SCOPE(type) \
|
111
|
+
STMT_TYPE_IN_RANGE(type, STMT_WITH, STMT_CATCH)
|
112
|
+
|
113
|
+
#define STMT_TYPE_IS_TRYING(type) \
|
114
|
+
STMT_TYPE_IN_RANGE(type, STMT_TRY, STMT_SUBROUTINE)
|
115
|
+
|
116
|
+
#define STMT_TYPE_IS_LOOP(type) ((type) >= STMT_DO_LOOP)
|
117
|
+
|
118
|
+
#define STMT_MAYBE_SCOPE(stmt) STMT_TYPE_MAYBE_SCOPE((stmt)->type)
|
119
|
+
#define STMT_LINKS_SCOPE(stmt) (STMT_TYPE_LINKS_SCOPE((stmt)->type) || \
|
120
|
+
((stmt)->flags & SIF_SCOPE))
|
121
|
+
#define STMT_IS_TRYING(stmt) STMT_TYPE_IS_TRYING((stmt)->type)
|
122
|
+
#define STMT_IS_LOOP(stmt) STMT_TYPE_IS_LOOP((stmt)->type)
|
123
|
+
|
124
|
+
typedef struct JSStmtInfo JSStmtInfo;
|
125
|
+
|
126
|
+
struct JSStmtInfo {
|
127
|
+
uint16 type; /* statement type */
|
128
|
+
uint16 flags; /* flags, see below */
|
129
|
+
ptrdiff_t update; /* loop update offset (top if none) */
|
130
|
+
ptrdiff_t breaks; /* offset of last break in loop */
|
131
|
+
ptrdiff_t continues; /* offset of last continue in loop */
|
132
|
+
union {
|
133
|
+
JSAtom *label; /* name of LABEL */
|
134
|
+
JSObject *blockObj; /* block scope object */
|
135
|
+
} u;
|
136
|
+
JSStmtInfo *down; /* info for enclosing statement */
|
137
|
+
JSStmtInfo *downScope; /* next enclosing lexical scope */
|
138
|
+
};
|
139
|
+
|
140
|
+
#define SIF_SCOPE 0x0001 /* statement has its own lexical scope */
|
141
|
+
#define SIF_BODY_BLOCK 0x0002 /* STMT_BLOCK type is a function body */
|
142
|
+
#define SIF_FOR_BLOCK 0x0004 /* for (let ...) induced block scope */
|
143
|
+
|
144
|
+
/*
|
145
|
+
* To reuse space in JSStmtInfo, rename breaks and continues for use during
|
146
|
+
* try/catch/finally code generation and backpatching. To match most common
|
147
|
+
* use cases, the macro argument is a struct, not a struct pointer. Only a
|
148
|
+
* loop, switch, or label statement info record can have breaks and continues,
|
149
|
+
* and only a for loop has an update backpatch chain, so it's safe to overlay
|
150
|
+
* these for the "trying" JSStmtTypes.
|
151
|
+
*/
|
152
|
+
#define CATCHNOTE(stmt) ((stmt).update)
|
153
|
+
#define GOSUBS(stmt) ((stmt).breaks)
|
154
|
+
#define GUARDJUMP(stmt) ((stmt).continues)
|
155
|
+
|
156
|
+
#define AT_TOP_LEVEL(tc) \
|
157
|
+
(!(tc)->topStmt || ((tc)->topStmt->flags & SIF_BODY_BLOCK))
|
158
|
+
|
159
|
+
#define SET_STATEMENT_TOP(stmt, top) \
|
160
|
+
((stmt)->update = (top), (stmt)->breaks = (stmt)->continues = (-1))
|
161
|
+
|
162
|
+
struct JSTreeContext { /* tree context for semantic checks */
|
163
|
+
uint16 flags; /* statement state flags, see below */
|
164
|
+
uint16 ngvars; /* max. no. of global variables/regexps */
|
165
|
+
uint32 globalUses; /* optimizable global var uses in total */
|
166
|
+
uint32 loopyGlobalUses;/* optimizable global var uses in loops */
|
167
|
+
uint16 scopeDepth; /* current lexical scope chain depth */
|
168
|
+
uint16 maxScopeDepth; /* maximum lexical scope chain depth */
|
169
|
+
JSStmtInfo *topStmt; /* top of statement info stack */
|
170
|
+
JSStmtInfo *topScopeStmt; /* top lexical scope statement */
|
171
|
+
JSObject *blockChain; /* compile time block scope chain (NB: one
|
172
|
+
deeper than the topScopeStmt/downScope
|
173
|
+
chain when in head of let block/expr) */
|
174
|
+
JSParseNode *blockNode; /* parse node for a lexical scope.
|
175
|
+
XXX combine with blockChain? */
|
176
|
+
JSAtomList decls; /* function, const, and var declarations */
|
177
|
+
JSParseContext *parseContext;
|
178
|
+
JSFunction *fun; /* function to store argument and variable
|
179
|
+
names when flags & TCF_IN_FUNCTION */
|
180
|
+
};
|
181
|
+
|
182
|
+
#define TCF_IN_FUNCTION 0x01 /* parsing inside function body */
|
183
|
+
#define TCF_RETURN_EXPR 0x02 /* function has 'return expr;' */
|
184
|
+
#define TCF_RETURN_VOID 0x04 /* function has 'return;' */
|
185
|
+
#define TCF_IN_FOR_INIT 0x08 /* parsing init expr of for; exclude 'in' */
|
186
|
+
#define TCF_FUN_CLOSURE_VS_VAR 0x10 /* function and var with same name */
|
187
|
+
#define TCF_FUN_USES_NONLOCALS 0x20 /* function refers to non-local names */
|
188
|
+
#define TCF_FUN_HEAVYWEIGHT 0x40 /* function needs Call object per call */
|
189
|
+
#define TCF_FUN_IS_GENERATOR 0x80 /* parsed yield statement in function */
|
190
|
+
#define TCF_HAS_DEFXMLNS 0x100 /* default xml namespace = ...; parsed */
|
191
|
+
#define TCF_HAS_FUNCTION_STMT 0x200 /* block contains a function statement */
|
192
|
+
#define TCF_GENEXP_LAMBDA 0x400 /* flag lambda from generator expression */
|
193
|
+
#define TCF_COMPILE_N_GO 0x800 /* compiler-and-go mode of script, can
|
194
|
+
optimize name references based on scope
|
195
|
+
chain */
|
196
|
+
|
197
|
+
/*
|
198
|
+
* Flags to propagate out of the blocks.
|
199
|
+
*/
|
200
|
+
#define TCF_RETURN_FLAGS (TCF_RETURN_EXPR | TCF_RETURN_VOID)
|
201
|
+
|
202
|
+
/*
|
203
|
+
* Flags to propagate from FunctionBody.
|
204
|
+
*/
|
205
|
+
#define TCF_FUN_FLAGS (TCF_FUN_IS_GENERATOR | \
|
206
|
+
TCF_FUN_HEAVYWEIGHT | \
|
207
|
+
TCF_FUN_USES_NONLOCALS | \
|
208
|
+
TCF_FUN_CLOSURE_VS_VAR)
|
209
|
+
|
210
|
+
#define TREE_CONTEXT_INIT(tc, pc) \
|
211
|
+
((tc)->flags = (tc)->ngvars = 0, \
|
212
|
+
(tc)->globalUses = (tc)->loopyGlobalUses = 0, \
|
213
|
+
(tc)->scopeDepth = (tc)->maxScopeDepth = 0, \
|
214
|
+
(tc)->topStmt = (tc)->topScopeStmt = NULL, \
|
215
|
+
(tc)->blockChain = NULL, \
|
216
|
+
ATOM_LIST_INIT(&(tc)->decls), \
|
217
|
+
(tc)->blockNode = NULL, \
|
218
|
+
(tc)->parseContext = (pc), \
|
219
|
+
(tc)->fun = NULL)
|
220
|
+
|
221
|
+
#define TREE_CONTEXT_FINISH(tc) \
|
222
|
+
((void)0)
|
223
|
+
|
224
|
+
/*
|
225
|
+
* Span-dependent instructions are jumps whose span (from the jump bytecode to
|
226
|
+
* the jump target) may require 2 or 4 bytes of immediate operand.
|
227
|
+
*/
|
228
|
+
typedef struct JSSpanDep JSSpanDep;
|
229
|
+
typedef struct JSJumpTarget JSJumpTarget;
|
230
|
+
|
231
|
+
struct JSSpanDep {
|
232
|
+
ptrdiff_t top; /* offset of first bytecode in an opcode */
|
233
|
+
ptrdiff_t offset; /* offset - 1 within opcode of jump operand */
|
234
|
+
ptrdiff_t before; /* original offset - 1 of jump operand */
|
235
|
+
JSJumpTarget *target; /* tagged target pointer or backpatch delta */
|
236
|
+
};
|
237
|
+
|
238
|
+
/*
|
239
|
+
* Jump targets are stored in an AVL tree, for O(log(n)) lookup with targets
|
240
|
+
* sorted by offset from left to right, so that targets after a span-dependent
|
241
|
+
* instruction whose jump offset operand must be extended can be found quickly
|
242
|
+
* and adjusted upward (toward higher offsets).
|
243
|
+
*/
|
244
|
+
struct JSJumpTarget {
|
245
|
+
ptrdiff_t offset; /* offset of span-dependent jump target */
|
246
|
+
int balance; /* AVL tree balance number */
|
247
|
+
JSJumpTarget *kids[2]; /* left and right AVL tree child pointers */
|
248
|
+
};
|
249
|
+
|
250
|
+
#define JT_LEFT 0
|
251
|
+
#define JT_RIGHT 1
|
252
|
+
#define JT_OTHER_DIR(dir) (1 - (dir))
|
253
|
+
#define JT_IMBALANCE(dir) (((dir) << 1) - 1)
|
254
|
+
#define JT_DIR(imbalance) (((imbalance) + 1) >> 1)
|
255
|
+
|
256
|
+
/*
|
257
|
+
* Backpatch deltas are encoded in JSSpanDep.target if JT_TAG_BIT is clear,
|
258
|
+
* so we can maintain backpatch chains when using span dependency records to
|
259
|
+
* hold jump offsets that overflow 16 bits.
|
260
|
+
*/
|
261
|
+
#define JT_TAG_BIT ((jsword) 1)
|
262
|
+
#define JT_UNTAG_SHIFT 1
|
263
|
+
#define JT_SET_TAG(jt) ((JSJumpTarget *)((jsword)(jt) | JT_TAG_BIT))
|
264
|
+
#define JT_CLR_TAG(jt) ((JSJumpTarget *)((jsword)(jt) & ~JT_TAG_BIT))
|
265
|
+
#define JT_HAS_TAG(jt) ((jsword)(jt) & JT_TAG_BIT)
|
266
|
+
|
267
|
+
#define BITS_PER_PTRDIFF (sizeof(ptrdiff_t) * JS_BITS_PER_BYTE)
|
268
|
+
#define BITS_PER_BPDELTA (BITS_PER_PTRDIFF - 1 - JT_UNTAG_SHIFT)
|
269
|
+
#define BPDELTA_MAX (((ptrdiff_t)1 << BITS_PER_BPDELTA) - 1)
|
270
|
+
#define BPDELTA_TO_JT(bp) ((JSJumpTarget *)((bp) << JT_UNTAG_SHIFT))
|
271
|
+
#define JT_TO_BPDELTA(jt) ((ptrdiff_t)((jsword)(jt) >> JT_UNTAG_SHIFT))
|
272
|
+
|
273
|
+
#define SD_SET_TARGET(sd,jt) ((sd)->target = JT_SET_TAG(jt))
|
274
|
+
#define SD_GET_TARGET(sd) (JS_ASSERT(JT_HAS_TAG((sd)->target)), \
|
275
|
+
JT_CLR_TAG((sd)->target))
|
276
|
+
#define SD_SET_BPDELTA(sd,bp) ((sd)->target = BPDELTA_TO_JT(bp))
|
277
|
+
#define SD_GET_BPDELTA(sd) (JS_ASSERT(!JT_HAS_TAG((sd)->target)), \
|
278
|
+
JT_TO_BPDELTA((sd)->target))
|
279
|
+
|
280
|
+
/* Avoid asserting twice by expanding SD_GET_TARGET in the "then" clause. */
|
281
|
+
#define SD_SPAN(sd,pivot) (SD_GET_TARGET(sd) \
|
282
|
+
? JT_CLR_TAG((sd)->target)->offset - (pivot) \
|
283
|
+
: 0)
|
284
|
+
|
285
|
+
typedef struct JSTryNode JSTryNode;
|
286
|
+
|
287
|
+
struct JSTryNode {
|
288
|
+
JSTryNote note;
|
289
|
+
JSTryNode *prev;
|
290
|
+
};
|
291
|
+
|
292
|
+
typedef struct JSEmittedObjectList {
|
293
|
+
uint32 length; /* number of emitted so far objects */
|
294
|
+
JSParsedObjectBox *lastPob; /* last emitted object */
|
295
|
+
} JSEmittedObjectList;
|
296
|
+
|
297
|
+
extern void
|
298
|
+
FinishParsedObjects(JSEmittedObjectList *emittedList, JSObjectArray *objectMap);
|
299
|
+
|
300
|
+
struct JSCodeGenerator {
|
301
|
+
JSTreeContext treeContext; /* base state: statement info stack, etc. */
|
302
|
+
|
303
|
+
JSArenaPool *codePool; /* pointer to thread code arena pool */
|
304
|
+
JSArenaPool *notePool; /* pointer to thread srcnote arena pool */
|
305
|
+
void *codeMark; /* low watermark in cg->codePool */
|
306
|
+
void *noteMark; /* low watermark in cg->notePool */
|
307
|
+
|
308
|
+
struct {
|
309
|
+
jsbytecode *base; /* base of JS bytecode vector */
|
310
|
+
jsbytecode *limit; /* one byte beyond end of bytecode */
|
311
|
+
jsbytecode *next; /* pointer to next free bytecode */
|
312
|
+
jssrcnote *notes; /* source notes, see below */
|
313
|
+
uintN noteCount; /* number of source notes so far */
|
314
|
+
uintN noteMask; /* growth increment for notes */
|
315
|
+
ptrdiff_t lastNoteOffset; /* code offset for last source note */
|
316
|
+
uintN currentLine; /* line number for tree-based srcnote gen */
|
317
|
+
} prolog, main, *current;
|
318
|
+
|
319
|
+
uintN firstLine; /* first line, for js_NewScriptFromCG */
|
320
|
+
JSAtomList atomList; /* literals indexed for mapping */
|
321
|
+
|
322
|
+
intN stackDepth; /* current stack depth in script frame */
|
323
|
+
uintN maxStackDepth; /* maximum stack depth so far */
|
324
|
+
|
325
|
+
uintN ntrynotes; /* number of allocated so far try notes */
|
326
|
+
JSTryNode *lastTryNode; /* the last allocated try node */
|
327
|
+
|
328
|
+
JSSpanDep *spanDeps; /* span dependent instruction records */
|
329
|
+
JSJumpTarget *jumpTargets; /* AVL tree of jump target offsets */
|
330
|
+
JSJumpTarget *jtFreeList; /* JT_LEFT-linked list of free structs */
|
331
|
+
uintN numSpanDeps; /* number of span dependencies */
|
332
|
+
uintN numJumpTargets; /* number of jump targets */
|
333
|
+
ptrdiff_t spanDepTodo; /* offset from main.base of potentially
|
334
|
+
unoptimized spandeps */
|
335
|
+
|
336
|
+
uintN arrayCompSlot; /* stack slot of array in comprehension */
|
337
|
+
|
338
|
+
uintN emitLevel; /* js_EmitTree recursion level */
|
339
|
+
JSAtomList constList; /* compile time constants */
|
340
|
+
|
341
|
+
JSEmittedObjectList objectList; /* list of emitted so far objects */
|
342
|
+
JSEmittedObjectList regexpList; /* list of emitted so far regexp
|
343
|
+
that will be cloned during execution */
|
344
|
+
|
345
|
+
JSCodeGenerator *parent; /* enclosing function or global context */
|
346
|
+
};
|
347
|
+
|
348
|
+
#define CG_BASE(cg) ((cg)->current->base)
|
349
|
+
#define CG_LIMIT(cg) ((cg)->current->limit)
|
350
|
+
#define CG_NEXT(cg) ((cg)->current->next)
|
351
|
+
#define CG_CODE(cg,offset) (CG_BASE(cg) + (offset))
|
352
|
+
#define CG_OFFSET(cg) PTRDIFF(CG_NEXT(cg), CG_BASE(cg), jsbytecode)
|
353
|
+
|
354
|
+
#define CG_NOTES(cg) ((cg)->current->notes)
|
355
|
+
#define CG_NOTE_COUNT(cg) ((cg)->current->noteCount)
|
356
|
+
#define CG_NOTE_MASK(cg) ((cg)->current->noteMask)
|
357
|
+
#define CG_LAST_NOTE_OFFSET(cg) ((cg)->current->lastNoteOffset)
|
358
|
+
#define CG_CURRENT_LINE(cg) ((cg)->current->currentLine)
|
359
|
+
|
360
|
+
#define CG_PROLOG_BASE(cg) ((cg)->prolog.base)
|
361
|
+
#define CG_PROLOG_LIMIT(cg) ((cg)->prolog.limit)
|
362
|
+
#define CG_PROLOG_NEXT(cg) ((cg)->prolog.next)
|
363
|
+
#define CG_PROLOG_CODE(cg,poff) (CG_PROLOG_BASE(cg) + (poff))
|
364
|
+
#define CG_PROLOG_OFFSET(cg) PTRDIFF(CG_PROLOG_NEXT(cg), CG_PROLOG_BASE(cg),\
|
365
|
+
jsbytecode)
|
366
|
+
|
367
|
+
#define CG_SWITCH_TO_MAIN(cg) ((cg)->current = &(cg)->main)
|
368
|
+
#define CG_SWITCH_TO_PROLOG(cg) ((cg)->current = &(cg)->prolog)
|
369
|
+
|
370
|
+
/*
|
371
|
+
* Initialize cg to allocate bytecode space from codePool, source note space
|
372
|
+
* from notePool, and all other arena-allocated temporaries from cx->tempPool.
|
373
|
+
*/
|
374
|
+
extern JS_FRIEND_API(void)
|
375
|
+
js_InitCodeGenerator(JSContext *cx, JSCodeGenerator *cg, JSParseContext *pc,
|
376
|
+
JSArenaPool *codePool, JSArenaPool *notePool,
|
377
|
+
uintN lineno);
|
378
|
+
|
379
|
+
/*
|
380
|
+
* Release cg->codePool, cg->notePool, and cx->tempPool to marks set by
|
381
|
+
* js_InitCodeGenerator. Note that cgs are magic: they own the arena pool
|
382
|
+
* "tops-of-stack" space above their codeMark, noteMark, and tempMark points.
|
383
|
+
* This means you cannot alloc from tempPool and save the pointer beyond the
|
384
|
+
* next JS_FinishCodeGenerator.
|
385
|
+
*/
|
386
|
+
extern JS_FRIEND_API(void)
|
387
|
+
js_FinishCodeGenerator(JSContext *cx, JSCodeGenerator *cg);
|
388
|
+
|
389
|
+
/*
|
390
|
+
* Emit one bytecode.
|
391
|
+
*/
|
392
|
+
extern ptrdiff_t
|
393
|
+
js_Emit1(JSContext *cx, JSCodeGenerator *cg, JSOp op);
|
394
|
+
|
395
|
+
/*
|
396
|
+
* Emit two bytecodes, an opcode (op) with a byte of immediate operand (op1).
|
397
|
+
*/
|
398
|
+
extern ptrdiff_t
|
399
|
+
js_Emit2(JSContext *cx, JSCodeGenerator *cg, JSOp op, jsbytecode op1);
|
400
|
+
|
401
|
+
/*
|
402
|
+
* Emit three bytecodes, an opcode with two bytes of immediate operands.
|
403
|
+
*/
|
404
|
+
extern ptrdiff_t
|
405
|
+
js_Emit3(JSContext *cx, JSCodeGenerator *cg, JSOp op, jsbytecode op1,
|
406
|
+
jsbytecode op2);
|
407
|
+
|
408
|
+
/*
|
409
|
+
* Emit (1 + extra) bytecodes, for N bytes of op and its immediate operand.
|
410
|
+
*/
|
411
|
+
extern ptrdiff_t
|
412
|
+
js_EmitN(JSContext *cx, JSCodeGenerator *cg, JSOp op, size_t extra);
|
413
|
+
|
414
|
+
/*
|
415
|
+
* Unsafe macro to call js_SetJumpOffset and return false if it does.
|
416
|
+
*/
|
417
|
+
#define CHECK_AND_SET_JUMP_OFFSET(cx,cg,pc,off) \
|
418
|
+
JS_BEGIN_MACRO \
|
419
|
+
if (!js_SetJumpOffset(cx, cg, pc, off)) \
|
420
|
+
return JS_FALSE; \
|
421
|
+
JS_END_MACRO
|
422
|
+
|
423
|
+
#define CHECK_AND_SET_JUMP_OFFSET_AT(cx,cg,off) \
|
424
|
+
CHECK_AND_SET_JUMP_OFFSET(cx, cg, CG_CODE(cg,off), CG_OFFSET(cg) - (off))
|
425
|
+
|
426
|
+
extern JSBool
|
427
|
+
js_SetJumpOffset(JSContext *cx, JSCodeGenerator *cg, jsbytecode *pc,
|
428
|
+
ptrdiff_t off);
|
429
|
+
|
430
|
+
/* Test whether we're in a statement of given type. */
|
431
|
+
extern JSBool
|
432
|
+
js_InStatement(JSTreeContext *tc, JSStmtType type);
|
433
|
+
|
434
|
+
/* Test whether we're in a with statement. */
|
435
|
+
#define js_InWithStatement(tc) js_InStatement(tc, STMT_WITH)
|
436
|
+
|
437
|
+
/*
|
438
|
+
* Test whether atom refers to a global variable (or is a reference error).
|
439
|
+
* Return true in *loopyp if any loops enclose the lexical reference, false
|
440
|
+
* otherwise.
|
441
|
+
*/
|
442
|
+
extern JSBool
|
443
|
+
js_IsGlobalReference(JSTreeContext *tc, JSAtom *atom, JSBool *loopyp);
|
444
|
+
|
445
|
+
/*
|
446
|
+
* Push the C-stack-allocated struct at stmt onto the stmtInfo stack.
|
447
|
+
*/
|
448
|
+
extern void
|
449
|
+
js_PushStatement(JSTreeContext *tc, JSStmtInfo *stmt, JSStmtType type,
|
450
|
+
ptrdiff_t top);
|
451
|
+
|
452
|
+
/*
|
453
|
+
* Push a block scope statement and link blockObj into tc->blockChain. To pop
|
454
|
+
* this statement info record, use js_PopStatement as usual, or if appropriate
|
455
|
+
* (if generating code), js_PopStatementCG.
|
456
|
+
*/
|
457
|
+
extern void
|
458
|
+
js_PushBlockScope(JSTreeContext *tc, JSStmtInfo *stmt, JSObject *blockObj,
|
459
|
+
ptrdiff_t top);
|
460
|
+
|
461
|
+
/*
|
462
|
+
* Pop tc->topStmt. If the top JSStmtInfo struct is not stack-allocated, it
|
463
|
+
* is up to the caller to free it.
|
464
|
+
*/
|
465
|
+
extern void
|
466
|
+
js_PopStatement(JSTreeContext *tc);
|
467
|
+
|
468
|
+
/*
|
469
|
+
* Like js_PopStatement(&cg->treeContext), also patch breaks and continues
|
470
|
+
* unless the top statement info record represents a try-catch-finally suite.
|
471
|
+
* May fail if a jump offset overflows.
|
472
|
+
*/
|
473
|
+
extern JSBool
|
474
|
+
js_PopStatementCG(JSContext *cx, JSCodeGenerator *cg);
|
475
|
+
|
476
|
+
/*
|
477
|
+
* Define and lookup a primitive jsval associated with the const named by atom.
|
478
|
+
* js_DefineCompileTimeConstant analyzes the constant-folded initializer at pn
|
479
|
+
* and saves the const's value in cg->constList, if it can be used at compile
|
480
|
+
* time. It returns true unless an error occurred.
|
481
|
+
*
|
482
|
+
* If the initializer's value could not be saved, js_DefineCompileTimeConstant
|
483
|
+
* calls will return the undefined value. js_DefineCompileTimeConstant tries
|
484
|
+
* to find a const value memorized for atom, returning true with *vp set to a
|
485
|
+
* value other than undefined if the constant was found, true with *vp set to
|
486
|
+
* JSVAL_VOID if not found, and false on error.
|
487
|
+
*/
|
488
|
+
extern JSBool
|
489
|
+
js_DefineCompileTimeConstant(JSContext *cx, JSCodeGenerator *cg, JSAtom *atom,
|
490
|
+
JSParseNode *pn);
|
491
|
+
|
492
|
+
/*
|
493
|
+
* Find a lexically scoped variable (one declared by let, catch, or an array
|
494
|
+
* comprehension) named by atom, looking in tc's compile-time scopes.
|
495
|
+
*
|
496
|
+
* If a WITH statement is reached along the scope stack, return its statement
|
497
|
+
* info record, so callers can tell that atom is ambiguous. If slotp is not
|
498
|
+
* null, then if atom is found, set *slotp to its stack slot, otherwise to -1.
|
499
|
+
* This means that if slotp is not null, all the block objects on the lexical
|
500
|
+
* scope chain must have had their depth slots computed by the code generator,
|
501
|
+
* so the caller must be under js_EmitTree.
|
502
|
+
*
|
503
|
+
* In any event, directly return the statement info record in which atom was
|
504
|
+
* found. Otherwise return null.
|
505
|
+
*/
|
506
|
+
extern JSStmtInfo *
|
507
|
+
js_LexicalLookup(JSTreeContext *tc, JSAtom *atom, jsint *slotp,
|
508
|
+
uintN decltype);
|
509
|
+
|
510
|
+
/*
|
511
|
+
* Emit code into cg for the tree rooted at pn.
|
512
|
+
*/
|
513
|
+
extern JSBool
|
514
|
+
js_EmitTree(JSContext *cx, JSCodeGenerator *cg, JSParseNode *pn);
|
515
|
+
|
516
|
+
/*
|
517
|
+
* Emit function code using cg for the tree rooted at body.
|
518
|
+
*/
|
519
|
+
extern JSBool
|
520
|
+
js_EmitFunctionScript(JSContext *cx, JSCodeGenerator *cg, JSParseNode *body);
|
521
|
+
|
522
|
+
/*
|
523
|
+
* Source notes generated along with bytecode for decompiling and debugging.
|
524
|
+
* A source note is a uint8 with 5 bits of type and 3 of offset from the pc of
|
525
|
+
* the previous note. If 3 bits of offset aren't enough, extended delta notes
|
526
|
+
* (SRC_XDELTA) consisting of 2 set high order bits followed by 6 offset bits
|
527
|
+
* are emitted before the next note. Some notes have operand offsets encoded
|
528
|
+
* immediately after them, in note bytes or byte-triples.
|
529
|
+
*
|
530
|
+
* Source Note Extended Delta
|
531
|
+
* +7-6-5-4-3+2-1-0+ +7-6-5+4-3-2-1-0+
|
532
|
+
* |note-type|delta| |1 1| ext-delta |
|
533
|
+
* +---------+-----+ +---+-----------+
|
534
|
+
*
|
535
|
+
* At most one "gettable" note (i.e., a note of type other than SRC_NEWLINE,
|
536
|
+
* SRC_SETLINE, and SRC_XDELTA) applies to a given bytecode.
|
537
|
+
*
|
538
|
+
* NB: the js_SrcNoteSpec array in jsemit.c is indexed by this enum, so its
|
539
|
+
* initializers need to match the order here.
|
540
|
+
*
|
541
|
+
* Note on adding new source notes: every pair of bytecodes (A, B) where A and
|
542
|
+
* B have disjoint sets of source notes that could apply to each bytecode may
|
543
|
+
* reuse the same note type value for two notes (snA, snB) that have the same
|
544
|
+
* arity, offsetBias, and isSpanDep initializers in js_SrcNoteSpec. This is
|
545
|
+
* why SRC_IF and SRC_INITPROP have the same value below. For bad historical
|
546
|
+
* reasons, some bytecodes below that could be overlayed have not been, but
|
547
|
+
* before using SRC_EXTENDED, consider compressing the existing note types.
|
548
|
+
*
|
549
|
+
* Don't forget to update JSXDR_BYTECODE_VERSION in jsxdrapi.h for all such
|
550
|
+
* incompatible source note or other bytecode changes.
|
551
|
+
*/
|
552
|
+
typedef enum JSSrcNoteType {
|
553
|
+
SRC_NULL = 0, /* terminates a note vector */
|
554
|
+
SRC_IF = 1, /* JSOP_IFEQ bytecode is from an if-then */
|
555
|
+
SRC_INITPROP = 1, /* disjoint meaning applied to JSOP_INITELEM or
|
556
|
+
to an index label in a regular (structuring)
|
557
|
+
or a destructuring object initialiser */
|
558
|
+
SRC_GENEXP = 1, /* JSOP_ANONFUNOBJ from generator expression */
|
559
|
+
SRC_IF_ELSE = 2, /* JSOP_IFEQ bytecode is from an if-then-else */
|
560
|
+
SRC_WHILE = 3, /* JSOP_IFEQ is from a while loop */
|
561
|
+
SRC_FOR = 4, /* JSOP_NOP or JSOP_POP in for loop head */
|
562
|
+
SRC_CONTINUE = 5, /* JSOP_GOTO is a continue, not a break;
|
563
|
+
also used on JSOP_ENDINIT if extra comma
|
564
|
+
at end of array literal: [1,2,,] */
|
565
|
+
SRC_DECL = 6, /* type of a declaration (var, const, let*) */
|
566
|
+
SRC_DESTRUCT = 6, /* JSOP_DUP starting a destructuring assignment
|
567
|
+
operation, with SRC_DECL_* offset operand */
|
568
|
+
SRC_PCDELTA = 7, /* distance forward from comma-operator to
|
569
|
+
next POP, or from CONDSWITCH to first CASE
|
570
|
+
opcode, etc. -- always a forward delta */
|
571
|
+
SRC_GROUPASSIGN = 7, /* SRC_DESTRUCT variant for [a, b] = [c, d] */
|
572
|
+
SRC_ASSIGNOP = 8, /* += or another assign-op follows */
|
573
|
+
SRC_COND = 9, /* JSOP_IFEQ is from conditional ?: operator */
|
574
|
+
SRC_BRACE = 10, /* mandatory brace, for scope or to avoid
|
575
|
+
dangling else */
|
576
|
+
SRC_HIDDEN = 11, /* opcode shouldn't be decompiled */
|
577
|
+
SRC_PCBASE = 12, /* distance back from annotated getprop or
|
578
|
+
setprop op to left-most obj.prop.subprop
|
579
|
+
bytecode -- always a backward delta */
|
580
|
+
SRC_LABEL = 13, /* JSOP_NOP for label: with atomid immediate */
|
581
|
+
SRC_LABELBRACE = 14, /* JSOP_NOP for label: {...} begin brace */
|
582
|
+
SRC_ENDBRACE = 15, /* JSOP_NOP for label: {...} end brace */
|
583
|
+
SRC_BREAK2LABEL = 16, /* JSOP_GOTO for 'break label' with atomid */
|
584
|
+
SRC_CONT2LABEL = 17, /* JSOP_GOTO for 'continue label' with atomid */
|
585
|
+
SRC_SWITCH = 18, /* JSOP_*SWITCH with offset to end of switch,
|
586
|
+
2nd off to first JSOP_CASE if condswitch */
|
587
|
+
SRC_FUNCDEF = 19, /* JSOP_NOP for function f() with atomid */
|
588
|
+
SRC_CATCH = 20, /* catch block has guard */
|
589
|
+
SRC_EXTENDED = 21, /* extended source note, 32-159, in next byte */
|
590
|
+
SRC_NEWLINE = 22, /* bytecode follows a source newline */
|
591
|
+
SRC_SETLINE = 23, /* a file-absolute source line number note */
|
592
|
+
SRC_XDELTA = 24 /* 24-31 are for extended delta notes */
|
593
|
+
} JSSrcNoteType;
|
594
|
+
|
595
|
+
/*
|
596
|
+
* Constants for the SRC_DECL source note. Note that span-dependent bytecode
|
597
|
+
* selection means that any SRC_DECL offset greater than SRC_DECL_LET may need
|
598
|
+
* to be adjusted, but these "offsets" are too small to span a span-dependent
|
599
|
+
* instruction, so can be used to denote distinct declaration syntaxes to the
|
600
|
+
* decompiler.
|
601
|
+
*
|
602
|
+
* NB: the var_prefix array in jsopcode.c depends on these dense indexes from
|
603
|
+
* SRC_DECL_VAR through SRC_DECL_LET.
|
604
|
+
*/
|
605
|
+
#define SRC_DECL_VAR 0
|
606
|
+
#define SRC_DECL_CONST 1
|
607
|
+
#define SRC_DECL_LET 2
|
608
|
+
#define SRC_DECL_NONE 3
|
609
|
+
|
610
|
+
#define SN_TYPE_BITS 5
|
611
|
+
#define SN_DELTA_BITS 3
|
612
|
+
#define SN_XDELTA_BITS 6
|
613
|
+
#define SN_TYPE_MASK (JS_BITMASK(SN_TYPE_BITS) << SN_DELTA_BITS)
|
614
|
+
#define SN_DELTA_MASK ((ptrdiff_t)JS_BITMASK(SN_DELTA_BITS))
|
615
|
+
#define SN_XDELTA_MASK ((ptrdiff_t)JS_BITMASK(SN_XDELTA_BITS))
|
616
|
+
|
617
|
+
#define SN_MAKE_NOTE(sn,t,d) (*(sn) = (jssrcnote) \
|
618
|
+
(((t) << SN_DELTA_BITS) \
|
619
|
+
| ((d) & SN_DELTA_MASK)))
|
620
|
+
#define SN_MAKE_XDELTA(sn,d) (*(sn) = (jssrcnote) \
|
621
|
+
((SRC_XDELTA << SN_DELTA_BITS) \
|
622
|
+
| ((d) & SN_XDELTA_MASK)))
|
623
|
+
|
624
|
+
#define SN_IS_XDELTA(sn) ((*(sn) >> SN_DELTA_BITS) >= SRC_XDELTA)
|
625
|
+
#define SN_TYPE(sn) ((JSSrcNoteType)(SN_IS_XDELTA(sn) \
|
626
|
+
? SRC_XDELTA \
|
627
|
+
: *(sn) >> SN_DELTA_BITS))
|
628
|
+
#define SN_SET_TYPE(sn,type) SN_MAKE_NOTE(sn, type, SN_DELTA(sn))
|
629
|
+
#define SN_IS_GETTABLE(sn) (SN_TYPE(sn) < SRC_NEWLINE)
|
630
|
+
|
631
|
+
#define SN_DELTA(sn) ((ptrdiff_t)(SN_IS_XDELTA(sn) \
|
632
|
+
? *(sn) & SN_XDELTA_MASK \
|
633
|
+
: *(sn) & SN_DELTA_MASK))
|
634
|
+
#define SN_SET_DELTA(sn,delta) (SN_IS_XDELTA(sn) \
|
635
|
+
? SN_MAKE_XDELTA(sn, delta) \
|
636
|
+
: SN_MAKE_NOTE(sn, SN_TYPE(sn), delta))
|
637
|
+
|
638
|
+
#define SN_DELTA_LIMIT ((ptrdiff_t)JS_BIT(SN_DELTA_BITS))
|
639
|
+
#define SN_XDELTA_LIMIT ((ptrdiff_t)JS_BIT(SN_XDELTA_BITS))
|
640
|
+
|
641
|
+
/*
|
642
|
+
* Offset fields follow certain notes and are frequency-encoded: an offset in
|
643
|
+
* [0,0x7f] consumes one byte, an offset in [0x80,0x7fffff] takes three, and
|
644
|
+
* the high bit of the first byte is set.
|
645
|
+
*/
|
646
|
+
#define SN_3BYTE_OFFSET_FLAG 0x80
|
647
|
+
#define SN_3BYTE_OFFSET_MASK 0x7f
|
648
|
+
|
649
|
+
typedef struct JSSrcNoteSpec {
|
650
|
+
const char *name; /* name for disassembly/debugging output */
|
651
|
+
uint8 arity; /* number of offset operands */
|
652
|
+
uint8 offsetBias; /* bias of offset(s) from annotated pc */
|
653
|
+
int8 isSpanDep; /* 1 or -1 if offsets could span extended ops,
|
654
|
+
0 otherwise; sign tells span direction */
|
655
|
+
} JSSrcNoteSpec;
|
656
|
+
|
657
|
+
extern JS_FRIEND_DATA(JSSrcNoteSpec) js_SrcNoteSpec[];
|
658
|
+
extern JS_FRIEND_API(uintN) js_SrcNoteLength(jssrcnote *sn);
|
659
|
+
|
660
|
+
#define SN_LENGTH(sn) ((js_SrcNoteSpec[SN_TYPE(sn)].arity == 0) ? 1 \
|
661
|
+
: js_SrcNoteLength(sn))
|
662
|
+
#define SN_NEXT(sn) ((sn) + SN_LENGTH(sn))
|
663
|
+
|
664
|
+
/* A source note array is terminated by an all-zero element. */
|
665
|
+
#define SN_MAKE_TERMINATOR(sn) (*(sn) = SRC_NULL)
|
666
|
+
#define SN_IS_TERMINATOR(sn) (*(sn) == SRC_NULL)
|
667
|
+
|
668
|
+
/*
|
669
|
+
* Append a new source note of the given type (and therefore size) to cg's
|
670
|
+
* notes dynamic array, updating cg->noteCount. Return the new note's index
|
671
|
+
* within the array pointed at by cg->current->notes. Return -1 if out of
|
672
|
+
* memory.
|
673
|
+
*/
|
674
|
+
extern intN
|
675
|
+
js_NewSrcNote(JSContext *cx, JSCodeGenerator *cg, JSSrcNoteType type);
|
676
|
+
|
677
|
+
extern intN
|
678
|
+
js_NewSrcNote2(JSContext *cx, JSCodeGenerator *cg, JSSrcNoteType type,
|
679
|
+
ptrdiff_t offset);
|
680
|
+
|
681
|
+
extern intN
|
682
|
+
js_NewSrcNote3(JSContext *cx, JSCodeGenerator *cg, JSSrcNoteType type,
|
683
|
+
ptrdiff_t offset1, ptrdiff_t offset2);
|
684
|
+
|
685
|
+
/*
|
686
|
+
* NB: this function can add at most one extra extended delta note.
|
687
|
+
*/
|
688
|
+
extern jssrcnote *
|
689
|
+
js_AddToSrcNoteDelta(JSContext *cx, JSCodeGenerator *cg, jssrcnote *sn,
|
690
|
+
ptrdiff_t delta);
|
691
|
+
|
692
|
+
/*
|
693
|
+
* Get and set the offset operand identified by which (0 for the first, etc.).
|
694
|
+
*/
|
695
|
+
extern JS_FRIEND_API(ptrdiff_t)
|
696
|
+
js_GetSrcNoteOffset(jssrcnote *sn, uintN which);
|
697
|
+
|
698
|
+
extern JSBool
|
699
|
+
js_SetSrcNoteOffset(JSContext *cx, JSCodeGenerator *cg, uintN index,
|
700
|
+
uintN which, ptrdiff_t offset);
|
701
|
+
|
702
|
+
/*
|
703
|
+
* Finish taking source notes in cx's notePool, copying final notes to the new
|
704
|
+
* stable store allocated by the caller and passed in via notes. Return false
|
705
|
+
* on malloc failure, which means this function reported an error.
|
706
|
+
*
|
707
|
+
* To compute the number of jssrcnotes to allocate and pass in via notes, use
|
708
|
+
* the CG_COUNT_FINAL_SRCNOTES macro. This macro knows a lot about details of
|
709
|
+
* js_FinishTakingSrcNotes, SO DON'T CHANGE jsemit.c's js_FinishTakingSrcNotes
|
710
|
+
* FUNCTION WITHOUT CHECKING WHETHER THIS MACRO NEEDS CORRESPONDING CHANGES!
|
711
|
+
*/
|
712
|
+
#define CG_COUNT_FINAL_SRCNOTES(cg, cnt) \
|
713
|
+
JS_BEGIN_MACRO \
|
714
|
+
ptrdiff_t diff_ = CG_PROLOG_OFFSET(cg) - (cg)->prolog.lastNoteOffset; \
|
715
|
+
cnt = (cg)->prolog.noteCount + (cg)->main.noteCount + 1; \
|
716
|
+
if ((cg)->prolog.noteCount && \
|
717
|
+
(cg)->prolog.currentLine != (cg)->firstLine) { \
|
718
|
+
if (diff_ > SN_DELTA_MASK) \
|
719
|
+
cnt += JS_HOWMANY(diff_ - SN_DELTA_MASK, SN_XDELTA_MASK); \
|
720
|
+
cnt += 2 + (((cg)->firstLine > SN_3BYTE_OFFSET_MASK) << 1); \
|
721
|
+
} else if (diff_ > 0) { \
|
722
|
+
if (cg->main.noteCount) { \
|
723
|
+
jssrcnote *sn_ = (cg)->main.notes; \
|
724
|
+
diff_ -= SN_IS_XDELTA(sn_) \
|
725
|
+
? SN_XDELTA_MASK - (*sn_ & SN_XDELTA_MASK) \
|
726
|
+
: SN_DELTA_MASK - (*sn_ & SN_DELTA_MASK); \
|
727
|
+
} \
|
728
|
+
if (diff_ > 0) \
|
729
|
+
cnt += JS_HOWMANY(diff_, SN_XDELTA_MASK); \
|
730
|
+
} \
|
731
|
+
JS_END_MACRO
|
732
|
+
|
733
|
+
extern JSBool
|
734
|
+
js_FinishTakingSrcNotes(JSContext *cx, JSCodeGenerator *cg, jssrcnote *notes);
|
735
|
+
|
736
|
+
extern void
|
737
|
+
js_FinishTakingTryNotes(JSCodeGenerator *cg, JSTryNoteArray *array);
|
738
|
+
|
739
|
+
JS_END_EXTERN_C
|
740
|
+
|
741
|
+
#endif /* jsemit_h___ */
|