jbarnette-johnson 1.0.0.200806240111 → 1.0.0.200807291507
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/MANIFEST +1 -0
- data/Rakefile +3 -10
- data/bin/johnson +2 -1
- data/ext/spidermonkey/context.c +3 -4
- data/ext/spidermonkey/context.h +1 -1
- data/ext/spidermonkey/conversions.c +39 -33
- data/ext/spidermonkey/debugger.c +5 -5
- data/ext/spidermonkey/immutable_node.c.erb +11 -11
- data/ext/spidermonkey/jroot.h +4 -4
- data/ext/spidermonkey/js_land_proxy.c +9 -8
- data/ext/spidermonkey/ruby_land_proxy.c +5 -4
- data/ext/spidermonkey/runtime.c +1 -1
- data/johnson.gemspec +36 -0
- data/lib/hoe.rb +0 -7
- data/lib/johnson/cli/options.rb +10 -4
- data/lib/johnson/spidermonkey/runtime.rb +2 -2
- data/lib/johnson/version.rb +4 -2
- data/lib/johnson.rb +1 -0
- data/test/johnson/runtime_test.rb +11 -0
- data/test/johnson/spidermonkey/ruby_land_proxy_test.rb +6 -0
- data/vendor/spidermonkey/.cvsignore +9 -0
- data/vendor/spidermonkey/Makefile.in +462 -0
- data/vendor/spidermonkey/Makefile.ref +364 -0
- data/vendor/spidermonkey/README.html +820 -0
- data/vendor/spidermonkey/SpiderMonkey.rsp +12 -0
- data/vendor/spidermonkey/Y.js +19 -0
- data/vendor/spidermonkey/build.mk +43 -0
- data/vendor/spidermonkey/config/AIX4.1.mk +65 -0
- data/vendor/spidermonkey/config/AIX4.2.mk +64 -0
- data/vendor/spidermonkey/config/AIX4.3.mk +65 -0
- data/vendor/spidermonkey/config/Darwin.mk +83 -0
- data/vendor/spidermonkey/config/Darwin1.3.mk +81 -0
- data/vendor/spidermonkey/config/Darwin1.4.mk +41 -0
- data/vendor/spidermonkey/config/Darwin5.2.mk +81 -0
- data/vendor/spidermonkey/config/Darwin5.3.mk +81 -0
- data/vendor/spidermonkey/config/HP-UXB.10.10.mk +77 -0
- data/vendor/spidermonkey/config/HP-UXB.10.20.mk +77 -0
- data/vendor/spidermonkey/config/HP-UXB.11.00.mk +80 -0
- data/vendor/spidermonkey/config/IRIX.mk +87 -0
- data/vendor/spidermonkey/config/IRIX5.3.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.1.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.2.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.3.mk +44 -0
- data/vendor/spidermonkey/config/IRIX6.5.mk +44 -0
- data/vendor/spidermonkey/config/Linux_All.mk +103 -0
- data/vendor/spidermonkey/config/Mac_OS10.0.mk +82 -0
- data/vendor/spidermonkey/config/OSF1V4.0.mk +72 -0
- data/vendor/spidermonkey/config/OSF1V5.0.mk +69 -0
- data/vendor/spidermonkey/config/SunOS4.1.4.mk +101 -0
- data/vendor/spidermonkey/config/SunOS5.10.mk +50 -0
- data/vendor/spidermonkey/config/SunOS5.3.mk +91 -0
- data/vendor/spidermonkey/config/SunOS5.4.mk +92 -0
- data/vendor/spidermonkey/config/SunOS5.5.1.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.5.mk +87 -0
- data/vendor/spidermonkey/config/SunOS5.6.mk +89 -0
- data/vendor/spidermonkey/config/SunOS5.7.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.8.mk +44 -0
- data/vendor/spidermonkey/config/SunOS5.9.mk +44 -0
- data/vendor/spidermonkey/config/WINNT4.0.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.0.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.1.mk +117 -0
- data/vendor/spidermonkey/config/WINNT5.2.mk +117 -0
- data/vendor/spidermonkey/config/WINNT6.0.mk +117 -0
- data/vendor/spidermonkey/config/dgux.mk +64 -0
- data/vendor/spidermonkey/config.mk +192 -0
- data/vendor/spidermonkey/editline/Makefile.ref +144 -0
- data/vendor/spidermonkey/editline/README +83 -0
- data/vendor/spidermonkey/editline/editline.3 +175 -0
- data/vendor/spidermonkey/editline/editline.c +1369 -0
- data/vendor/spidermonkey/editline/editline.h +135 -0
- data/vendor/spidermonkey/editline/sysunix.c +182 -0
- data/vendor/spidermonkey/editline/unix.h +82 -0
- data/vendor/spidermonkey/fdlibm/.cvsignore +7 -0
- data/vendor/spidermonkey/fdlibm/Makefile.in +127 -0
- data/vendor/spidermonkey/fdlibm/Makefile.ref +192 -0
- data/vendor/spidermonkey/fdlibm/e_acos.c +147 -0
- data/vendor/spidermonkey/fdlibm/e_acosh.c +105 -0
- data/vendor/spidermonkey/fdlibm/e_asin.c +156 -0
- data/vendor/spidermonkey/fdlibm/e_atan2.c +165 -0
- data/vendor/spidermonkey/fdlibm/e_atanh.c +110 -0
- data/vendor/spidermonkey/fdlibm/e_cosh.c +133 -0
- data/vendor/spidermonkey/fdlibm/e_exp.c +202 -0
- data/vendor/spidermonkey/fdlibm/e_fmod.c +184 -0
- data/vendor/spidermonkey/fdlibm/e_gamma.c +71 -0
- data/vendor/spidermonkey/fdlibm/e_gamma_r.c +70 -0
- data/vendor/spidermonkey/fdlibm/e_hypot.c +173 -0
- data/vendor/spidermonkey/fdlibm/e_j0.c +524 -0
- data/vendor/spidermonkey/fdlibm/e_j1.c +523 -0
- data/vendor/spidermonkey/fdlibm/e_jn.c +315 -0
- data/vendor/spidermonkey/fdlibm/e_lgamma.c +71 -0
- data/vendor/spidermonkey/fdlibm/e_lgamma_r.c +347 -0
- data/vendor/spidermonkey/fdlibm/e_log.c +184 -0
- data/vendor/spidermonkey/fdlibm/e_log10.c +134 -0
- data/vendor/spidermonkey/fdlibm/e_pow.c +386 -0
- data/vendor/spidermonkey/fdlibm/e_rem_pio2.c +222 -0
- data/vendor/spidermonkey/fdlibm/e_remainder.c +120 -0
- data/vendor/spidermonkey/fdlibm/e_scalb.c +89 -0
- data/vendor/spidermonkey/fdlibm/e_sinh.c +122 -0
- data/vendor/spidermonkey/fdlibm/e_sqrt.c +497 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.h +273 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.mak +1453 -0
- data/vendor/spidermonkey/fdlibm/fdlibm.mdp +0 -0
- data/vendor/spidermonkey/fdlibm/k_cos.c +135 -0
- data/vendor/spidermonkey/fdlibm/k_rem_pio2.c +354 -0
- data/vendor/spidermonkey/fdlibm/k_sin.c +114 -0
- data/vendor/spidermonkey/fdlibm/k_standard.c +785 -0
- data/vendor/spidermonkey/fdlibm/k_tan.c +170 -0
- data/vendor/spidermonkey/fdlibm/s_asinh.c +101 -0
- data/vendor/spidermonkey/fdlibm/s_atan.c +175 -0
- data/vendor/spidermonkey/fdlibm/s_cbrt.c +133 -0
- data/vendor/spidermonkey/fdlibm/s_ceil.c +120 -0
- data/vendor/spidermonkey/fdlibm/s_copysign.c +72 -0
- data/vendor/spidermonkey/fdlibm/s_cos.c +118 -0
- data/vendor/spidermonkey/fdlibm/s_erf.c +356 -0
- data/vendor/spidermonkey/fdlibm/s_expm1.c +267 -0
- data/vendor/spidermonkey/fdlibm/s_fabs.c +70 -0
- data/vendor/spidermonkey/fdlibm/s_finite.c +71 -0
- data/vendor/spidermonkey/fdlibm/s_floor.c +121 -0
- data/vendor/spidermonkey/fdlibm/s_frexp.c +99 -0
- data/vendor/spidermonkey/fdlibm/s_ilogb.c +85 -0
- data/vendor/spidermonkey/fdlibm/s_isnan.c +74 -0
- data/vendor/spidermonkey/fdlibm/s_ldexp.c +66 -0
- data/vendor/spidermonkey/fdlibm/s_lib_version.c +73 -0
- data/vendor/spidermonkey/fdlibm/s_log1p.c +211 -0
- data/vendor/spidermonkey/fdlibm/s_logb.c +79 -0
- data/vendor/spidermonkey/fdlibm/s_matherr.c +64 -0
- data/vendor/spidermonkey/fdlibm/s_modf.c +132 -0
- data/vendor/spidermonkey/fdlibm/s_nextafter.c +124 -0
- data/vendor/spidermonkey/fdlibm/s_rint.c +131 -0
- data/vendor/spidermonkey/fdlibm/s_scalbn.c +107 -0
- data/vendor/spidermonkey/fdlibm/s_signgam.c +40 -0
- data/vendor/spidermonkey/fdlibm/s_significand.c +68 -0
- data/vendor/spidermonkey/fdlibm/s_sin.c +118 -0
- data/vendor/spidermonkey/fdlibm/s_tan.c +112 -0
- data/vendor/spidermonkey/fdlibm/s_tanh.c +122 -0
- data/vendor/spidermonkey/fdlibm/w_acos.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_acosh.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_asin.c +80 -0
- data/vendor/spidermonkey/fdlibm/w_atan2.c +79 -0
- data/vendor/spidermonkey/fdlibm/w_atanh.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_cosh.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_exp.c +88 -0
- data/vendor/spidermonkey/fdlibm/w_fmod.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_gamma.c +85 -0
- data/vendor/spidermonkey/fdlibm/w_gamma_r.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_hypot.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_j0.c +105 -0
- data/vendor/spidermonkey/fdlibm/w_j1.c +106 -0
- data/vendor/spidermonkey/fdlibm/w_jn.c +128 -0
- data/vendor/spidermonkey/fdlibm/w_lgamma.c +85 -0
- data/vendor/spidermonkey/fdlibm/w_lgamma_r.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_log.c +78 -0
- data/vendor/spidermonkey/fdlibm/w_log10.c +81 -0
- data/vendor/spidermonkey/fdlibm/w_pow.c +99 -0
- data/vendor/spidermonkey/fdlibm/w_remainder.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_scalb.c +95 -0
- data/vendor/spidermonkey/fdlibm/w_sinh.c +77 -0
- data/vendor/spidermonkey/fdlibm/w_sqrt.c +77 -0
- data/vendor/spidermonkey/javascript-trace.d +73 -0
- data/vendor/spidermonkey/js.c +3951 -0
- data/vendor/spidermonkey/js.mak +4438 -0
- data/vendor/spidermonkey/js.mdp +0 -0
- data/vendor/spidermonkey/js.msg +307 -0
- data/vendor/spidermonkey/js.pkg +2 -0
- data/vendor/spidermonkey/js3240.rc +79 -0
- data/vendor/spidermonkey/jsOS240.def +654 -0
- data/vendor/spidermonkey/jsapi.c +5836 -0
- data/vendor/spidermonkey/jsapi.h +2624 -0
- data/vendor/spidermonkey/jsarena.c +450 -0
- data/vendor/spidermonkey/jsarena.h +318 -0
- data/vendor/spidermonkey/jsarray.c +2988 -0
- data/vendor/spidermonkey/jsarray.h +124 -0
- data/vendor/spidermonkey/jsatom.c +1045 -0
- data/vendor/spidermonkey/jsatom.h +442 -0
- data/vendor/spidermonkey/jsbit.h +253 -0
- data/vendor/spidermonkey/jsbool.c +176 -0
- data/vendor/spidermonkey/jsbool.h +73 -0
- data/vendor/spidermonkey/jsclist.h +139 -0
- data/vendor/spidermonkey/jscntxt.c +1348 -0
- data/vendor/spidermonkey/jscntxt.h +1120 -0
- data/vendor/spidermonkey/jscompat.h +57 -0
- data/vendor/spidermonkey/jsconfig.h +248 -0
- data/vendor/spidermonkey/jsconfig.mk +181 -0
- data/vendor/spidermonkey/jscpucfg.c +383 -0
- data/vendor/spidermonkey/jscpucfg.h +212 -0
- data/vendor/spidermonkey/jsdate.c +2398 -0
- data/vendor/spidermonkey/jsdate.h +124 -0
- data/vendor/spidermonkey/jsdbgapi.c +1799 -0
- data/vendor/spidermonkey/jsdbgapi.h +464 -0
- data/vendor/spidermonkey/jsdhash.c +868 -0
- data/vendor/spidermonkey/jsdhash.h +592 -0
- data/vendor/spidermonkey/jsdtoa.c +3167 -0
- data/vendor/spidermonkey/jsdtoa.h +130 -0
- data/vendor/spidermonkey/jsdtracef.c +317 -0
- data/vendor/spidermonkey/jsdtracef.h +77 -0
- data/vendor/spidermonkey/jsemit.c +6909 -0
- data/vendor/spidermonkey/jsemit.h +741 -0
- data/vendor/spidermonkey/jsexn.c +1371 -0
- data/vendor/spidermonkey/jsexn.h +96 -0
- data/vendor/spidermonkey/jsfile.c +2736 -0
- data/vendor/spidermonkey/jsfile.h +56 -0
- data/vendor/spidermonkey/jsfile.msg +90 -0
- data/vendor/spidermonkey/jsfun.c +2634 -0
- data/vendor/spidermonkey/jsfun.h +254 -0
- data/vendor/spidermonkey/jsgc.c +3554 -0
- data/vendor/spidermonkey/jsgc.h +403 -0
- data/vendor/spidermonkey/jshash.c +476 -0
- data/vendor/spidermonkey/jshash.h +151 -0
- data/vendor/spidermonkey/jsify.pl +485 -0
- data/vendor/spidermonkey/jsinterp.c +6981 -0
- data/vendor/spidermonkey/jsinterp.h +521 -0
- data/vendor/spidermonkey/jsinvoke.c +43 -0
- data/vendor/spidermonkey/jsiter.c +1067 -0
- data/vendor/spidermonkey/jsiter.h +122 -0
- data/vendor/spidermonkey/jskeyword.tbl +124 -0
- data/vendor/spidermonkey/jskwgen.c +460 -0
- data/vendor/spidermonkey/jslibmath.h +266 -0
- data/vendor/spidermonkey/jslock.c +1309 -0
- data/vendor/spidermonkey/jslock.h +313 -0
- data/vendor/spidermonkey/jslocko.asm +60 -0
- data/vendor/spidermonkey/jslog2.c +94 -0
- data/vendor/spidermonkey/jslong.c +264 -0
- data/vendor/spidermonkey/jslong.h +412 -0
- data/vendor/spidermonkey/jsmath.c +568 -0
- data/vendor/spidermonkey/jsmath.h +57 -0
- data/vendor/spidermonkey/jsnum.c +1228 -0
- data/vendor/spidermonkey/jsnum.h +283 -0
- data/vendor/spidermonkey/jsobj.c +5266 -0
- data/vendor/spidermonkey/jsobj.h +709 -0
- data/vendor/spidermonkey/jsopcode.c +5245 -0
- data/vendor/spidermonkey/jsopcode.h +394 -0
- data/vendor/spidermonkey/jsopcode.tbl +523 -0
- data/vendor/spidermonkey/jsotypes.h +202 -0
- data/vendor/spidermonkey/jsparse.c +6680 -0
- data/vendor/spidermonkey/jsparse.h +511 -0
- data/vendor/spidermonkey/jsprf.c +1262 -0
- data/vendor/spidermonkey/jsprf.h +150 -0
- data/vendor/spidermonkey/jsproto.tbl +128 -0
- data/vendor/spidermonkey/jsprvtd.h +267 -0
- data/vendor/spidermonkey/jspubtd.h +744 -0
- data/vendor/spidermonkey/jsregexp.c +4352 -0
- data/vendor/spidermonkey/jsregexp.h +183 -0
- data/vendor/spidermonkey/jsreops.tbl +145 -0
- data/vendor/spidermonkey/jsscan.c +2003 -0
- data/vendor/spidermonkey/jsscan.h +387 -0
- data/vendor/spidermonkey/jsscope.c +1948 -0
- data/vendor/spidermonkey/jsscope.h +418 -0
- data/vendor/spidermonkey/jsscript.c +1832 -0
- data/vendor/spidermonkey/jsscript.h +287 -0
- data/vendor/spidermonkey/jsshell.msg +50 -0
- data/vendor/spidermonkey/jsstddef.h +83 -0
- data/vendor/spidermonkey/jsstr.c +5004 -0
- data/vendor/spidermonkey/jsstr.h +641 -0
- data/vendor/spidermonkey/jstypes.h +475 -0
- data/vendor/spidermonkey/jsutil.c +345 -0
- data/vendor/spidermonkey/jsutil.h +157 -0
- data/vendor/spidermonkey/jsxdrapi.c +800 -0
- data/vendor/spidermonkey/jsxdrapi.h +218 -0
- data/vendor/spidermonkey/jsxml.c +8471 -0
- data/vendor/spidermonkey/jsxml.h +349 -0
- data/vendor/spidermonkey/lock_SunOS.s +119 -0
- data/vendor/spidermonkey/perfect.js +39 -0
- data/vendor/spidermonkey/plify_jsdhash.sed +36 -0
- data/vendor/spidermonkey/prmjtime.c +846 -0
- data/vendor/spidermonkey/prmjtime.h +103 -0
- data/vendor/spidermonkey/resource.h +15 -0
- data/vendor/spidermonkey/rules.mk +197 -0
- data/vendor/spidermonkey/win32.order +384 -0
- metadata +4 -3
@@ -0,0 +1,2988 @@
|
|
1
|
+
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2
|
+
* vim: set sw=4 ts=8 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 array class.
|
43
|
+
*
|
44
|
+
* Array objects begin as "dense" arrays, optimized for numeric-only property
|
45
|
+
* access over a vector of slots (obj->dslots) with high load factor. Array
|
46
|
+
* methods optimize for denseness by testing that the object's class is
|
47
|
+
* &js_ArrayClass, and can then directly manipulate the slots for efficiency.
|
48
|
+
*
|
49
|
+
* We track these pieces of metadata for arrays in dense mode:
|
50
|
+
* - the array's length property as a uint32, in JSSLOT_ARRAY_LENGTH,
|
51
|
+
* - the number of indices that are filled (non-holes), in JSSLOT_ARRAY_COUNT,
|
52
|
+
* - the net number of slots starting at dslots (DENSELEN), in dslots[-1] if
|
53
|
+
* dslots is non-NULL.
|
54
|
+
*
|
55
|
+
* In dense mode, holes in the array are represented by JSVAL_HOLE. The final
|
56
|
+
* slot in fslots (JSSLOT_ARRAY_LOOKUP_HOLDER) is used to store the single jsid
|
57
|
+
* "in use" by a lookupProperty caller.
|
58
|
+
*
|
59
|
+
* Arrays are converted to use js_SlowArrayClass when any of these conditions
|
60
|
+
* are met:
|
61
|
+
* - the load factor (COUNT / DENSELEN) is less than 0.25, and there are
|
62
|
+
* more than MIN_SPARSE_INDEX slots total
|
63
|
+
* - a property is set that is non-numeric (and not "length"); or
|
64
|
+
* - a hole is filled below DENSELEN (possibly implicitly through methods like
|
65
|
+
* |reverse| or |splice|).
|
66
|
+
*
|
67
|
+
* In the latter two cases, property creation order is no longer index order,
|
68
|
+
* which necessitates use of a structure that keeps track of property creation
|
69
|
+
* order. (ES4, due to expectations baked into web script, requires that
|
70
|
+
* enumeration order be the order in which properties were created.)
|
71
|
+
*
|
72
|
+
* An alternative in the latter case (out-of-order index set) would be to
|
73
|
+
* maintain the scope to track property enumeration order, but still use
|
74
|
+
* the fast slot access. That would have the same memory cost as just using
|
75
|
+
* a js_SlowArrayClass, but have the same performance characteristics as
|
76
|
+
* a dense array for slot accesses, at some cost in code complexity.
|
77
|
+
*/
|
78
|
+
#include "jsstddef.h"
|
79
|
+
#include <stdlib.h>
|
80
|
+
#include <string.h>
|
81
|
+
#include "jstypes.h"
|
82
|
+
#include "jsutil.h" /* Added by JSIFY */
|
83
|
+
#include "jsapi.h"
|
84
|
+
#include "jsarray.h"
|
85
|
+
#include "jsatom.h"
|
86
|
+
#include "jsbit.h"
|
87
|
+
#include "jsbool.h"
|
88
|
+
#include "jscntxt.h"
|
89
|
+
#include "jsconfig.h"
|
90
|
+
#include "jsdbgapi.h" /* for js_TraceWatchPoints */
|
91
|
+
#include "jsdtoa.h"
|
92
|
+
#include "jsfun.h"
|
93
|
+
#include "jsgc.h"
|
94
|
+
#include "jsinterp.h"
|
95
|
+
#include "jslock.h"
|
96
|
+
#include "jsnum.h"
|
97
|
+
#include "jsobj.h"
|
98
|
+
#include "jsscope.h"
|
99
|
+
#include "jsstr.h"
|
100
|
+
|
101
|
+
/* 2^32 - 1 as a number and a string */
|
102
|
+
#define MAXINDEX 4294967295u
|
103
|
+
#define MAXSTR "4294967295"
|
104
|
+
|
105
|
+
#define ARRAY_SET_DENSE_LENGTH(obj, max) \
|
106
|
+
(JS_ASSERT((obj)->dslots), (obj)->dslots[-1] = (jsval)(max))
|
107
|
+
|
108
|
+
/* Small arrays are dense, no matter what. */
|
109
|
+
#define MIN_SPARSE_INDEX 32
|
110
|
+
|
111
|
+
#define INDEX_TOO_BIG(index) ((index) > JS_BIT(29) - 1)
|
112
|
+
#define INDEX_TOO_SPARSE(array, index) \
|
113
|
+
(INDEX_TOO_BIG(index) || \
|
114
|
+
((index) > ARRAY_DENSE_LENGTH(array) && (index) >= MIN_SPARSE_INDEX && \
|
115
|
+
(index) > (uint32)((array)->fslots[JSSLOT_ARRAY_COUNT] + 1) * 4))
|
116
|
+
|
117
|
+
JS_STATIC_ASSERT(sizeof(JSScopeProperty) > 4 * sizeof(jsval));
|
118
|
+
|
119
|
+
static JSBool
|
120
|
+
MakeArraySlow(JSContext *cx, JSObject *obj);
|
121
|
+
|
122
|
+
#define ENSURE_SLOW_ARRAY(cx, obj) \
|
123
|
+
(OBJ_GET_CLASS(cx, obj) == &js_SlowArrayClass || MakeArraySlow(cx, obj))
|
124
|
+
|
125
|
+
/*
|
126
|
+
* Determine if the id represents an array index or an XML property index.
|
127
|
+
*
|
128
|
+
* An id is an array index according to ECMA by (15.4):
|
129
|
+
*
|
130
|
+
* "Array objects give special treatment to a certain class of property names.
|
131
|
+
* A property name P (in the form of a string value) is an array index if and
|
132
|
+
* only if ToString(ToUint32(P)) is equal to P and ToUint32(P) is not equal
|
133
|
+
* to 2^32-1."
|
134
|
+
*
|
135
|
+
* In our implementation, it would be sufficient to check for JSVAL_IS_INT(id)
|
136
|
+
* except that by using signed 32-bit integers we miss the top half of the
|
137
|
+
* valid range. This function checks the string representation itself; note
|
138
|
+
* that calling a standard conversion routine might allow strings such as
|
139
|
+
* "08" or "4.0" as array indices, which they are not.
|
140
|
+
*/
|
141
|
+
JSBool
|
142
|
+
js_IdIsIndex(jsval id, jsuint *indexp)
|
143
|
+
{
|
144
|
+
JSString *str;
|
145
|
+
jschar *cp;
|
146
|
+
|
147
|
+
if (JSVAL_IS_INT(id)) {
|
148
|
+
jsint i;
|
149
|
+
i = JSVAL_TO_INT(id);
|
150
|
+
if (i < 0)
|
151
|
+
return JS_FALSE;
|
152
|
+
*indexp = (jsuint)i;
|
153
|
+
return JS_TRUE;
|
154
|
+
}
|
155
|
+
|
156
|
+
/* NB: id should be a string, but jsxml.c may call us with an object id. */
|
157
|
+
if (!JSVAL_IS_STRING(id))
|
158
|
+
return JS_FALSE;
|
159
|
+
|
160
|
+
str = JSVAL_TO_STRING(id);
|
161
|
+
cp = JSSTRING_CHARS(str);
|
162
|
+
if (JS7_ISDEC(*cp) && JSSTRING_LENGTH(str) < sizeof(MAXSTR)) {
|
163
|
+
jsuint index = JS7_UNDEC(*cp++);
|
164
|
+
jsuint oldIndex = 0;
|
165
|
+
jsuint c = 0;
|
166
|
+
if (index != 0) {
|
167
|
+
while (JS7_ISDEC(*cp)) {
|
168
|
+
oldIndex = index;
|
169
|
+
c = JS7_UNDEC(*cp);
|
170
|
+
index = 10*index + c;
|
171
|
+
cp++;
|
172
|
+
}
|
173
|
+
}
|
174
|
+
|
175
|
+
/* Ensure that all characters were consumed and we didn't overflow. */
|
176
|
+
if (*cp == 0 &&
|
177
|
+
(oldIndex < (MAXINDEX / 10) ||
|
178
|
+
(oldIndex == (MAXINDEX / 10) && c < (MAXINDEX % 10))))
|
179
|
+
{
|
180
|
+
*indexp = index;
|
181
|
+
return JS_TRUE;
|
182
|
+
}
|
183
|
+
}
|
184
|
+
return JS_FALSE;
|
185
|
+
}
|
186
|
+
|
187
|
+
static jsuint
|
188
|
+
ValueIsLength(JSContext *cx, jsval* vp)
|
189
|
+
{
|
190
|
+
jsint i;
|
191
|
+
jsdouble d;
|
192
|
+
jsuint length;
|
193
|
+
|
194
|
+
if (JSVAL_IS_INT(*vp)) {
|
195
|
+
i = JSVAL_TO_INT(*vp);
|
196
|
+
if (i < 0)
|
197
|
+
goto error;
|
198
|
+
return (jsuint) i;
|
199
|
+
}
|
200
|
+
|
201
|
+
d = js_ValueToNumber(cx, vp);
|
202
|
+
if (JSVAL_IS_NULL(*vp))
|
203
|
+
goto error;
|
204
|
+
|
205
|
+
if (JSDOUBLE_IS_NaN(d))
|
206
|
+
goto error;
|
207
|
+
length = (jsuint) d;
|
208
|
+
if (d != (jsdouble) length)
|
209
|
+
goto error;
|
210
|
+
return length;
|
211
|
+
|
212
|
+
error:
|
213
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
214
|
+
JSMSG_BAD_ARRAY_LENGTH);
|
215
|
+
*vp = JSVAL_NULL;
|
216
|
+
return 0;
|
217
|
+
}
|
218
|
+
|
219
|
+
JSBool
|
220
|
+
js_GetLengthProperty(JSContext *cx, JSObject *obj, jsuint *lengthp)
|
221
|
+
{
|
222
|
+
JSTempValueRooter tvr;
|
223
|
+
jsid id;
|
224
|
+
JSBool ok;
|
225
|
+
jsint i;
|
226
|
+
|
227
|
+
if (OBJ_IS_ARRAY(cx, obj)) {
|
228
|
+
*lengthp = obj->fslots[JSSLOT_ARRAY_LENGTH];
|
229
|
+
return JS_TRUE;
|
230
|
+
}
|
231
|
+
|
232
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
233
|
+
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
234
|
+
ok = OBJ_GET_PROPERTY(cx, obj, id, &tvr.u.value);
|
235
|
+
if (ok) {
|
236
|
+
if (JSVAL_IS_INT(tvr.u.value)) {
|
237
|
+
i = JSVAL_TO_INT(tvr.u.value);
|
238
|
+
*lengthp = (jsuint)i; /* jsuint cast does ToUint32 */
|
239
|
+
} else {
|
240
|
+
*lengthp = js_ValueToECMAUint32(cx, &tvr.u.value);
|
241
|
+
ok = !JSVAL_IS_NULL(tvr.u.value);
|
242
|
+
}
|
243
|
+
}
|
244
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
245
|
+
return ok;
|
246
|
+
}
|
247
|
+
|
248
|
+
static JSBool
|
249
|
+
IndexToValue(JSContext *cx, jsuint index, jsval *vp)
|
250
|
+
{
|
251
|
+
if (index <= JSVAL_INT_MAX) {
|
252
|
+
*vp = INT_TO_JSVAL(index);
|
253
|
+
return JS_TRUE;
|
254
|
+
}
|
255
|
+
return JS_NewDoubleValue(cx, (jsdouble)index, vp);
|
256
|
+
}
|
257
|
+
|
258
|
+
static JSBool
|
259
|
+
IndexToId(JSContext *cx, jsuint index, jsid *idp)
|
260
|
+
{
|
261
|
+
JSString *str;
|
262
|
+
|
263
|
+
if (index <= JSVAL_INT_MAX) {
|
264
|
+
*idp = INT_TO_JSID(index);
|
265
|
+
return JS_TRUE;
|
266
|
+
}
|
267
|
+
str = js_NumberToString(cx, index);
|
268
|
+
if (!str)
|
269
|
+
return JS_FALSE;
|
270
|
+
return js_ValueToStringId(cx, STRING_TO_JSVAL(str), idp);
|
271
|
+
}
|
272
|
+
|
273
|
+
static JSBool
|
274
|
+
BigIndexToId(JSContext *cx, JSObject *obj, jsuint index, JSBool createAtom,
|
275
|
+
jsid *idp)
|
276
|
+
{
|
277
|
+
jschar buf[10], *start;
|
278
|
+
JSClass *clasp;
|
279
|
+
JSAtom *atom;
|
280
|
+
JS_STATIC_ASSERT((jsuint)-1 == 4294967295U);
|
281
|
+
|
282
|
+
JS_ASSERT(index > JSVAL_INT_MAX);
|
283
|
+
|
284
|
+
start = JS_ARRAY_END(buf);
|
285
|
+
do {
|
286
|
+
--start;
|
287
|
+
*start = (jschar)('0' + index % 10);
|
288
|
+
index /= 10;
|
289
|
+
} while (index != 0);
|
290
|
+
|
291
|
+
/*
|
292
|
+
* Skip the atomization if the class is known to store atoms corresponding
|
293
|
+
* to big indexes together with elements. In such case we know that the
|
294
|
+
* array does not have an element at the given index if its atom does not
|
295
|
+
* exist. Fast arrays (clasp == &js_ArrayClass) don't use atoms for
|
296
|
+
* any indexes, though it would be rare to see them have a big index
|
297
|
+
* in any case.
|
298
|
+
*/
|
299
|
+
if (!createAtom &&
|
300
|
+
((clasp = OBJ_GET_CLASS(cx, obj)) == &js_SlowArrayClass ||
|
301
|
+
clasp == &js_ArgumentsClass ||
|
302
|
+
clasp == &js_ObjectClass)) {
|
303
|
+
atom = js_GetExistingStringAtom(cx, start, JS_ARRAY_END(buf) - start);
|
304
|
+
if (!atom) {
|
305
|
+
*idp = JSVAL_VOID;
|
306
|
+
return JS_TRUE;
|
307
|
+
}
|
308
|
+
} else {
|
309
|
+
atom = js_AtomizeChars(cx, start, JS_ARRAY_END(buf) - start, 0);
|
310
|
+
if (!atom)
|
311
|
+
return JS_FALSE;
|
312
|
+
}
|
313
|
+
|
314
|
+
*idp = ATOM_TO_JSID(atom);
|
315
|
+
return JS_TRUE;
|
316
|
+
}
|
317
|
+
|
318
|
+
static JSBool
|
319
|
+
ResizeSlots(JSContext *cx, JSObject *obj, uint32 oldlen, uint32 len)
|
320
|
+
{
|
321
|
+
jsval *slots, *newslots;
|
322
|
+
|
323
|
+
if (len == 0) {
|
324
|
+
if (obj->dslots) {
|
325
|
+
JS_free(cx, obj->dslots - 1);
|
326
|
+
obj->dslots = NULL;
|
327
|
+
}
|
328
|
+
return JS_TRUE;
|
329
|
+
}
|
330
|
+
|
331
|
+
if (len > ~(uint32)0 / sizeof(jsval)) {
|
332
|
+
js_ReportAllocationOverflow(cx);
|
333
|
+
return JS_FALSE;
|
334
|
+
}
|
335
|
+
|
336
|
+
slots = obj->dslots ? obj->dslots - 1 : NULL;
|
337
|
+
newslots = (jsval *) JS_realloc(cx, slots, sizeof (jsval) * (len + 1));
|
338
|
+
if (!newslots)
|
339
|
+
return JS_FALSE;
|
340
|
+
|
341
|
+
obj->dslots = newslots + 1;
|
342
|
+
ARRAY_SET_DENSE_LENGTH(obj, len);
|
343
|
+
|
344
|
+
for (slots = obj->dslots + oldlen; slots < obj->dslots + len; slots++)
|
345
|
+
*slots = JSVAL_HOLE;
|
346
|
+
|
347
|
+
return JS_TRUE;
|
348
|
+
}
|
349
|
+
|
350
|
+
#define ARRAY_GROWBY 8
|
351
|
+
|
352
|
+
static JSBool
|
353
|
+
EnsureLength(JSContext *cx, JSObject *obj, uint32 len)
|
354
|
+
{
|
355
|
+
uint32 oldlen = ARRAY_DENSE_LENGTH(obj);
|
356
|
+
|
357
|
+
if (len > oldlen) {
|
358
|
+
return ResizeSlots(cx, obj, oldlen,
|
359
|
+
len + ARRAY_GROWBY - (len % ARRAY_GROWBY));
|
360
|
+
}
|
361
|
+
return JS_TRUE;
|
362
|
+
}
|
363
|
+
|
364
|
+
/*
|
365
|
+
* If the property at the given index exists, get its value into location
|
366
|
+
* pointed by vp and set *hole to false. Otherwise set *hole to true and *vp
|
367
|
+
* to JSVAL_VOID. This function assumes that the location pointed by vp is
|
368
|
+
* properly rooted and can be used as GC-protected storage for temporaries.
|
369
|
+
*/
|
370
|
+
static JSBool
|
371
|
+
GetArrayElement(JSContext *cx, JSObject *obj, jsuint index, JSBool *hole,
|
372
|
+
jsval *vp)
|
373
|
+
{
|
374
|
+
jsid id;
|
375
|
+
JSObject *obj2;
|
376
|
+
JSProperty *prop;
|
377
|
+
|
378
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj) && index < ARRAY_DENSE_LENGTH(obj) &&
|
379
|
+
(*vp = obj->dslots[index]) != JSVAL_HOLE) {
|
380
|
+
*hole = JS_FALSE;
|
381
|
+
return JS_TRUE;
|
382
|
+
}
|
383
|
+
|
384
|
+
if (index <= JSVAL_INT_MAX) {
|
385
|
+
id = INT_TO_JSID(index);
|
386
|
+
} else {
|
387
|
+
if (!BigIndexToId(cx, obj, index, JS_FALSE, &id))
|
388
|
+
return JS_FALSE;
|
389
|
+
if (id == JSVAL_VOID) {
|
390
|
+
*hole = JS_TRUE;
|
391
|
+
*vp = JSVAL_VOID;
|
392
|
+
return JS_TRUE;
|
393
|
+
}
|
394
|
+
}
|
395
|
+
|
396
|
+
if (!OBJ_LOOKUP_PROPERTY(cx, obj, id, &obj2, &prop))
|
397
|
+
return JS_FALSE;
|
398
|
+
if (!prop) {
|
399
|
+
*hole = JS_TRUE;
|
400
|
+
*vp = JSVAL_VOID;
|
401
|
+
} else {
|
402
|
+
OBJ_DROP_PROPERTY(cx, obj2, prop);
|
403
|
+
if (!OBJ_GET_PROPERTY(cx, obj, id, vp))
|
404
|
+
return JS_FALSE;
|
405
|
+
*hole = JS_FALSE;
|
406
|
+
}
|
407
|
+
return JS_TRUE;
|
408
|
+
}
|
409
|
+
|
410
|
+
/*
|
411
|
+
* Set the value of the property at the given index to v assuming v is rooted.
|
412
|
+
*/
|
413
|
+
static JSBool
|
414
|
+
SetArrayElement(JSContext *cx, JSObject *obj, jsuint index, jsval v)
|
415
|
+
{
|
416
|
+
jsid id;
|
417
|
+
|
418
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj)) {
|
419
|
+
if (INDEX_TOO_SPARSE(obj, index)) {
|
420
|
+
if (!MakeArraySlow(cx, obj))
|
421
|
+
return JS_FALSE;
|
422
|
+
} else {
|
423
|
+
|
424
|
+
if (!EnsureLength(cx, obj, index + 1))
|
425
|
+
return JS_FALSE;
|
426
|
+
if (index >= (uint32)obj->fslots[JSSLOT_ARRAY_LENGTH])
|
427
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = index + 1;
|
428
|
+
if (obj->dslots[index] == JSVAL_HOLE)
|
429
|
+
obj->fslots[JSSLOT_ARRAY_COUNT]++;
|
430
|
+
obj->dslots[index] = v;
|
431
|
+
return JS_TRUE;
|
432
|
+
}
|
433
|
+
}
|
434
|
+
|
435
|
+
if (index <= JSVAL_INT_MAX) {
|
436
|
+
id = INT_TO_JSID(index);
|
437
|
+
} else {
|
438
|
+
if (!BigIndexToId(cx, obj, index, JS_TRUE, &id))
|
439
|
+
return JS_FALSE;
|
440
|
+
JS_ASSERT(id != JSVAL_VOID);
|
441
|
+
}
|
442
|
+
return OBJ_SET_PROPERTY(cx, obj, id, &v);
|
443
|
+
}
|
444
|
+
|
445
|
+
static JSBool
|
446
|
+
DeleteArrayElement(JSContext *cx, JSObject *obj, jsuint index)
|
447
|
+
{
|
448
|
+
jsid id;
|
449
|
+
jsval junk;
|
450
|
+
|
451
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj)) {
|
452
|
+
if (index < ARRAY_DENSE_LENGTH(obj)) {
|
453
|
+
if (obj->dslots[index] != JSVAL_HOLE)
|
454
|
+
obj->fslots[JSSLOT_ARRAY_COUNT]--;
|
455
|
+
obj->dslots[index] = JSVAL_HOLE;
|
456
|
+
}
|
457
|
+
return JS_TRUE;
|
458
|
+
}
|
459
|
+
|
460
|
+
if (index <= JSVAL_INT_MAX) {
|
461
|
+
id = INT_TO_JSID(index);
|
462
|
+
} else {
|
463
|
+
if (!BigIndexToId(cx, obj, index, JS_FALSE, &id))
|
464
|
+
return JS_FALSE;
|
465
|
+
if (id == JSVAL_VOID)
|
466
|
+
return JS_TRUE;
|
467
|
+
}
|
468
|
+
return OBJ_DELETE_PROPERTY(cx, obj, id, &junk);
|
469
|
+
}
|
470
|
+
|
471
|
+
/*
|
472
|
+
* When hole is true, delete the property at the given index. Otherwise set
|
473
|
+
* its value to v assuming v is rooted.
|
474
|
+
*/
|
475
|
+
static JSBool
|
476
|
+
SetOrDeleteArrayElement(JSContext *cx, JSObject *obj, jsuint index,
|
477
|
+
JSBool hole, jsval v)
|
478
|
+
{
|
479
|
+
if (hole) {
|
480
|
+
JS_ASSERT(v == JSVAL_VOID);
|
481
|
+
return DeleteArrayElement(cx, obj, index);
|
482
|
+
}
|
483
|
+
return SetArrayElement(cx, obj, index, v);
|
484
|
+
}
|
485
|
+
|
486
|
+
JSBool
|
487
|
+
js_SetLengthProperty(JSContext *cx, JSObject *obj, jsuint length)
|
488
|
+
{
|
489
|
+
jsval v;
|
490
|
+
jsid id;
|
491
|
+
|
492
|
+
if (!IndexToValue(cx, length, &v))
|
493
|
+
return JS_FALSE;
|
494
|
+
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
495
|
+
return OBJ_SET_PROPERTY(cx, obj, id, &v);
|
496
|
+
}
|
497
|
+
|
498
|
+
JSBool
|
499
|
+
js_HasLengthProperty(JSContext *cx, JSObject *obj, jsuint *lengthp)
|
500
|
+
{
|
501
|
+
JSErrorReporter older;
|
502
|
+
JSTempValueRooter tvr;
|
503
|
+
jsid id;
|
504
|
+
JSBool ok;
|
505
|
+
|
506
|
+
older = JS_SetErrorReporter(cx, NULL);
|
507
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
508
|
+
id = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
509
|
+
ok = OBJ_GET_PROPERTY(cx, obj, id, &tvr.u.value);
|
510
|
+
JS_SetErrorReporter(cx, older);
|
511
|
+
if (ok) {
|
512
|
+
*lengthp = ValueIsLength(cx, &tvr.u.value);
|
513
|
+
ok = !JSVAL_IS_NULL(tvr.u.value);
|
514
|
+
}
|
515
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
516
|
+
return ok;
|
517
|
+
}
|
518
|
+
|
519
|
+
JSBool
|
520
|
+
js_IsArrayLike(JSContext *cx, JSObject *obj, JSBool *answerp, jsuint *lengthp)
|
521
|
+
{
|
522
|
+
JSClass *clasp;
|
523
|
+
|
524
|
+
clasp = OBJ_GET_CLASS(cx, obj);
|
525
|
+
*answerp = (clasp == &js_ArgumentsClass || clasp == &js_ArrayClass ||
|
526
|
+
clasp == &js_SlowArrayClass);
|
527
|
+
if (!*answerp) {
|
528
|
+
*lengthp = 0;
|
529
|
+
return JS_TRUE;
|
530
|
+
}
|
531
|
+
return js_GetLengthProperty(cx, obj, lengthp);
|
532
|
+
}
|
533
|
+
|
534
|
+
/*
|
535
|
+
* The 'length' property of all native Array instances is a shared permanent
|
536
|
+
* property of Array.prototype, so it appears to be a direct property of each
|
537
|
+
* array instance delegating to that Array.prototype. It accesses the private
|
538
|
+
* slot reserved by js_ArrayClass.
|
539
|
+
*
|
540
|
+
* Since SpiderMonkey supports cross-class prototype-based delegation, we have
|
541
|
+
* to be careful about the length getter and setter being called on an object
|
542
|
+
* not of Array class. For the getter, we search obj's prototype chain for the
|
543
|
+
* array that caused this getter to be invoked. In the setter case to overcome
|
544
|
+
* the JSPROP_SHARED attribute, we must define a shadowing length property.
|
545
|
+
*/
|
546
|
+
static JSBool
|
547
|
+
array_length_getter(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
548
|
+
{
|
549
|
+
do {
|
550
|
+
if (OBJ_IS_ARRAY(cx, obj))
|
551
|
+
return IndexToValue(cx, obj->fslots[JSSLOT_ARRAY_LENGTH], vp);
|
552
|
+
} while ((obj = OBJ_GET_PROTO(cx, obj)) != NULL);
|
553
|
+
return JS_TRUE;
|
554
|
+
}
|
555
|
+
|
556
|
+
static JSBool
|
557
|
+
array_length_setter(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
558
|
+
{
|
559
|
+
jsuint newlen, oldlen, gap, index;
|
560
|
+
jsval junk;
|
561
|
+
JSObject *iter;
|
562
|
+
JSTempValueRooter tvr;
|
563
|
+
JSBool ok;
|
564
|
+
|
565
|
+
if (!OBJ_IS_ARRAY(cx, obj)) {
|
566
|
+
jsid lengthId = ATOM_TO_JSID(cx->runtime->atomState.lengthAtom);
|
567
|
+
|
568
|
+
return OBJ_DEFINE_PROPERTY(cx, obj, lengthId, *vp, NULL, NULL,
|
569
|
+
JSPROP_ENUMERATE, NULL);
|
570
|
+
}
|
571
|
+
|
572
|
+
newlen = ValueIsLength(cx, vp);
|
573
|
+
if (JSVAL_IS_NULL(*vp))
|
574
|
+
return JS_FALSE;
|
575
|
+
oldlen = obj->fslots[JSSLOT_ARRAY_LENGTH];
|
576
|
+
|
577
|
+
if (oldlen == newlen)
|
578
|
+
return JS_TRUE;
|
579
|
+
|
580
|
+
if (!IndexToValue(cx, newlen, vp))
|
581
|
+
return JS_FALSE;
|
582
|
+
|
583
|
+
if (oldlen < newlen) {
|
584
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = newlen;
|
585
|
+
return JS_TRUE;
|
586
|
+
}
|
587
|
+
|
588
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj)) {
|
589
|
+
if (ARRAY_DENSE_LENGTH(obj) && !ResizeSlots(cx, obj, oldlen, newlen))
|
590
|
+
return JS_FALSE;
|
591
|
+
} else if (oldlen - newlen < (1 << 24)) {
|
592
|
+
do {
|
593
|
+
--oldlen;
|
594
|
+
if (!JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) ||
|
595
|
+
!DeleteArrayElement(cx, obj, oldlen)) {
|
596
|
+
return JS_FALSE;
|
597
|
+
}
|
598
|
+
} while (oldlen != newlen);
|
599
|
+
} else {
|
600
|
+
/*
|
601
|
+
* We are going to remove a lot of indexes in a presumably sparse
|
602
|
+
* array. So instead of looping through indexes between newlen and
|
603
|
+
* oldlen, we iterate through all properties and remove those that
|
604
|
+
* correspond to indexes in the half-open range [newlen, oldlen). See
|
605
|
+
* bug 322135.
|
606
|
+
*/
|
607
|
+
iter = JS_NewPropertyIterator(cx, obj);
|
608
|
+
if (!iter)
|
609
|
+
return JS_FALSE;
|
610
|
+
|
611
|
+
/* Protect iter against GC in OBJ_DELETE_PROPERTY. */
|
612
|
+
JS_PUSH_TEMP_ROOT_OBJECT(cx, iter, &tvr);
|
613
|
+
gap = oldlen - newlen;
|
614
|
+
for (;;) {
|
615
|
+
ok = (JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
616
|
+
JS_NextProperty(cx, iter, &id));
|
617
|
+
if (!ok)
|
618
|
+
break;
|
619
|
+
if (id == JSVAL_VOID)
|
620
|
+
break;
|
621
|
+
if (js_IdIsIndex(id, &index) && index - newlen < gap) {
|
622
|
+
ok = OBJ_DELETE_PROPERTY(cx, obj, id, &junk);
|
623
|
+
if (!ok)
|
624
|
+
break;
|
625
|
+
}
|
626
|
+
}
|
627
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
628
|
+
if (!ok)
|
629
|
+
return JS_FALSE;
|
630
|
+
}
|
631
|
+
|
632
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = newlen;
|
633
|
+
return JS_TRUE;
|
634
|
+
}
|
635
|
+
|
636
|
+
static JSBool
|
637
|
+
array_lookupProperty(JSContext *cx, JSObject *obj, jsid id, JSObject **objp,
|
638
|
+
JSProperty **propp)
|
639
|
+
{
|
640
|
+
uint32 i;
|
641
|
+
|
642
|
+
if (!OBJ_IS_DENSE_ARRAY(cx, obj))
|
643
|
+
return js_LookupProperty(cx, obj, id, objp, propp);
|
644
|
+
|
645
|
+
/*
|
646
|
+
* We have only indexed properties up to DENSELEN (excepting holes), plus
|
647
|
+
* the length property. For all else, we delegate to the prototype.
|
648
|
+
*/
|
649
|
+
if ((!js_IdIsIndex(id, &i) &&
|
650
|
+
id != ATOM_TO_JSID(cx->runtime->atomState.lengthAtom)) ||
|
651
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] == 0 ||
|
652
|
+
i >= ARRAY_DENSE_LENGTH(obj) ||
|
653
|
+
obj->dslots[i] == JSVAL_HOLE)
|
654
|
+
{
|
655
|
+
JSObject *proto = STOBJ_GET_PROTO(obj);
|
656
|
+
|
657
|
+
if (!proto) {
|
658
|
+
*objp = NULL;
|
659
|
+
*propp = NULL;
|
660
|
+
return JS_TRUE;
|
661
|
+
}
|
662
|
+
|
663
|
+
return OBJ_LOOKUP_PROPERTY(cx, proto, id, objp, propp);
|
664
|
+
}
|
665
|
+
|
666
|
+
/* FIXME 417501: threadsafety: could race with a lookup on another thread.
|
667
|
+
* If we can only have a single lookup active per context, we could
|
668
|
+
* pigeonhole this on the context instead. */
|
669
|
+
JS_ASSERT(STOBJ_GET_SLOT(obj, JSSLOT_ARRAY_LOOKUP_HOLDER) == JSVAL_VOID);
|
670
|
+
STOBJ_SET_SLOT(obj, JSSLOT_ARRAY_LOOKUP_HOLDER, (jsval)id);
|
671
|
+
*propp = (JSProperty *)&(obj->fslots[JSSLOT_ARRAY_LOOKUP_HOLDER]);
|
672
|
+
*objp = obj;
|
673
|
+
return JS_TRUE;
|
674
|
+
}
|
675
|
+
|
676
|
+
static void
|
677
|
+
array_dropProperty(JSContext *cx, JSObject *obj, JSProperty *prop)
|
678
|
+
{
|
679
|
+
JS_ASSERT(!OBJ_IS_DENSE_ARRAY(cx, obj) ||
|
680
|
+
STOBJ_GET_SLOT(obj, JSSLOT_ARRAY_LOOKUP_HOLDER) != JSVAL_VOID);
|
681
|
+
#ifdef DEBUG
|
682
|
+
STOBJ_SET_SLOT(obj, JSSLOT_ARRAY_LOOKUP_HOLDER, JSVAL_VOID);
|
683
|
+
#endif
|
684
|
+
}
|
685
|
+
|
686
|
+
static JSBool
|
687
|
+
array_getProperty(JSContext *cx, JSObject *obj, jsid id, jsval *vp)
|
688
|
+
{
|
689
|
+
uint32 i;
|
690
|
+
|
691
|
+
if (id == ATOM_TO_JSID(cx->runtime->atomState.lengthAtom))
|
692
|
+
return IndexToValue(cx, obj->fslots[JSSLOT_ARRAY_LENGTH], vp);
|
693
|
+
|
694
|
+
if (id == ATOM_TO_JSID(cx->runtime->atomState.protoAtom)) {
|
695
|
+
*vp = STOBJ_GET_SLOT(obj, JSSLOT_PROTO);
|
696
|
+
return JS_TRUE;
|
697
|
+
}
|
698
|
+
|
699
|
+
if (!OBJ_IS_DENSE_ARRAY(cx, obj))
|
700
|
+
return js_GetProperty(cx, obj, id, vp);
|
701
|
+
|
702
|
+
if (!js_IdIsIndex(ID_TO_VALUE(id), &i) || i >= ARRAY_DENSE_LENGTH(obj) ||
|
703
|
+
obj->dslots[i] == JSVAL_HOLE) {
|
704
|
+
JSObject *proto = STOBJ_GET_PROTO(obj);
|
705
|
+
if (!proto) {
|
706
|
+
*vp = JSVAL_VOID;
|
707
|
+
return JS_TRUE;
|
708
|
+
}
|
709
|
+
|
710
|
+
return OBJ_GET_PROPERTY(cx, proto, id, vp);
|
711
|
+
}
|
712
|
+
|
713
|
+
*vp = obj->dslots[i];
|
714
|
+
return JS_TRUE;
|
715
|
+
}
|
716
|
+
|
717
|
+
static JSBool
|
718
|
+
slowarray_addProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
|
719
|
+
{
|
720
|
+
jsuint index, length;
|
721
|
+
|
722
|
+
if (!js_IdIsIndex(id, &index))
|
723
|
+
return JS_TRUE;
|
724
|
+
length = obj->fslots[JSSLOT_ARRAY_LENGTH];
|
725
|
+
if (index >= length)
|
726
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = index + 1;
|
727
|
+
return JS_TRUE;
|
728
|
+
}
|
729
|
+
|
730
|
+
static void
|
731
|
+
slowarray_trace(JSTracer *trc, JSObject *obj)
|
732
|
+
{
|
733
|
+
uint32 length = obj->fslots[JSSLOT_ARRAY_LENGTH];
|
734
|
+
|
735
|
+
JS_ASSERT(STOBJ_GET_CLASS(obj) == &js_SlowArrayClass);
|
736
|
+
|
737
|
+
/*
|
738
|
+
* Move JSSLOT_ARRAY_LENGTH aside to prevent the GC from treating
|
739
|
+
* untagged integer values as objects or strings.
|
740
|
+
*/
|
741
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = JSVAL_VOID;
|
742
|
+
js_TraceObject(trc, obj);
|
743
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = length;
|
744
|
+
}
|
745
|
+
|
746
|
+
static JSObjectOps js_SlowArrayObjectOps;
|
747
|
+
|
748
|
+
static JSObjectOps *
|
749
|
+
slowarray_getObjectOps(JSContext *cx, JSClass *clasp)
|
750
|
+
{
|
751
|
+
return &js_SlowArrayObjectOps;
|
752
|
+
}
|
753
|
+
|
754
|
+
static JSBool
|
755
|
+
array_setProperty(JSContext *cx, JSObject *obj, jsid id, jsval *vp)
|
756
|
+
{
|
757
|
+
uint32 i;
|
758
|
+
|
759
|
+
if (id == ATOM_TO_JSID(cx->runtime->atomState.lengthAtom))
|
760
|
+
return array_length_setter(cx, obj, id, vp);
|
761
|
+
|
762
|
+
if (!OBJ_IS_DENSE_ARRAY(cx, obj))
|
763
|
+
return js_SetProperty(cx, obj, id, vp);
|
764
|
+
|
765
|
+
if (!js_IdIsIndex(id, &i) || INDEX_TOO_SPARSE(obj, i)) {
|
766
|
+
if (!MakeArraySlow(cx, obj))
|
767
|
+
return JS_FALSE;
|
768
|
+
return js_SetProperty(cx, obj, id, vp);
|
769
|
+
}
|
770
|
+
|
771
|
+
if (!EnsureLength(cx, obj, i + 1))
|
772
|
+
return JS_FALSE;
|
773
|
+
|
774
|
+
if (i >= (uint32)obj->fslots[JSSLOT_ARRAY_LENGTH])
|
775
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = i + 1;
|
776
|
+
if (obj->dslots[i] == JSVAL_HOLE)
|
777
|
+
obj->fslots[JSSLOT_ARRAY_COUNT]++;
|
778
|
+
obj->dslots[i] = *vp;
|
779
|
+
return JS_TRUE;
|
780
|
+
}
|
781
|
+
|
782
|
+
static JSBool
|
783
|
+
array_defineProperty(JSContext *cx, JSObject *obj, jsid id, jsval value,
|
784
|
+
JSPropertyOp getter, JSPropertyOp setter, uintN attrs,
|
785
|
+
JSProperty **propp)
|
786
|
+
{
|
787
|
+
uint32 i;
|
788
|
+
|
789
|
+
if (id == ATOM_TO_JSID(cx->runtime->atomState.lengthAtom))
|
790
|
+
return JS_TRUE;
|
791
|
+
|
792
|
+
if (!js_IdIsIndex(ID_TO_VALUE(id), &i) || attrs != JSPROP_ENUMERATE) {
|
793
|
+
if (!ENSURE_SLOW_ARRAY(cx, obj))
|
794
|
+
return JS_FALSE;
|
795
|
+
return js_DefineProperty(cx, obj, id, value, getter, setter, attrs,
|
796
|
+
propp);
|
797
|
+
}
|
798
|
+
|
799
|
+
return array_setProperty(cx, obj, id, &value);
|
800
|
+
}
|
801
|
+
|
802
|
+
static JSBool
|
803
|
+
array_getAttributes(JSContext *cx, JSObject *obj, jsid id, JSProperty *prop,
|
804
|
+
uintN *attrsp)
|
805
|
+
{
|
806
|
+
*attrsp = id == ATOM_TO_JSID(cx->runtime->atomState.lengthAtom)
|
807
|
+
? JSPROP_PERMANENT : JSPROP_ENUMERATE;
|
808
|
+
return JS_TRUE;
|
809
|
+
}
|
810
|
+
|
811
|
+
static JSBool
|
812
|
+
array_setAttributes(JSContext *cx, JSObject *obj, jsid id, JSProperty *prop,
|
813
|
+
uintN *attrsp)
|
814
|
+
{
|
815
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
816
|
+
JSMSG_CANT_SET_ARRAY_ATTRS);
|
817
|
+
return JS_FALSE;
|
818
|
+
}
|
819
|
+
|
820
|
+
static JSBool
|
821
|
+
array_deleteProperty(JSContext *cx, JSObject *obj, jsval id, jsval *rval)
|
822
|
+
{
|
823
|
+
uint32 i;
|
824
|
+
|
825
|
+
if (!OBJ_IS_DENSE_ARRAY(cx, obj))
|
826
|
+
return js_DeleteProperty(cx, obj, id, rval);
|
827
|
+
|
828
|
+
if (id == ATOM_TO_JSID(cx->runtime->atomState.lengthAtom)) {
|
829
|
+
*rval = JSVAL_FALSE;
|
830
|
+
return JS_TRUE;
|
831
|
+
}
|
832
|
+
|
833
|
+
if (js_IdIsIndex(id, &i) && i < ARRAY_DENSE_LENGTH(obj) &&
|
834
|
+
obj->dslots[i] != JSVAL_HOLE) {
|
835
|
+
obj->fslots[JSSLOT_ARRAY_COUNT]--;
|
836
|
+
obj->dslots[i] = JSVAL_HOLE;
|
837
|
+
}
|
838
|
+
|
839
|
+
*rval = JSVAL_TRUE;
|
840
|
+
return JS_TRUE;
|
841
|
+
}
|
842
|
+
|
843
|
+
/*
|
844
|
+
* JSObjectOps.enumerate implementation.
|
845
|
+
*
|
846
|
+
* For a fast array, JSENUMERATE_INIT captures in the enumeration state both
|
847
|
+
* the length of the array and the bitmap indicating the positions of holes in
|
848
|
+
* the array. This ensures that adding or deleting array elements does not
|
849
|
+
* affect the sequence of indexes JSENUMERATE_NEXT returns.
|
850
|
+
*
|
851
|
+
* For a common case of an array without holes, to represent the state we pack
|
852
|
+
* the (nextEnumerationIndex, arrayLength) pair as a pseudo-boolean jsval.
|
853
|
+
* This is possible when length <= PACKED_UINT_PAIR_BITS. For arrays with
|
854
|
+
* greater length or holes we allocate the JSIndexIterState structure and
|
855
|
+
* store it as an int-tagged private pointer jsval. For a slow array we
|
856
|
+
* delegate the enumeration implementation to js_Enumerate in
|
857
|
+
* slowarray_enumerate.
|
858
|
+
*
|
859
|
+
* Array mutations can turn a fast array into a slow one after the enumeration
|
860
|
+
* starts. When this happens, slowarray_enumerate receives a state created
|
861
|
+
* when the array was fast. To distinguish such fast state from a slow state,
|
862
|
+
* which is an int-tagged pointer that js_Enumerate creates, we set not one
|
863
|
+
* but two lowest bits when tagging a JSIndexIterState pointer -- see
|
864
|
+
* INDEX_ITER_TAG usage below. Thus, when slowarray_enumerate receives a state
|
865
|
+
* tagged with JSVAL_BOOLEAN or with two lowest bits set, it knows that this
|
866
|
+
* is a fast state so it calls array_enumerate to continue enumerating the
|
867
|
+
* indexes present in the original fast array.
|
868
|
+
*/
|
869
|
+
|
870
|
+
#define PACKED_UINT_PAIR_BITS 14
|
871
|
+
#define PACKED_UINT_PAIR_MASK JS_BITMASK(PACKED_UINT_PAIR_BITS)
|
872
|
+
|
873
|
+
#define UINT_PAIR_TO_BOOLEAN_JSVAL(i,j) \
|
874
|
+
(JS_ASSERT((uint32) (i) <= PACKED_UINT_PAIR_MASK), \
|
875
|
+
JS_ASSERT((uint32) (j) <= PACKED_UINT_PAIR_MASK), \
|
876
|
+
((jsval) (i) << (PACKED_UINT_PAIR_BITS + JSVAL_TAGBITS)) | \
|
877
|
+
((jsval) (j) << (JSVAL_TAGBITS)) | \
|
878
|
+
(jsval) JSVAL_BOOLEAN)
|
879
|
+
|
880
|
+
#define BOOLEAN_JSVAL_TO_UINT_PAIR(v,i,j) \
|
881
|
+
(JS_ASSERT(JSVAL_TAG(v) == JSVAL_BOOLEAN), \
|
882
|
+
(i) = (uint32) ((v) >> (PACKED_UINT_PAIR_BITS + JSVAL_TAGBITS)), \
|
883
|
+
(j) = (uint32) ((v) >> JSVAL_TAGBITS) & PACKED_UINT_PAIR_MASK, \
|
884
|
+
JS_ASSERT((i) <= PACKED_UINT_PAIR_MASK))
|
885
|
+
|
886
|
+
JS_STATIC_ASSERT(PACKED_UINT_PAIR_BITS * 2 + JSVAL_TAGBITS <= JS_BITS_PER_WORD);
|
887
|
+
|
888
|
+
typedef struct JSIndexIterState {
|
889
|
+
uint32 index;
|
890
|
+
uint32 length;
|
891
|
+
JSBool hasHoles;
|
892
|
+
|
893
|
+
/*
|
894
|
+
* Variable-length bitmap representing array's holes. It must not be
|
895
|
+
* accessed when hasHoles is false.
|
896
|
+
*/
|
897
|
+
jsbitmap holes[1];
|
898
|
+
} JSIndexIterState;
|
899
|
+
|
900
|
+
#define INDEX_ITER_TAG 3
|
901
|
+
|
902
|
+
JS_STATIC_ASSERT(JSVAL_INT == 1);
|
903
|
+
|
904
|
+
static JSBool
|
905
|
+
array_enumerate(JSContext *cx, JSObject *obj, JSIterateOp enum_op,
|
906
|
+
jsval *statep, jsid *idp)
|
907
|
+
{
|
908
|
+
uint32 length, i;
|
909
|
+
JSIndexIterState *ii;
|
910
|
+
|
911
|
+
switch (enum_op) {
|
912
|
+
case JSENUMERATE_INIT:
|
913
|
+
JS_ASSERT(OBJ_IS_DENSE_ARRAY(cx, obj));
|
914
|
+
length = ARRAY_DENSE_LENGTH(obj);
|
915
|
+
if (idp && !IndexToId(cx, length, idp))
|
916
|
+
return JS_FALSE;
|
917
|
+
ii = NULL;
|
918
|
+
for (i = 0; i != length; ++i) {
|
919
|
+
if (obj->dslots[i] == JSVAL_HOLE) {
|
920
|
+
if (!ii) {
|
921
|
+
ii = (JSIndexIterState *)
|
922
|
+
JS_malloc(cx, offsetof(JSIndexIterState, holes) +
|
923
|
+
JS_BITMAP_SIZE(length));
|
924
|
+
if (!ii)
|
925
|
+
return JS_FALSE;
|
926
|
+
ii->hasHoles = JS_TRUE;
|
927
|
+
memset(ii->holes, 0, JS_BITMAP_SIZE(length));
|
928
|
+
}
|
929
|
+
JS_SET_BIT(ii->holes, i);
|
930
|
+
}
|
931
|
+
}
|
932
|
+
if (!ii) {
|
933
|
+
/* Array has no holes. */
|
934
|
+
if (length <= PACKED_UINT_PAIR_MASK) {
|
935
|
+
*statep = UINT_PAIR_TO_BOOLEAN_JSVAL(0, length);
|
936
|
+
break;
|
937
|
+
}
|
938
|
+
ii = (JSIndexIterState *)
|
939
|
+
JS_malloc(cx, offsetof(JSIndexIterState, holes));
|
940
|
+
if (!ii)
|
941
|
+
return JS_FALSE;
|
942
|
+
ii->hasHoles = JS_FALSE;
|
943
|
+
}
|
944
|
+
ii->index = 0;
|
945
|
+
ii->length = length;
|
946
|
+
*statep = (jsval) ii | INDEX_ITER_TAG;
|
947
|
+
JS_ASSERT(*statep & JSVAL_INT);
|
948
|
+
break;
|
949
|
+
|
950
|
+
case JSENUMERATE_NEXT:
|
951
|
+
if (JSVAL_TAG(*statep) == JSVAL_BOOLEAN) {
|
952
|
+
BOOLEAN_JSVAL_TO_UINT_PAIR(*statep, i, length);
|
953
|
+
if (i != length) {
|
954
|
+
*idp = INT_TO_JSID(i);
|
955
|
+
*statep = UINT_PAIR_TO_BOOLEAN_JSVAL(i + 1, length);
|
956
|
+
break;
|
957
|
+
}
|
958
|
+
} else {
|
959
|
+
JS_ASSERT((*statep & INDEX_ITER_TAG) == INDEX_ITER_TAG);
|
960
|
+
ii = (JSIndexIterState *) (*statep & ~INDEX_ITER_TAG);
|
961
|
+
i = ii->index;
|
962
|
+
if (i != ii->length) {
|
963
|
+
/* Skip holes if any. */
|
964
|
+
if (ii->hasHoles) {
|
965
|
+
while (JS_TEST_BIT(ii->holes, i) && ++i != ii->length)
|
966
|
+
continue;
|
967
|
+
}
|
968
|
+
if (i != ii->length) {
|
969
|
+
ii->index = i + 1;
|
970
|
+
return IndexToId(cx, i, idp);
|
971
|
+
}
|
972
|
+
}
|
973
|
+
}
|
974
|
+
/* FALL THROUGH */
|
975
|
+
|
976
|
+
case JSENUMERATE_DESTROY:
|
977
|
+
if (JSVAL_TAG(*statep) != JSVAL_BOOLEAN) {
|
978
|
+
JS_ASSERT((*statep & INDEX_ITER_TAG) == INDEX_ITER_TAG);
|
979
|
+
ii = (JSIndexIterState *) (*statep & ~INDEX_ITER_TAG);
|
980
|
+
JS_free(cx, ii);
|
981
|
+
}
|
982
|
+
*statep = JSVAL_NULL;
|
983
|
+
break;
|
984
|
+
}
|
985
|
+
return JS_TRUE;
|
986
|
+
}
|
987
|
+
|
988
|
+
static JSBool
|
989
|
+
slowarray_enumerate(JSContext *cx, JSObject *obj, JSIterateOp enum_op,
|
990
|
+
jsval *statep, jsid *idp)
|
991
|
+
{
|
992
|
+
JSBool ok;
|
993
|
+
|
994
|
+
/* Are we continuing an enumeration that started when we were dense? */
|
995
|
+
if (enum_op != JSENUMERATE_INIT) {
|
996
|
+
if (JSVAL_TAG(*statep) == JSVAL_BOOLEAN ||
|
997
|
+
(*statep & INDEX_ITER_TAG) == INDEX_ITER_TAG) {
|
998
|
+
return array_enumerate(cx, obj, enum_op, statep, idp);
|
999
|
+
}
|
1000
|
+
JS_ASSERT((*statep & INDEX_ITER_TAG) == JSVAL_INT);
|
1001
|
+
}
|
1002
|
+
ok = js_Enumerate(cx, obj, enum_op, statep, idp);
|
1003
|
+
JS_ASSERT(*statep == JSVAL_NULL || (*statep & INDEX_ITER_TAG) == JSVAL_INT);
|
1004
|
+
return ok;
|
1005
|
+
}
|
1006
|
+
|
1007
|
+
static void
|
1008
|
+
array_finalize(JSContext *cx, JSObject *obj)
|
1009
|
+
{
|
1010
|
+
if (obj->dslots)
|
1011
|
+
JS_free(cx, obj->dslots - 1);
|
1012
|
+
obj->dslots = NULL;
|
1013
|
+
}
|
1014
|
+
|
1015
|
+
static void
|
1016
|
+
array_trace(JSTracer *trc, JSObject *obj)
|
1017
|
+
{
|
1018
|
+
uint32 length;
|
1019
|
+
size_t i;
|
1020
|
+
jsval v;
|
1021
|
+
|
1022
|
+
JS_ASSERT(OBJ_IS_DENSE_ARRAY(cx, obj));
|
1023
|
+
|
1024
|
+
length = ARRAY_DENSE_LENGTH(obj);
|
1025
|
+
for (i = 0; i < length; i++) {
|
1026
|
+
v = obj->dslots[i];
|
1027
|
+
if (JSVAL_IS_TRACEABLE(v)) {
|
1028
|
+
JS_SET_TRACING_INDEX(trc, "array_dslots", i);
|
1029
|
+
JS_CallTracer(trc, JSVAL_TO_TRACEABLE(v), JSVAL_TRACE_KIND(v));
|
1030
|
+
}
|
1031
|
+
}
|
1032
|
+
|
1033
|
+
for (i = JSSLOT_PROTO; i <= JSSLOT_PARENT; ++i) {
|
1034
|
+
v = STOBJ_GET_SLOT(obj, i);
|
1035
|
+
if (JSVAL_IS_TRACEABLE(v)) {
|
1036
|
+
JS_SET_TRACING_DETAILS(trc, js_PrintObjectSlotName, obj, i);
|
1037
|
+
JS_CallTracer(trc, JSVAL_TO_TRACEABLE(v), JSVAL_TRACE_KIND(v));
|
1038
|
+
}
|
1039
|
+
}
|
1040
|
+
}
|
1041
|
+
|
1042
|
+
static JSObjectMap *
|
1043
|
+
array_newObjectMap(JSContext *cx, jsrefcount nrefs, JSObjectOps *ops,
|
1044
|
+
JSClass *clasp, JSObject *obj)
|
1045
|
+
{
|
1046
|
+
#ifdef DEBUG
|
1047
|
+
extern JSClass js_ArrayClass;
|
1048
|
+
extern JSObjectOps js_ArrayObjectOps;
|
1049
|
+
#endif
|
1050
|
+
JSObjectMap *map = (JSObjectMap *) JS_malloc(cx, sizeof(*map));
|
1051
|
+
if (!map)
|
1052
|
+
return NULL;
|
1053
|
+
|
1054
|
+
map->nrefs = nrefs;
|
1055
|
+
JS_ASSERT(ops == &js_ArrayObjectOps);
|
1056
|
+
map->ops = ops;
|
1057
|
+
JS_ASSERT(clasp == &js_ArrayClass);
|
1058
|
+
map->freeslot = JSSLOT_FREE(clasp);
|
1059
|
+
|
1060
|
+
return map;
|
1061
|
+
}
|
1062
|
+
|
1063
|
+
void
|
1064
|
+
array_destroyObjectMap(JSContext *cx, JSObjectMap *map)
|
1065
|
+
{
|
1066
|
+
JS_free(cx, map);
|
1067
|
+
}
|
1068
|
+
|
1069
|
+
JSObjectOps js_ArrayObjectOps = {
|
1070
|
+
array_newObjectMap, array_destroyObjectMap,
|
1071
|
+
array_lookupProperty, array_defineProperty,
|
1072
|
+
array_getProperty, array_setProperty,
|
1073
|
+
array_getAttributes, array_setAttributes,
|
1074
|
+
array_deleteProperty, js_DefaultValue,
|
1075
|
+
array_enumerate, js_CheckAccess,
|
1076
|
+
NULL, array_dropProperty,
|
1077
|
+
NULL, NULL,
|
1078
|
+
NULL, js_HasInstance,
|
1079
|
+
js_SetProtoOrParent, js_SetProtoOrParent,
|
1080
|
+
array_trace, NULL,
|
1081
|
+
NULL, NULL
|
1082
|
+
};
|
1083
|
+
|
1084
|
+
static JSObjectOps *
|
1085
|
+
array_getObjectOps(JSContext *cx, JSClass *clasp)
|
1086
|
+
{
|
1087
|
+
return &js_ArrayObjectOps;
|
1088
|
+
}
|
1089
|
+
|
1090
|
+
JSClass js_ArrayClass = {
|
1091
|
+
"Array",
|
1092
|
+
JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_Array) |
|
1093
|
+
JSCLASS_HAS_RESERVED_SLOTS(1) | JSCLASS_NEW_ENUMERATE,
|
1094
|
+
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
|
1095
|
+
JS_EnumerateStub, JS_ResolveStub, js_TryValueOf, array_finalize,
|
1096
|
+
array_getObjectOps, NULL, NULL, NULL,
|
1097
|
+
NULL, NULL, NULL, NULL
|
1098
|
+
};
|
1099
|
+
|
1100
|
+
JSClass js_SlowArrayClass = {
|
1101
|
+
"Array",
|
1102
|
+
JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_Array),
|
1103
|
+
slowarray_addProperty, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
|
1104
|
+
JS_EnumerateStub, JS_ResolveStub, js_TryValueOf, JS_FinalizeStub,
|
1105
|
+
slowarray_getObjectOps, NULL, NULL, NULL,
|
1106
|
+
NULL, NULL, NULL, NULL
|
1107
|
+
};
|
1108
|
+
|
1109
|
+
/*
|
1110
|
+
* Convert an array object from fast-and-dense to slow-and-flexible.
|
1111
|
+
*/
|
1112
|
+
static JSBool
|
1113
|
+
MakeArraySlow(JSContext *cx, JSObject *obj)
|
1114
|
+
{
|
1115
|
+
JSObjectMap *map, *oldmap;
|
1116
|
+
uint32 i, length;
|
1117
|
+
|
1118
|
+
JS_ASSERT(OBJ_GET_CLASS(cx, obj) == &js_ArrayClass);
|
1119
|
+
|
1120
|
+
/* Create a native scope. */
|
1121
|
+
map = js_NewObjectMap(cx, obj->map->nrefs, &js_SlowArrayObjectOps,
|
1122
|
+
&js_SlowArrayClass, obj);
|
1123
|
+
if (!map)
|
1124
|
+
return JS_FALSE;
|
1125
|
+
|
1126
|
+
length = ARRAY_DENSE_LENGTH(obj);
|
1127
|
+
if (length) {
|
1128
|
+
map->freeslot = STOBJ_NSLOTS(obj) + JS_INITIAL_NSLOTS;
|
1129
|
+
obj->dslots[-1] = JS_INITIAL_NSLOTS + length;
|
1130
|
+
} else {
|
1131
|
+
map->freeslot = STOBJ_NSLOTS(obj);
|
1132
|
+
}
|
1133
|
+
|
1134
|
+
/* Create new properties pointing to existing values in dslots */
|
1135
|
+
for (i = 0; i < length; i++) {
|
1136
|
+
jsid id;
|
1137
|
+
JSScopeProperty *sprop;
|
1138
|
+
|
1139
|
+
if (!JS_ValueToId(cx, INT_TO_JSVAL(i), &id))
|
1140
|
+
goto out_bad;
|
1141
|
+
|
1142
|
+
if (obj->dslots[i] == JSVAL_HOLE) {
|
1143
|
+
obj->dslots[i] = JSVAL_VOID;
|
1144
|
+
continue;
|
1145
|
+
}
|
1146
|
+
|
1147
|
+
sprop = js_AddScopeProperty(cx, (JSScope *)map, id, NULL, NULL,
|
1148
|
+
i + JS_INITIAL_NSLOTS, JSPROP_ENUMERATE,
|
1149
|
+
0, 0);
|
1150
|
+
if (!sprop)
|
1151
|
+
goto out_bad;
|
1152
|
+
}
|
1153
|
+
|
1154
|
+
/* Render our formerly-reserved count property GC-safe. */
|
1155
|
+
obj->fslots[JSSLOT_ARRAY_COUNT] = JSVAL_VOID;
|
1156
|
+
|
1157
|
+
/* Make sure we preserve any flags borrowing bits in JSSLOT_CLASS. */
|
1158
|
+
obj->fslots[JSSLOT_CLASS] ^= (jsval) &js_ArrayClass;
|
1159
|
+
obj->fslots[JSSLOT_CLASS] |= (jsval) &js_SlowArrayClass;
|
1160
|
+
|
1161
|
+
/* Swap in our new map. */
|
1162
|
+
oldmap = obj->map;
|
1163
|
+
obj->map = map;
|
1164
|
+
array_destroyObjectMap(cx, oldmap);
|
1165
|
+
|
1166
|
+
return JS_TRUE;
|
1167
|
+
|
1168
|
+
out_bad:
|
1169
|
+
js_DestroyObjectMap(cx, map);
|
1170
|
+
return JS_FALSE;
|
1171
|
+
}
|
1172
|
+
|
1173
|
+
enum ArrayToStringOp {
|
1174
|
+
TO_STRING,
|
1175
|
+
TO_LOCALE_STRING,
|
1176
|
+
TO_SOURCE
|
1177
|
+
};
|
1178
|
+
|
1179
|
+
/*
|
1180
|
+
* When op is TO_STRING or TO_LOCALE_STRING sep indicates a separator to use
|
1181
|
+
* or "," when sep is NULL.
|
1182
|
+
* When op is TO_SOURCE sep must be NULL.
|
1183
|
+
*/
|
1184
|
+
static JSBool
|
1185
|
+
array_join_sub(JSContext *cx, JSObject *obj, enum ArrayToStringOp op,
|
1186
|
+
JSString *sep, jsval *rval)
|
1187
|
+
{
|
1188
|
+
JSBool ok, hole;
|
1189
|
+
jsuint length, index;
|
1190
|
+
jschar *chars, *ochars;
|
1191
|
+
size_t nchars, growth, seplen, tmplen, extratail;
|
1192
|
+
const jschar *sepstr;
|
1193
|
+
JSString *str;
|
1194
|
+
JSHashEntry *he;
|
1195
|
+
JSAtom *atom;
|
1196
|
+
|
1197
|
+
JS_CHECK_RECURSION(cx, return JS_FALSE);
|
1198
|
+
|
1199
|
+
ok = js_GetLengthProperty(cx, obj, &length);
|
1200
|
+
if (!ok)
|
1201
|
+
return JS_FALSE;
|
1202
|
+
|
1203
|
+
he = js_EnterSharpObject(cx, obj, NULL, &chars);
|
1204
|
+
if (!he)
|
1205
|
+
return JS_FALSE;
|
1206
|
+
#ifdef DEBUG
|
1207
|
+
growth = (size_t) -1;
|
1208
|
+
#endif
|
1209
|
+
|
1210
|
+
if (op == TO_SOURCE) {
|
1211
|
+
if (IS_SHARP(he)) {
|
1212
|
+
#if JS_HAS_SHARP_VARS
|
1213
|
+
nchars = js_strlen(chars);
|
1214
|
+
#else
|
1215
|
+
chars[0] = '[';
|
1216
|
+
chars[1] = ']';
|
1217
|
+
chars[2] = 0;
|
1218
|
+
nchars = 2;
|
1219
|
+
#endif
|
1220
|
+
goto make_string;
|
1221
|
+
}
|
1222
|
+
|
1223
|
+
/*
|
1224
|
+
* Always allocate 2 extra chars for closing ']' and terminating 0
|
1225
|
+
* and then preallocate 1 + extratail to include starting '['.
|
1226
|
+
*/
|
1227
|
+
extratail = 2;
|
1228
|
+
growth = (1 + extratail) * sizeof(jschar);
|
1229
|
+
if (!chars) {
|
1230
|
+
nchars = 0;
|
1231
|
+
chars = (jschar *) malloc(growth);
|
1232
|
+
if (!chars)
|
1233
|
+
goto done;
|
1234
|
+
} else {
|
1235
|
+
MAKE_SHARP(he);
|
1236
|
+
nchars = js_strlen(chars);
|
1237
|
+
growth += nchars * sizeof(jschar);
|
1238
|
+
chars = (jschar *)realloc((ochars = chars), growth);
|
1239
|
+
if (!chars) {
|
1240
|
+
free(ochars);
|
1241
|
+
goto done;
|
1242
|
+
}
|
1243
|
+
}
|
1244
|
+
chars[nchars++] = '[';
|
1245
|
+
JS_ASSERT(sep == NULL);
|
1246
|
+
sepstr = NULL; /* indicates to use ", " as separator */
|
1247
|
+
seplen = 2;
|
1248
|
+
} else {
|
1249
|
+
/*
|
1250
|
+
* Free any sharp variable definition in chars. Normally, we would
|
1251
|
+
* MAKE_SHARP(he) so that only the first sharp variable annotation is
|
1252
|
+
* a definition, and all the rest are references, but in the current
|
1253
|
+
* case of (op != TO_SOURCE), we don't need chars at all.
|
1254
|
+
*/
|
1255
|
+
if (chars)
|
1256
|
+
JS_free(cx, chars);
|
1257
|
+
chars = NULL;
|
1258
|
+
nchars = 0;
|
1259
|
+
extratail = 1; /* allocate extra char for terminating 0 */
|
1260
|
+
|
1261
|
+
/* Return the empty string on a cycle as well as on empty join. */
|
1262
|
+
if (IS_BUSY(he) || length == 0) {
|
1263
|
+
js_LeaveSharpObject(cx, NULL);
|
1264
|
+
*rval = JS_GetEmptyStringValue(cx);
|
1265
|
+
return ok;
|
1266
|
+
}
|
1267
|
+
|
1268
|
+
/* Flag he as BUSY so we can distinguish a cycle from a join-point. */
|
1269
|
+
MAKE_BUSY(he);
|
1270
|
+
|
1271
|
+
if (sep) {
|
1272
|
+
JSSTRING_CHARS_AND_LENGTH(sep, sepstr, seplen);
|
1273
|
+
} else {
|
1274
|
+
sepstr = NULL; /* indicates to use "," as separator */
|
1275
|
+
seplen = 1;
|
1276
|
+
}
|
1277
|
+
}
|
1278
|
+
|
1279
|
+
/* Use rval to locally root each element value as we loop and convert. */
|
1280
|
+
for (index = 0; index < length; index++) {
|
1281
|
+
ok = (JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
1282
|
+
GetArrayElement(cx, obj, index, &hole, rval));
|
1283
|
+
if (!ok)
|
1284
|
+
goto done;
|
1285
|
+
if (hole ||
|
1286
|
+
(op != TO_SOURCE &&
|
1287
|
+
(JSVAL_IS_VOID(*rval) || JSVAL_IS_NULL(*rval)))) {
|
1288
|
+
str = cx->runtime->emptyString;
|
1289
|
+
} else {
|
1290
|
+
if (op == TO_LOCALE_STRING) {
|
1291
|
+
JSObject *robj;
|
1292
|
+
|
1293
|
+
atom = cx->runtime->atomState.toLocaleStringAtom;
|
1294
|
+
ok = js_ValueToObject(cx, *rval, &robj);
|
1295
|
+
if (ok) {
|
1296
|
+
/* Re-use *rval to protect robj temporarily. */
|
1297
|
+
*rval = OBJECT_TO_JSVAL(robj);
|
1298
|
+
ok = js_TryMethod(cx, robj, atom, 0, NULL, rval);
|
1299
|
+
}
|
1300
|
+
if (!ok)
|
1301
|
+
goto done;
|
1302
|
+
str = js_ValueToString(cx, *rval);
|
1303
|
+
} else if (op == TO_STRING) {
|
1304
|
+
str = js_ValueToString(cx, *rval);
|
1305
|
+
} else {
|
1306
|
+
JS_ASSERT(op == TO_SOURCE);
|
1307
|
+
str = js_ValueToSource(cx, *rval);
|
1308
|
+
}
|
1309
|
+
if (!str) {
|
1310
|
+
ok = JS_FALSE;
|
1311
|
+
goto done;
|
1312
|
+
}
|
1313
|
+
}
|
1314
|
+
|
1315
|
+
/*
|
1316
|
+
* Do not append separator after the last element unless it is a hole
|
1317
|
+
* and we are in toSource. In that case we append single ",".
|
1318
|
+
*/
|
1319
|
+
if (index + 1 == length)
|
1320
|
+
seplen = (hole && op == TO_SOURCE) ? 1 : 0;
|
1321
|
+
|
1322
|
+
/* Allocate 1 at end for closing bracket and zero. */
|
1323
|
+
tmplen = JSSTRING_LENGTH(str);
|
1324
|
+
growth = nchars + tmplen + seplen + extratail;
|
1325
|
+
if (nchars > growth || tmplen > growth ||
|
1326
|
+
growth > (size_t)-1 / sizeof(jschar)) {
|
1327
|
+
if (chars) {
|
1328
|
+
free(chars);
|
1329
|
+
chars = NULL;
|
1330
|
+
}
|
1331
|
+
goto done;
|
1332
|
+
}
|
1333
|
+
growth *= sizeof(jschar);
|
1334
|
+
JS_COUNT_OPERATION(cx, JSOW_ALLOCATION);
|
1335
|
+
if (!chars) {
|
1336
|
+
chars = (jschar *) malloc(growth);
|
1337
|
+
if (!chars)
|
1338
|
+
goto done;
|
1339
|
+
} else {
|
1340
|
+
chars = (jschar *) realloc((ochars = chars), growth);
|
1341
|
+
if (!chars) {
|
1342
|
+
free(ochars);
|
1343
|
+
goto done;
|
1344
|
+
}
|
1345
|
+
}
|
1346
|
+
|
1347
|
+
js_strncpy(&chars[nchars], JSSTRING_CHARS(str), tmplen);
|
1348
|
+
nchars += tmplen;
|
1349
|
+
|
1350
|
+
if (seplen) {
|
1351
|
+
if (sepstr) {
|
1352
|
+
js_strncpy(&chars[nchars], sepstr, seplen);
|
1353
|
+
} else {
|
1354
|
+
JS_ASSERT(seplen == 1 || seplen == 2);
|
1355
|
+
chars[nchars] = ',';
|
1356
|
+
if (seplen == 2)
|
1357
|
+
chars[nchars + 1] = ' ';
|
1358
|
+
}
|
1359
|
+
nchars += seplen;
|
1360
|
+
}
|
1361
|
+
}
|
1362
|
+
|
1363
|
+
done:
|
1364
|
+
if (op == TO_SOURCE) {
|
1365
|
+
if (chars)
|
1366
|
+
chars[nchars++] = ']';
|
1367
|
+
} else {
|
1368
|
+
CLEAR_BUSY(he);
|
1369
|
+
}
|
1370
|
+
js_LeaveSharpObject(cx, NULL);
|
1371
|
+
if (!ok) {
|
1372
|
+
if (chars)
|
1373
|
+
free(chars);
|
1374
|
+
return ok;
|
1375
|
+
}
|
1376
|
+
|
1377
|
+
make_string:
|
1378
|
+
if (!chars) {
|
1379
|
+
JS_ReportOutOfMemory(cx);
|
1380
|
+
return JS_FALSE;
|
1381
|
+
}
|
1382
|
+
chars[nchars] = 0;
|
1383
|
+
JS_ASSERT(growth == (size_t)-1 || (nchars + 1) * sizeof(jschar) == growth);
|
1384
|
+
str = js_NewString(cx, chars, nchars);
|
1385
|
+
if (!str) {
|
1386
|
+
free(chars);
|
1387
|
+
return JS_FALSE;
|
1388
|
+
}
|
1389
|
+
*rval = STRING_TO_JSVAL(str);
|
1390
|
+
return JS_TRUE;
|
1391
|
+
}
|
1392
|
+
|
1393
|
+
#if JS_HAS_TOSOURCE
|
1394
|
+
static JSBool
|
1395
|
+
array_toSource(JSContext *cx, uintN argc, jsval *vp)
|
1396
|
+
{
|
1397
|
+
JSObject *obj;
|
1398
|
+
|
1399
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
1400
|
+
if (OBJ_GET_CLASS(cx, obj) != &js_SlowArrayClass &&
|
1401
|
+
!JS_InstanceOf(cx, obj, &js_ArrayClass, vp + 2)) {
|
1402
|
+
return JS_FALSE;
|
1403
|
+
}
|
1404
|
+
return array_join_sub(cx, obj, TO_SOURCE, NULL, vp);
|
1405
|
+
}
|
1406
|
+
#endif
|
1407
|
+
|
1408
|
+
static JSBool
|
1409
|
+
array_toString(JSContext *cx, uintN argc, jsval *vp)
|
1410
|
+
{
|
1411
|
+
JSObject *obj;
|
1412
|
+
|
1413
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
1414
|
+
if (OBJ_GET_CLASS(cx, obj) != &js_SlowArrayClass &&
|
1415
|
+
!JS_InstanceOf(cx, obj, &js_ArrayClass, vp + 2)) {
|
1416
|
+
return JS_FALSE;
|
1417
|
+
}
|
1418
|
+
return array_join_sub(cx, obj, TO_STRING, NULL, vp);
|
1419
|
+
}
|
1420
|
+
|
1421
|
+
static JSBool
|
1422
|
+
array_toLocaleString(JSContext *cx, uintN argc, jsval *vp)
|
1423
|
+
{
|
1424
|
+
JSObject *obj;
|
1425
|
+
|
1426
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
1427
|
+
if (OBJ_GET_CLASS(cx, obj) != &js_SlowArrayClass &&
|
1428
|
+
!JS_InstanceOf(cx, obj, &js_ArrayClass, vp + 2)) {
|
1429
|
+
return JS_FALSE;
|
1430
|
+
}
|
1431
|
+
|
1432
|
+
/*
|
1433
|
+
* Passing comma here as the separator. Need a way to get a
|
1434
|
+
* locale-specific version.
|
1435
|
+
*/
|
1436
|
+
return array_join_sub(cx, obj, TO_LOCALE_STRING, NULL, vp);
|
1437
|
+
}
|
1438
|
+
|
1439
|
+
static JSBool
|
1440
|
+
InitArrayElements(JSContext *cx, JSObject *obj, jsuint start, jsuint end,
|
1441
|
+
jsval *vector)
|
1442
|
+
{
|
1443
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj)) {
|
1444
|
+
if (!EnsureLength(cx, obj, end))
|
1445
|
+
return JS_FALSE;
|
1446
|
+
|
1447
|
+
if (end > (uint32)obj->fslots[JSSLOT_ARRAY_LENGTH])
|
1448
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = end;
|
1449
|
+
|
1450
|
+
memcpy(obj->dslots + start, vector, sizeof(jsval) * (end - start));
|
1451
|
+
return JS_TRUE;
|
1452
|
+
}
|
1453
|
+
|
1454
|
+
while (start != end) {
|
1455
|
+
if (!JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) ||
|
1456
|
+
!SetArrayElement(cx, obj, start++, *vector++)) {
|
1457
|
+
return JS_FALSE;
|
1458
|
+
}
|
1459
|
+
}
|
1460
|
+
return JS_TRUE;
|
1461
|
+
}
|
1462
|
+
|
1463
|
+
static JSBool
|
1464
|
+
InitArrayObject(JSContext *cx, JSObject *obj, jsuint length, jsval *vector)
|
1465
|
+
{
|
1466
|
+
JS_ASSERT(OBJ_IS_ARRAY(cx, obj));
|
1467
|
+
|
1468
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = length;
|
1469
|
+
|
1470
|
+
if (vector) {
|
1471
|
+
if (!EnsureLength(cx, obj, length))
|
1472
|
+
return JS_FALSE;
|
1473
|
+
memcpy(obj->dslots, vector, length * sizeof (jsval));
|
1474
|
+
obj->fslots[JSSLOT_ARRAY_COUNT] = length;
|
1475
|
+
} else {
|
1476
|
+
obj->fslots[JSSLOT_ARRAY_COUNT] = 0;
|
1477
|
+
}
|
1478
|
+
return JS_TRUE;
|
1479
|
+
}
|
1480
|
+
|
1481
|
+
/*
|
1482
|
+
* Perl-inspired join, reverse, and sort.
|
1483
|
+
*/
|
1484
|
+
static JSBool
|
1485
|
+
array_join(JSContext *cx, uintN argc, jsval *vp)
|
1486
|
+
{
|
1487
|
+
JSString *str;
|
1488
|
+
JSObject *obj;
|
1489
|
+
|
1490
|
+
if (JSVAL_IS_VOID(vp[2])) {
|
1491
|
+
str = NULL;
|
1492
|
+
} else {
|
1493
|
+
str = js_ValueToString(cx, vp[2]);
|
1494
|
+
if (!str)
|
1495
|
+
return JS_FALSE;
|
1496
|
+
vp[2] = STRING_TO_JSVAL(str);
|
1497
|
+
}
|
1498
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
1499
|
+
return obj && array_join_sub(cx, obj, TO_STRING, str, vp);
|
1500
|
+
}
|
1501
|
+
|
1502
|
+
static JSBool
|
1503
|
+
array_reverse(JSContext *cx, uintN argc, jsval *vp)
|
1504
|
+
{
|
1505
|
+
JSObject *obj;
|
1506
|
+
JSTempValueRooter tvr;
|
1507
|
+
jsuint len, half, i;
|
1508
|
+
JSBool ok, hole, hole2;
|
1509
|
+
|
1510
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
1511
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &len))
|
1512
|
+
return JS_FALSE;
|
1513
|
+
|
1514
|
+
ok = JS_TRUE;
|
1515
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
1516
|
+
half = len / 2;
|
1517
|
+
for (i = 0; i < half; i++) {
|
1518
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
1519
|
+
GetArrayElement(cx, obj, i, &hole, &tvr.u.value) &&
|
1520
|
+
GetArrayElement(cx, obj, len - i - 1, &hole2, vp) &&
|
1521
|
+
SetOrDeleteArrayElement(cx, obj, len - i - 1, hole, tvr.u.value) &&
|
1522
|
+
SetOrDeleteArrayElement(cx, obj, i, hole2, *vp);
|
1523
|
+
if (!ok)
|
1524
|
+
break;
|
1525
|
+
}
|
1526
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
1527
|
+
|
1528
|
+
*vp = OBJECT_TO_JSVAL(obj);
|
1529
|
+
return ok;
|
1530
|
+
}
|
1531
|
+
|
1532
|
+
typedef struct MSortArgs {
|
1533
|
+
size_t elsize;
|
1534
|
+
JSComparator cmp;
|
1535
|
+
void *arg;
|
1536
|
+
JSBool fastcopy;
|
1537
|
+
} MSortArgs;
|
1538
|
+
|
1539
|
+
/* Helper function for js_MergeSort. */
|
1540
|
+
static JSBool
|
1541
|
+
MergeArrays(MSortArgs *msa, void *src, void *dest, size_t run1, size_t run2)
|
1542
|
+
{
|
1543
|
+
void *arg, *a, *b, *c;
|
1544
|
+
size_t elsize, runtotal;
|
1545
|
+
int cmp_result;
|
1546
|
+
JSComparator cmp;
|
1547
|
+
JSBool fastcopy;
|
1548
|
+
|
1549
|
+
runtotal = run1 + run2;
|
1550
|
+
|
1551
|
+
elsize = msa->elsize;
|
1552
|
+
cmp = msa->cmp;
|
1553
|
+
arg = msa->arg;
|
1554
|
+
fastcopy = msa->fastcopy;
|
1555
|
+
|
1556
|
+
#define CALL_CMP(a, b) \
|
1557
|
+
if (!cmp(arg, (a), (b), &cmp_result)) return JS_FALSE;
|
1558
|
+
|
1559
|
+
/* Copy runs already in sorted order. */
|
1560
|
+
b = (char *)src + run1 * elsize;
|
1561
|
+
a = (char *)b - elsize;
|
1562
|
+
CALL_CMP(a, b);
|
1563
|
+
if (cmp_result <= 0) {
|
1564
|
+
memcpy(dest, src, runtotal * elsize);
|
1565
|
+
return JS_TRUE;
|
1566
|
+
}
|
1567
|
+
|
1568
|
+
#define COPY_ONE(p,q,n) \
|
1569
|
+
(fastcopy ? (void)(*(jsval*)(p) = *(jsval*)(q)) : (void)memcpy(p, q, n))
|
1570
|
+
|
1571
|
+
a = src;
|
1572
|
+
c = dest;
|
1573
|
+
for (; runtotal != 0; runtotal--) {
|
1574
|
+
JSBool from_a = run2 == 0;
|
1575
|
+
if (!from_a && run1 != 0) {
|
1576
|
+
CALL_CMP(a,b);
|
1577
|
+
from_a = cmp_result <= 0;
|
1578
|
+
}
|
1579
|
+
|
1580
|
+
if (from_a) {
|
1581
|
+
COPY_ONE(c, a, elsize);
|
1582
|
+
run1--;
|
1583
|
+
a = (char *)a + elsize;
|
1584
|
+
} else {
|
1585
|
+
COPY_ONE(c, b, elsize);
|
1586
|
+
run2--;
|
1587
|
+
b = (char *)b + elsize;
|
1588
|
+
}
|
1589
|
+
c = (char *)c + elsize;
|
1590
|
+
}
|
1591
|
+
#undef COPY_ONE
|
1592
|
+
#undef CALL_CMP
|
1593
|
+
|
1594
|
+
return JS_TRUE;
|
1595
|
+
}
|
1596
|
+
|
1597
|
+
/*
|
1598
|
+
* This sort is stable, i.e. sequence of equal elements is preserved.
|
1599
|
+
* See also bug #224128.
|
1600
|
+
*/
|
1601
|
+
JSBool
|
1602
|
+
js_MergeSort(void *src, size_t nel, size_t elsize,
|
1603
|
+
JSComparator cmp, void *arg, void *tmp)
|
1604
|
+
{
|
1605
|
+
void *swap, *vec1, *vec2;
|
1606
|
+
MSortArgs msa;
|
1607
|
+
size_t i, j, lo, hi, run;
|
1608
|
+
JSBool fastcopy;
|
1609
|
+
int cmp_result;
|
1610
|
+
|
1611
|
+
/* Avoid memcpy overhead for word-sized and word-aligned elements. */
|
1612
|
+
fastcopy = (elsize == sizeof(jsval) &&
|
1613
|
+
(((jsuword) src | (jsuword) tmp) & JSVAL_ALIGN) == 0);
|
1614
|
+
#define COPY_ONE(p,q,n) \
|
1615
|
+
(fastcopy ? (void)(*(jsval*)(p) = *(jsval*)(q)) : (void)memcpy(p, q, n))
|
1616
|
+
#define CALL_CMP(a, b) \
|
1617
|
+
if (!cmp(arg, (a), (b), &cmp_result)) return JS_FALSE;
|
1618
|
+
#define INS_SORT_INT 4
|
1619
|
+
|
1620
|
+
/*
|
1621
|
+
* Apply insertion sort to small chunks to reduce the number of merge
|
1622
|
+
* passes needed.
|
1623
|
+
*/
|
1624
|
+
for (lo = 0; lo < nel; lo += INS_SORT_INT) {
|
1625
|
+
hi = lo + INS_SORT_INT;
|
1626
|
+
if (hi >= nel)
|
1627
|
+
hi = nel;
|
1628
|
+
for (i = lo + 1; i < hi; i++) {
|
1629
|
+
vec1 = (char *)src + i * elsize;
|
1630
|
+
vec2 = (char *)vec1 - elsize;
|
1631
|
+
for (j = i; j > lo; j--) {
|
1632
|
+
CALL_CMP(vec2, vec1);
|
1633
|
+
/* "<=" instead of "<" insures the sort is stable */
|
1634
|
+
if (cmp_result <= 0) {
|
1635
|
+
break;
|
1636
|
+
}
|
1637
|
+
|
1638
|
+
/* Swap elements, using "tmp" as tmp storage */
|
1639
|
+
COPY_ONE(tmp, vec2, elsize);
|
1640
|
+
COPY_ONE(vec2, vec1, elsize);
|
1641
|
+
COPY_ONE(vec1, tmp, elsize);
|
1642
|
+
vec1 = vec2;
|
1643
|
+
vec2 = (char *)vec1 - elsize;
|
1644
|
+
}
|
1645
|
+
}
|
1646
|
+
}
|
1647
|
+
#undef CALL_CMP
|
1648
|
+
#undef COPY_ONE
|
1649
|
+
|
1650
|
+
msa.elsize = elsize;
|
1651
|
+
msa.cmp = cmp;
|
1652
|
+
msa.arg = arg;
|
1653
|
+
msa.fastcopy = fastcopy;
|
1654
|
+
|
1655
|
+
vec1 = src;
|
1656
|
+
vec2 = tmp;
|
1657
|
+
for (run = INS_SORT_INT; run < nel; run *= 2) {
|
1658
|
+
for (lo = 0; lo < nel; lo += 2 * run) {
|
1659
|
+
hi = lo + run;
|
1660
|
+
if (hi >= nel) {
|
1661
|
+
memcpy((char *)vec2 + lo * elsize, (char *)vec1 + lo * elsize,
|
1662
|
+
(nel - lo) * elsize);
|
1663
|
+
break;
|
1664
|
+
}
|
1665
|
+
if (!MergeArrays(&msa, (char *)vec1 + lo * elsize,
|
1666
|
+
(char *)vec2 + lo * elsize, run,
|
1667
|
+
hi + run > nel ? nel - hi : run)) {
|
1668
|
+
return JS_FALSE;
|
1669
|
+
}
|
1670
|
+
}
|
1671
|
+
swap = vec1;
|
1672
|
+
vec1 = vec2;
|
1673
|
+
vec2 = swap;
|
1674
|
+
}
|
1675
|
+
if (src != vec1)
|
1676
|
+
memcpy(src, tmp, nel * elsize);
|
1677
|
+
|
1678
|
+
return JS_TRUE;
|
1679
|
+
}
|
1680
|
+
|
1681
|
+
typedef struct CompareArgs {
|
1682
|
+
JSContext *context;
|
1683
|
+
jsval fval;
|
1684
|
+
jsval *elemroot; /* stack needed for js_Invoke */
|
1685
|
+
} CompareArgs;
|
1686
|
+
|
1687
|
+
static JSBool
|
1688
|
+
sort_compare(void *arg, const void *a, const void *b, int *result)
|
1689
|
+
{
|
1690
|
+
jsval av = *(const jsval *)a, bv = *(const jsval *)b;
|
1691
|
+
CompareArgs *ca = (CompareArgs *) arg;
|
1692
|
+
JSContext *cx = ca->context;
|
1693
|
+
jsval *invokevp, *sp;
|
1694
|
+
jsdouble cmp;
|
1695
|
+
|
1696
|
+
/**
|
1697
|
+
* array_sort deals with holes and undefs on its own and they should not
|
1698
|
+
* come here.
|
1699
|
+
*/
|
1700
|
+
JS_ASSERT(av != JSVAL_VOID);
|
1701
|
+
JS_ASSERT(bv != JSVAL_VOID);
|
1702
|
+
|
1703
|
+
if (!JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP))
|
1704
|
+
return JS_FALSE;
|
1705
|
+
|
1706
|
+
invokevp = ca->elemroot;
|
1707
|
+
sp = invokevp;
|
1708
|
+
*sp++ = ca->fval;
|
1709
|
+
*sp++ = JSVAL_NULL;
|
1710
|
+
*sp++ = av;
|
1711
|
+
*sp++ = bv;
|
1712
|
+
|
1713
|
+
if (!js_Invoke(cx, 2, invokevp, 0))
|
1714
|
+
return JS_FALSE;
|
1715
|
+
|
1716
|
+
cmp = js_ValueToNumber(cx, invokevp);
|
1717
|
+
if (JSVAL_IS_NULL(*invokevp))
|
1718
|
+
return JS_FALSE;
|
1719
|
+
|
1720
|
+
/* Clamp cmp to -1, 0, 1. */
|
1721
|
+
*result = 0;
|
1722
|
+
if (!JSDOUBLE_IS_NaN(cmp) && cmp != 0)
|
1723
|
+
*result = cmp > 0 ? 1 : -1;
|
1724
|
+
|
1725
|
+
/*
|
1726
|
+
* XXX else report some kind of error here? ECMA talks about 'consistent
|
1727
|
+
* compare functions' that don't return NaN, but is silent about what the
|
1728
|
+
* result should be. So we currently ignore it.
|
1729
|
+
*/
|
1730
|
+
|
1731
|
+
return JS_TRUE;
|
1732
|
+
}
|
1733
|
+
|
1734
|
+
static int
|
1735
|
+
sort_compare_strings(void *arg, const void *a, const void *b, int *result)
|
1736
|
+
{
|
1737
|
+
jsval av = *(const jsval *)a, bv = *(const jsval *)b;
|
1738
|
+
|
1739
|
+
JS_ASSERT(JSVAL_IS_STRING(av));
|
1740
|
+
JS_ASSERT(JSVAL_IS_STRING(bv));
|
1741
|
+
if (!JS_CHECK_OPERATION_LIMIT((JSContext *)arg, JSOW_JUMP))
|
1742
|
+
return JS_FALSE;
|
1743
|
+
|
1744
|
+
*result = (int) js_CompareStrings(JSVAL_TO_STRING(av), JSVAL_TO_STRING(bv));
|
1745
|
+
return JS_TRUE;
|
1746
|
+
}
|
1747
|
+
|
1748
|
+
/*
|
1749
|
+
* The array_sort function below assumes JSVAL_NULL is zero in order to
|
1750
|
+
* perform initialization using memset. Other parts of SpiderMonkey likewise
|
1751
|
+
* "know" that JSVAL_NULL is zero; this static assertion covers all cases.
|
1752
|
+
*/
|
1753
|
+
JS_STATIC_ASSERT(JSVAL_NULL == 0);
|
1754
|
+
|
1755
|
+
static JSBool
|
1756
|
+
array_sort(JSContext *cx, uintN argc, jsval *vp)
|
1757
|
+
{
|
1758
|
+
jsval *argv, fval, *vec, *mergesort_tmp, v;
|
1759
|
+
JSObject *obj;
|
1760
|
+
CompareArgs ca;
|
1761
|
+
jsuint len, newlen, i, undefs;
|
1762
|
+
JSTempValueRooter tvr;
|
1763
|
+
JSBool hole, ok;
|
1764
|
+
size_t elemsize;
|
1765
|
+
JSString *str;
|
1766
|
+
|
1767
|
+
/*
|
1768
|
+
* Optimize the default compare function case if all of obj's elements
|
1769
|
+
* have values of type string.
|
1770
|
+
*/
|
1771
|
+
JSBool all_strings;
|
1772
|
+
|
1773
|
+
argv = JS_ARGV(cx, vp);
|
1774
|
+
if (argc > 0) {
|
1775
|
+
if (JSVAL_IS_PRIMITIVE(argv[0])) {
|
1776
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
1777
|
+
JSMSG_BAD_SORT_ARG);
|
1778
|
+
return JS_FALSE;
|
1779
|
+
}
|
1780
|
+
fval = argv[0]; /* non-default compare function */
|
1781
|
+
} else {
|
1782
|
+
fval = JSVAL_NULL;
|
1783
|
+
}
|
1784
|
+
|
1785
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
1786
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &len))
|
1787
|
+
return JS_FALSE;
|
1788
|
+
if (len == 0) {
|
1789
|
+
*vp = OBJECT_TO_JSVAL(obj);
|
1790
|
+
return JS_TRUE;
|
1791
|
+
}
|
1792
|
+
|
1793
|
+
/*
|
1794
|
+
* We need a temporary array of 2 * len jsvals to hold the array elements
|
1795
|
+
* and the scratch space for merge sort. Check that its size does not
|
1796
|
+
* overflow size_t, which would allow for indexing beyond the end of the
|
1797
|
+
* malloc'd vector.
|
1798
|
+
*/
|
1799
|
+
#if JS_BITS_PER_WORD == 32
|
1800
|
+
if ((size_t)len > ~(size_t)0 / (2 * sizeof(jsval))) {
|
1801
|
+
js_ReportAllocationOverflow(cx);
|
1802
|
+
return JS_FALSE;
|
1803
|
+
}
|
1804
|
+
#endif
|
1805
|
+
vec = (jsval *) JS_malloc(cx, 2 * (size_t) len * sizeof(jsval));
|
1806
|
+
if (!vec)
|
1807
|
+
return JS_FALSE;
|
1808
|
+
|
1809
|
+
/*
|
1810
|
+
* Initialize vec as a root. We will clear elements of vec one by
|
1811
|
+
* one while increasing tvr.count when we know that the property at
|
1812
|
+
* the corresponding index exists and its value must be rooted.
|
1813
|
+
*
|
1814
|
+
* In this way when sorting a huge mostly sparse array we will not
|
1815
|
+
* access the tail of vec corresponding to properties that do not
|
1816
|
+
* exist, allowing OS to avoiding committing RAM. See bug 330812.
|
1817
|
+
*
|
1818
|
+
* After this point control must flow through label out: to exit.
|
1819
|
+
*/
|
1820
|
+
JS_PUSH_TEMP_ROOT(cx, 0, vec, &tvr);
|
1821
|
+
|
1822
|
+
/*
|
1823
|
+
* By ECMA 262, 15.4.4.11, a property that does not exist (which we
|
1824
|
+
* call a "hole") is always greater than an existing property with
|
1825
|
+
* value undefined and that is always greater than any other property.
|
1826
|
+
* Thus to sort holes and undefs we simply count them, sort the rest
|
1827
|
+
* of elements, append undefs after them and then make holes after
|
1828
|
+
* undefs.
|
1829
|
+
*/
|
1830
|
+
undefs = 0;
|
1831
|
+
newlen = 0;
|
1832
|
+
all_strings = JS_TRUE;
|
1833
|
+
for (i = 0; i < len; i++) {
|
1834
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP);
|
1835
|
+
if (!ok)
|
1836
|
+
goto out;
|
1837
|
+
|
1838
|
+
/* Clear vec[newlen] before including it in the rooted set. */
|
1839
|
+
vec[newlen] = JSVAL_NULL;
|
1840
|
+
tvr.count = newlen + 1;
|
1841
|
+
ok = GetArrayElement(cx, obj, i, &hole, &vec[newlen]);
|
1842
|
+
if (!ok)
|
1843
|
+
goto out;
|
1844
|
+
|
1845
|
+
if (hole)
|
1846
|
+
continue;
|
1847
|
+
|
1848
|
+
if (vec[newlen] == JSVAL_VOID) {
|
1849
|
+
++undefs;
|
1850
|
+
continue;
|
1851
|
+
}
|
1852
|
+
|
1853
|
+
/* We know JSVAL_IS_STRING yields 0 or 1, so avoid a branch via &=. */
|
1854
|
+
all_strings &= JSVAL_IS_STRING(vec[newlen]);
|
1855
|
+
|
1856
|
+
++newlen;
|
1857
|
+
}
|
1858
|
+
|
1859
|
+
if (newlen == 0) {
|
1860
|
+
/* The array has only holes and undefs. */
|
1861
|
+
ok = JS_TRUE;
|
1862
|
+
goto out;
|
1863
|
+
}
|
1864
|
+
|
1865
|
+
/*
|
1866
|
+
* The first newlen elements of vec are copied from the array object
|
1867
|
+
* (above). The remaining newlen positions are used as GC-rooted scratch
|
1868
|
+
* space for mergesort. We must clear the space before including it to
|
1869
|
+
* the root set covered by tvr.count. We assume JSVAL_NULL==0 to optimize
|
1870
|
+
* initialization using memset.
|
1871
|
+
*/
|
1872
|
+
mergesort_tmp = vec + newlen;
|
1873
|
+
memset(mergesort_tmp, 0, newlen * sizeof(jsval));
|
1874
|
+
tvr.count = newlen * 2;
|
1875
|
+
|
1876
|
+
/* Here len == 2 * (newlen + undefs + number_of_holes). */
|
1877
|
+
if (fval == JSVAL_NULL) {
|
1878
|
+
/*
|
1879
|
+
* Sort using the default comparator converting all elements to
|
1880
|
+
* strings.
|
1881
|
+
*/
|
1882
|
+
if (all_strings) {
|
1883
|
+
elemsize = sizeof(jsval);
|
1884
|
+
} else {
|
1885
|
+
/*
|
1886
|
+
* To avoid string conversion on each compare we do it only once
|
1887
|
+
* prior to sorting. But we also need the space for the original
|
1888
|
+
* values to recover the sorting result. To reuse
|
1889
|
+
* sort_compare_strings we move the original values to the odd
|
1890
|
+
* indexes in vec, put the string conversion results in the even
|
1891
|
+
* indexes and pass 2 * sizeof(jsval) as an element size to the
|
1892
|
+
* sorting function. In this way sort_compare_strings will only
|
1893
|
+
* see the string values when it casts the compare arguments as
|
1894
|
+
* pointers to jsval.
|
1895
|
+
*
|
1896
|
+
* This requires doubling the temporary storage including the
|
1897
|
+
* scratch space for the merge sort. Since vec already contains
|
1898
|
+
* the rooted scratch space for newlen elements at the tail, we
|
1899
|
+
* can use it to rearrange and convert to strings first and try
|
1900
|
+
* realloc only when we know that we successfully converted all
|
1901
|
+
* the elements.
|
1902
|
+
*/
|
1903
|
+
#if JS_BITS_PER_WORD == 32
|
1904
|
+
if ((size_t)newlen > ~(size_t)0 / (4 * sizeof(jsval))) {
|
1905
|
+
js_ReportAllocationOverflow(cx);
|
1906
|
+
ok = JS_FALSE;
|
1907
|
+
goto out;
|
1908
|
+
}
|
1909
|
+
#endif
|
1910
|
+
|
1911
|
+
/*
|
1912
|
+
* Rearrange and string-convert the elements of the vector from
|
1913
|
+
* the tail here and, after sorting, move the results back
|
1914
|
+
* starting from the start to prevent overwrite the existing
|
1915
|
+
* elements.
|
1916
|
+
*/
|
1917
|
+
i = newlen;
|
1918
|
+
do {
|
1919
|
+
--i;
|
1920
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP);
|
1921
|
+
if (!ok)
|
1922
|
+
goto out;
|
1923
|
+
v = vec[i];
|
1924
|
+
str = js_ValueToString(cx, v);
|
1925
|
+
if (!str) {
|
1926
|
+
ok = JS_FALSE;
|
1927
|
+
goto out;
|
1928
|
+
}
|
1929
|
+
vec[2 * i] = STRING_TO_JSVAL(str);
|
1930
|
+
vec[2 * i + 1] = v;
|
1931
|
+
} while (i != 0);
|
1932
|
+
|
1933
|
+
JS_ASSERT(tvr.u.array == vec);
|
1934
|
+
vec = (jsval *) JS_realloc(cx, vec,
|
1935
|
+
4 * (size_t) newlen * sizeof(jsval));
|
1936
|
+
if (!vec) {
|
1937
|
+
vec = tvr.u.array;
|
1938
|
+
ok = JS_FALSE;
|
1939
|
+
goto out;
|
1940
|
+
}
|
1941
|
+
tvr.u.array = vec;
|
1942
|
+
mergesort_tmp = vec + 2 * newlen;
|
1943
|
+
memset(mergesort_tmp, 0, newlen * 2 * sizeof(jsval));
|
1944
|
+
tvr.count = newlen * 4;
|
1945
|
+
elemsize = 2 * sizeof(jsval);
|
1946
|
+
}
|
1947
|
+
ok = js_MergeSort(vec, (size_t) newlen, elemsize,
|
1948
|
+
sort_compare_strings, cx, mergesort_tmp);
|
1949
|
+
if (!ok)
|
1950
|
+
goto out;
|
1951
|
+
if (!all_strings) {
|
1952
|
+
/*
|
1953
|
+
* We want to make the following loop fast and to unroot the
|
1954
|
+
* cached results of toString invocations before the operation
|
1955
|
+
* callback has a chance to run the GC. For this reason we do
|
1956
|
+
* not call JS_CHECK_OPERATION_LIMIT in the loop.
|
1957
|
+
*/
|
1958
|
+
i = 0;
|
1959
|
+
do {
|
1960
|
+
vec[i] = vec[2 * i + 1];
|
1961
|
+
} while (++i != newlen);
|
1962
|
+
}
|
1963
|
+
} else {
|
1964
|
+
void *mark;
|
1965
|
+
|
1966
|
+
ca.context = cx;
|
1967
|
+
ca.fval = fval;
|
1968
|
+
ca.elemroot = js_AllocStack(cx, 2 + 2, &mark);
|
1969
|
+
if (!ca.elemroot) {
|
1970
|
+
ok = JS_FALSE;
|
1971
|
+
goto out;
|
1972
|
+
}
|
1973
|
+
ok = js_MergeSort(vec, (size_t) newlen, sizeof(jsval),
|
1974
|
+
sort_compare, &ca, mergesort_tmp);
|
1975
|
+
js_FreeStack(cx, mark);
|
1976
|
+
if (!ok)
|
1977
|
+
goto out;
|
1978
|
+
}
|
1979
|
+
|
1980
|
+
/*
|
1981
|
+
* We no longer need to root the scratch space for the merge sort, so
|
1982
|
+
* unroot it now to make the job of a potential GC under InitArrayElements
|
1983
|
+
* easier.
|
1984
|
+
*/
|
1985
|
+
tvr.count = newlen;
|
1986
|
+
ok = InitArrayElements(cx, obj, 0, newlen, vec);
|
1987
|
+
if (!ok)
|
1988
|
+
goto out;
|
1989
|
+
|
1990
|
+
out:
|
1991
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
1992
|
+
JS_free(cx, vec);
|
1993
|
+
if (!ok)
|
1994
|
+
return JS_FALSE;
|
1995
|
+
|
1996
|
+
/* Set undefs that sorted after the rest of elements. */
|
1997
|
+
while (undefs != 0) {
|
1998
|
+
--undefs;
|
1999
|
+
if (!JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) ||
|
2000
|
+
!SetArrayElement(cx, obj, newlen++, JSVAL_VOID)) {
|
2001
|
+
return JS_FALSE;
|
2002
|
+
}
|
2003
|
+
}
|
2004
|
+
|
2005
|
+
/* Re-create any holes that sorted to the end of the array. */
|
2006
|
+
while (len > newlen) {
|
2007
|
+
if (!JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) ||
|
2008
|
+
!DeleteArrayElement(cx, obj, --len)) {
|
2009
|
+
return JS_FALSE;
|
2010
|
+
}
|
2011
|
+
}
|
2012
|
+
*vp = OBJECT_TO_JSVAL(obj);
|
2013
|
+
return JS_TRUE;
|
2014
|
+
}
|
2015
|
+
|
2016
|
+
/*
|
2017
|
+
* Perl-inspired push, pop, shift, unshift, and splice methods.
|
2018
|
+
*/
|
2019
|
+
static JSBool
|
2020
|
+
array_push_slowly(JSContext *cx, JSObject *obj, uintN argc, jsval *vp)
|
2021
|
+
{
|
2022
|
+
jsuint length, newlength;
|
2023
|
+
|
2024
|
+
if (!js_GetLengthProperty(cx, obj, &length))
|
2025
|
+
return JS_FALSE;
|
2026
|
+
newlength = length + argc;
|
2027
|
+
if (!InitArrayElements(cx, obj, length, newlength, vp + 2))
|
2028
|
+
return JS_FALSE;
|
2029
|
+
|
2030
|
+
/* Per ECMA-262, return the new array length. */
|
2031
|
+
if (!IndexToValue(cx, newlength, vp))
|
2032
|
+
return JS_FALSE;
|
2033
|
+
return js_SetLengthProperty(cx, obj, newlength);
|
2034
|
+
}
|
2035
|
+
|
2036
|
+
static JSBool
|
2037
|
+
array_push(JSContext *cx, uintN argc, jsval *vp)
|
2038
|
+
{
|
2039
|
+
JSObject *obj;
|
2040
|
+
uint32 length;
|
2041
|
+
|
2042
|
+
/* Insist on one argument and obj of the expected class. */
|
2043
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2044
|
+
if (!obj)
|
2045
|
+
return JS_FALSE;
|
2046
|
+
if (argc != 1 || !OBJ_IS_DENSE_ARRAY(cx, obj))
|
2047
|
+
return array_push_slowly(cx, obj, argc, vp);
|
2048
|
+
|
2049
|
+
length = obj->fslots[JSSLOT_ARRAY_LENGTH];
|
2050
|
+
if (INDEX_TOO_SPARSE(obj, length)) {
|
2051
|
+
if (!MakeArraySlow(cx, obj))
|
2052
|
+
return JS_FALSE;
|
2053
|
+
return array_push_slowly(cx, obj, argc, vp);
|
2054
|
+
}
|
2055
|
+
|
2056
|
+
if (!EnsureLength(cx, obj, length + 1))
|
2057
|
+
return JS_FALSE;
|
2058
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = length + 1;
|
2059
|
+
|
2060
|
+
JS_ASSERT(obj->dslots[length] == JSVAL_HOLE);
|
2061
|
+
obj->fslots[JSSLOT_ARRAY_COUNT]++;
|
2062
|
+
obj->dslots[length] = vp[2];
|
2063
|
+
return IndexToValue(cx, obj->fslots[JSSLOT_ARRAY_LENGTH], vp);
|
2064
|
+
}
|
2065
|
+
|
2066
|
+
JSBool
|
2067
|
+
array_pop(JSContext *cx, uintN argc, jsval *vp)
|
2068
|
+
{
|
2069
|
+
JSObject *obj;
|
2070
|
+
jsuint index;
|
2071
|
+
JSBool hole;
|
2072
|
+
|
2073
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2074
|
+
if (!obj)
|
2075
|
+
return JS_FALSE;
|
2076
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj)) {
|
2077
|
+
*vp = JSVAL_VOID;
|
2078
|
+
index = obj->fslots[JSSLOT_ARRAY_LENGTH];
|
2079
|
+
if (index == 0)
|
2080
|
+
return JS_TRUE;
|
2081
|
+
index--;
|
2082
|
+
if (index < ARRAY_DENSE_LENGTH(obj)) {
|
2083
|
+
*vp = obj->dslots[index];
|
2084
|
+
JS_ASSERT(*vp != JSVAL_HOLE);
|
2085
|
+
if (index == 0) {
|
2086
|
+
JS_free(cx, obj->dslots - 1);
|
2087
|
+
obj->dslots = NULL;
|
2088
|
+
} else {
|
2089
|
+
ARRAY_SET_DENSE_LENGTH(obj, index);
|
2090
|
+
}
|
2091
|
+
obj->fslots[JSSLOT_ARRAY_COUNT]--;
|
2092
|
+
}
|
2093
|
+
|
2094
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = index;
|
2095
|
+
return JS_TRUE;
|
2096
|
+
}
|
2097
|
+
|
2098
|
+
if (!js_GetLengthProperty(cx, obj, &index))
|
2099
|
+
return JS_FALSE;
|
2100
|
+
if (index == 0) {
|
2101
|
+
*vp = JSVAL_VOID;
|
2102
|
+
} else {
|
2103
|
+
index--;
|
2104
|
+
|
2105
|
+
/* Get the to-be-deleted property's value into vp. */
|
2106
|
+
if (!GetArrayElement(cx, obj, index, &hole, vp))
|
2107
|
+
return JS_FALSE;
|
2108
|
+
if (!hole && !DeleteArrayElement(cx, obj, index))
|
2109
|
+
return JS_FALSE;
|
2110
|
+
}
|
2111
|
+
return js_SetLengthProperty(cx, obj, index);
|
2112
|
+
}
|
2113
|
+
|
2114
|
+
static JSBool
|
2115
|
+
array_shift(JSContext *cx, uintN argc, jsval *vp)
|
2116
|
+
{
|
2117
|
+
JSObject *obj;
|
2118
|
+
jsuint length, i;
|
2119
|
+
JSBool hole, ok;
|
2120
|
+
JSTempValueRooter tvr;
|
2121
|
+
|
2122
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2123
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &length))
|
2124
|
+
return JS_FALSE;
|
2125
|
+
if (length == 0) {
|
2126
|
+
*vp = JSVAL_VOID;
|
2127
|
+
} else {
|
2128
|
+
length--;
|
2129
|
+
|
2130
|
+
/* Get the to-be-deleted property's value into vp ASAP. */
|
2131
|
+
if (!GetArrayElement(cx, obj, 0, &hole, vp))
|
2132
|
+
return JS_FALSE;
|
2133
|
+
|
2134
|
+
/* Slide down the array above the first element. */
|
2135
|
+
ok = JS_TRUE;
|
2136
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
2137
|
+
for (i = 0; i != length; i++) {
|
2138
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2139
|
+
GetArrayElement(cx, obj, i + 1, &hole, &tvr.u.value) &&
|
2140
|
+
SetOrDeleteArrayElement(cx, obj, i, hole, tvr.u.value);
|
2141
|
+
if (!ok)
|
2142
|
+
break;
|
2143
|
+
}
|
2144
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
2145
|
+
if (!ok)
|
2146
|
+
return JS_FALSE;
|
2147
|
+
|
2148
|
+
/* Delete the only or last element when it exist. */
|
2149
|
+
if (!hole && !DeleteArrayElement(cx, obj, length))
|
2150
|
+
return JS_FALSE;
|
2151
|
+
}
|
2152
|
+
return js_SetLengthProperty(cx, obj, length);
|
2153
|
+
}
|
2154
|
+
|
2155
|
+
static JSBool
|
2156
|
+
array_unshift(JSContext *cx, uintN argc, jsval *vp)
|
2157
|
+
{
|
2158
|
+
JSObject *obj;
|
2159
|
+
jsval *argv;
|
2160
|
+
jsuint length, last;
|
2161
|
+
JSBool hole, ok;
|
2162
|
+
JSTempValueRooter tvr;
|
2163
|
+
|
2164
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2165
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &length))
|
2166
|
+
return JS_FALSE;
|
2167
|
+
if (argc > 0) {
|
2168
|
+
/* Slide up the array to make room for argc at the bottom. */
|
2169
|
+
argv = JS_ARGV(cx, vp);
|
2170
|
+
if (length > 0) {
|
2171
|
+
last = length;
|
2172
|
+
ok = JS_TRUE;
|
2173
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
2174
|
+
do {
|
2175
|
+
--last;
|
2176
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2177
|
+
GetArrayElement(cx, obj, last, &hole, &tvr.u.value) &&
|
2178
|
+
SetOrDeleteArrayElement(cx, obj, last + argc, hole,
|
2179
|
+
tvr.u.value);
|
2180
|
+
if (!ok)
|
2181
|
+
break;
|
2182
|
+
} while (last != 0);
|
2183
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
2184
|
+
if (!ok)
|
2185
|
+
return JS_FALSE;
|
2186
|
+
}
|
2187
|
+
|
2188
|
+
/* Copy from argv to the bottom of the array. */
|
2189
|
+
if (!InitArrayElements(cx, obj, 0, argc, argv))
|
2190
|
+
return JS_FALSE;
|
2191
|
+
|
2192
|
+
length += argc;
|
2193
|
+
if (!js_SetLengthProperty(cx, obj, length))
|
2194
|
+
return JS_FALSE;
|
2195
|
+
}
|
2196
|
+
|
2197
|
+
/* Follow Perl by returning the new array length. */
|
2198
|
+
return IndexToValue(cx, length, vp);
|
2199
|
+
}
|
2200
|
+
|
2201
|
+
static JSBool
|
2202
|
+
array_splice(JSContext *cx, uintN argc, jsval *vp)
|
2203
|
+
{
|
2204
|
+
jsval *argv;
|
2205
|
+
JSObject *obj;
|
2206
|
+
jsuint length, begin, end, count, delta, last;
|
2207
|
+
jsdouble d;
|
2208
|
+
JSBool hole, ok;
|
2209
|
+
JSObject *obj2;
|
2210
|
+
JSTempValueRooter tvr;
|
2211
|
+
|
2212
|
+
/* Nothing to do if no args. Otherwise get length. */
|
2213
|
+
if (argc == 0)
|
2214
|
+
return JS_TRUE;
|
2215
|
+
argv = JS_ARGV(cx, vp);
|
2216
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2217
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &length))
|
2218
|
+
return JS_FALSE;
|
2219
|
+
|
2220
|
+
/* Convert the first argument into a starting index. */
|
2221
|
+
d = js_ValueToNumber(cx, argv);
|
2222
|
+
if (JSVAL_IS_NULL(*argv))
|
2223
|
+
return JS_FALSE;
|
2224
|
+
d = js_DoubleToInteger(d);
|
2225
|
+
if (d < 0) {
|
2226
|
+
d += length;
|
2227
|
+
if (d < 0)
|
2228
|
+
d = 0;
|
2229
|
+
} else if (d > length) {
|
2230
|
+
d = length;
|
2231
|
+
}
|
2232
|
+
begin = (jsuint)d; /* d has been clamped to uint32 */
|
2233
|
+
argc--;
|
2234
|
+
argv++;
|
2235
|
+
|
2236
|
+
/* Convert the second argument from a count into a fencepost index. */
|
2237
|
+
delta = length - begin;
|
2238
|
+
if (argc == 0) {
|
2239
|
+
count = delta;
|
2240
|
+
end = length;
|
2241
|
+
} else {
|
2242
|
+
d = js_ValueToNumber(cx, argv);
|
2243
|
+
if (JSVAL_IS_NULL(*argv))
|
2244
|
+
return JS_FALSE;
|
2245
|
+
d = js_DoubleToInteger(d);
|
2246
|
+
if (d < 0)
|
2247
|
+
d = 0;
|
2248
|
+
else if (d > delta)
|
2249
|
+
d = delta;
|
2250
|
+
count = (jsuint)d;
|
2251
|
+
end = begin + count;
|
2252
|
+
argc--;
|
2253
|
+
argv++;
|
2254
|
+
}
|
2255
|
+
|
2256
|
+
/*
|
2257
|
+
* Create a new array value to return. Our ECMA v2 proposal specs
|
2258
|
+
* that splice always returns an array value, even when given no
|
2259
|
+
* arguments. We think this is best because it eliminates the need
|
2260
|
+
* for callers to do an extra test to handle the empty splice case.
|
2261
|
+
*/
|
2262
|
+
obj2 = js_NewArrayObject(cx, 0, NULL);
|
2263
|
+
if (!obj2)
|
2264
|
+
return JS_FALSE;
|
2265
|
+
*vp = OBJECT_TO_JSVAL(obj2);
|
2266
|
+
|
2267
|
+
/* After this, control must flow through label out: to exit. */
|
2268
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
2269
|
+
|
2270
|
+
/* If there are elements to remove, put them into the return value. */
|
2271
|
+
if (count > 0) {
|
2272
|
+
for (last = begin; last < end; last++) {
|
2273
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2274
|
+
GetArrayElement(cx, obj, last, &hole, &tvr.u.value);
|
2275
|
+
if (!ok)
|
2276
|
+
goto out;
|
2277
|
+
|
2278
|
+
/* Copy tvr.u.value to new array unless it's a hole. */
|
2279
|
+
if (!hole) {
|
2280
|
+
ok = SetArrayElement(cx, obj2, last - begin, tvr.u.value);
|
2281
|
+
if (!ok)
|
2282
|
+
goto out;
|
2283
|
+
}
|
2284
|
+
}
|
2285
|
+
|
2286
|
+
ok = js_SetLengthProperty(cx, obj2, end - begin);
|
2287
|
+
if (!ok)
|
2288
|
+
goto out;
|
2289
|
+
}
|
2290
|
+
|
2291
|
+
/* Find the direction (up or down) to copy and make way for argv. */
|
2292
|
+
if (argc > count) {
|
2293
|
+
delta = (jsuint)argc - count;
|
2294
|
+
last = length;
|
2295
|
+
/* (uint) end could be 0, so can't use vanilla >= test */
|
2296
|
+
while (last-- > end) {
|
2297
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2298
|
+
GetArrayElement(cx, obj, last, &hole, &tvr.u.value) &&
|
2299
|
+
SetOrDeleteArrayElement(cx, obj, last + delta, hole,
|
2300
|
+
tvr.u.value);
|
2301
|
+
if (!ok)
|
2302
|
+
goto out;
|
2303
|
+
}
|
2304
|
+
length += delta;
|
2305
|
+
} else if (argc < count) {
|
2306
|
+
delta = count - (jsuint)argc;
|
2307
|
+
for (last = end; last < length; last++) {
|
2308
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2309
|
+
GetArrayElement(cx, obj, last, &hole, &tvr.u.value) &&
|
2310
|
+
SetOrDeleteArrayElement(cx, obj, last - delta, hole,
|
2311
|
+
tvr.u.value);
|
2312
|
+
if (!ok)
|
2313
|
+
goto out;
|
2314
|
+
}
|
2315
|
+
length -= delta;
|
2316
|
+
}
|
2317
|
+
|
2318
|
+
/* Copy from argv into the hole to complete the splice. */
|
2319
|
+
ok = InitArrayElements(cx, obj, begin, begin + argc, argv);
|
2320
|
+
if (!ok)
|
2321
|
+
goto out;
|
2322
|
+
|
2323
|
+
/* Update length in case we deleted elements from the end. */
|
2324
|
+
ok = js_SetLengthProperty(cx, obj, length);
|
2325
|
+
|
2326
|
+
out:
|
2327
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
2328
|
+
return ok;
|
2329
|
+
}
|
2330
|
+
|
2331
|
+
/*
|
2332
|
+
* Python-esque sequence operations.
|
2333
|
+
*/
|
2334
|
+
static JSBool
|
2335
|
+
array_concat(JSContext *cx, uintN argc, jsval *vp)
|
2336
|
+
{
|
2337
|
+
jsval *argv, v;
|
2338
|
+
JSObject *nobj, *aobj;
|
2339
|
+
jsuint length, alength, slot;
|
2340
|
+
uintN i;
|
2341
|
+
JSBool hole, ok;
|
2342
|
+
JSTempValueRooter tvr;
|
2343
|
+
|
2344
|
+
/* Treat our |this| object as the first argument; see ECMA 15.4.4.4. */
|
2345
|
+
argv = JS_ARGV(cx, vp) - 1;
|
2346
|
+
JS_ASSERT(JS_THIS_OBJECT(cx, vp) == JSVAL_TO_OBJECT(argv[0]));
|
2347
|
+
|
2348
|
+
/* Create a new Array object and root it using *vp. */
|
2349
|
+
aobj = JS_THIS_OBJECT(cx, vp);
|
2350
|
+
if (OBJ_IS_DENSE_ARRAY(cx, aobj)) {
|
2351
|
+
nobj = js_NewArrayObject(cx, ARRAY_DENSE_LENGTH(aobj), aobj->dslots);
|
2352
|
+
if (!nobj)
|
2353
|
+
return JS_FALSE;
|
2354
|
+
length = aobj->fslots[JSSLOT_ARRAY_LENGTH];
|
2355
|
+
nobj->fslots[JSSLOT_ARRAY_LENGTH] = length;
|
2356
|
+
*vp = OBJECT_TO_JSVAL(nobj);
|
2357
|
+
if (argc == 0)
|
2358
|
+
return JS_TRUE;
|
2359
|
+
argc--;
|
2360
|
+
argv++;
|
2361
|
+
} else {
|
2362
|
+
nobj = js_NewArrayObject(cx, 0, NULL);
|
2363
|
+
if (!nobj)
|
2364
|
+
return JS_FALSE;
|
2365
|
+
*vp = OBJECT_TO_JSVAL(nobj);
|
2366
|
+
length = 0;
|
2367
|
+
}
|
2368
|
+
|
2369
|
+
/* After this, control must flow through label out: to exit. */
|
2370
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
2371
|
+
|
2372
|
+
/* Loop over [0, argc] to concat args into nobj, expanding all Arrays. */
|
2373
|
+
for (i = 0; i <= argc; i++) {
|
2374
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP);
|
2375
|
+
if (!ok)
|
2376
|
+
goto out;
|
2377
|
+
v = argv[i];
|
2378
|
+
if (!JSVAL_IS_PRIMITIVE(v)) {
|
2379
|
+
JSObject *wobj;
|
2380
|
+
|
2381
|
+
aobj = JSVAL_TO_OBJECT(v);
|
2382
|
+
wobj = js_GetWrappedObject(cx, aobj);
|
2383
|
+
if (OBJ_IS_ARRAY(cx, wobj)) {
|
2384
|
+
ok = OBJ_GET_PROPERTY(cx, aobj,
|
2385
|
+
ATOM_TO_JSID(cx->runtime->atomState
|
2386
|
+
.lengthAtom),
|
2387
|
+
&tvr.u.value);
|
2388
|
+
if (!ok)
|
2389
|
+
goto out;
|
2390
|
+
alength = ValueIsLength(cx, &tvr.u.value);
|
2391
|
+
ok = !JSVAL_IS_NULL(tvr.u.value);
|
2392
|
+
if (!ok)
|
2393
|
+
goto out;
|
2394
|
+
for (slot = 0; slot < alength; slot++) {
|
2395
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2396
|
+
GetArrayElement(cx, aobj, slot, &hole,
|
2397
|
+
&tvr.u.value);
|
2398
|
+
if (!ok)
|
2399
|
+
goto out;
|
2400
|
+
|
2401
|
+
/*
|
2402
|
+
* Per ECMA 262, 15.4.4.4, step 9, ignore non-existent
|
2403
|
+
* properties.
|
2404
|
+
*/
|
2405
|
+
if (!hole) {
|
2406
|
+
ok = SetArrayElement(cx, nobj, length + slot,
|
2407
|
+
tvr.u.value);
|
2408
|
+
if (!ok)
|
2409
|
+
goto out;
|
2410
|
+
}
|
2411
|
+
}
|
2412
|
+
length += alength;
|
2413
|
+
continue;
|
2414
|
+
}
|
2415
|
+
}
|
2416
|
+
|
2417
|
+
ok = SetArrayElement(cx, nobj, length, v);
|
2418
|
+
if (!ok)
|
2419
|
+
goto out;
|
2420
|
+
length++;
|
2421
|
+
}
|
2422
|
+
|
2423
|
+
ok = js_SetLengthProperty(cx, nobj, length);
|
2424
|
+
|
2425
|
+
out:
|
2426
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
2427
|
+
return ok;
|
2428
|
+
}
|
2429
|
+
|
2430
|
+
static JSBool
|
2431
|
+
array_slice(JSContext *cx, uintN argc, jsval *vp)
|
2432
|
+
{
|
2433
|
+
jsval *argv;
|
2434
|
+
JSObject *nobj, *obj;
|
2435
|
+
jsuint length, begin, end, slot;
|
2436
|
+
jsdouble d;
|
2437
|
+
JSBool hole, ok;
|
2438
|
+
JSTempValueRooter tvr;
|
2439
|
+
|
2440
|
+
argv = JS_ARGV(cx, vp);
|
2441
|
+
|
2442
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2443
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &length))
|
2444
|
+
return JS_FALSE;
|
2445
|
+
begin = 0;
|
2446
|
+
end = length;
|
2447
|
+
|
2448
|
+
if (argc > 0) {
|
2449
|
+
d = js_ValueToNumber(cx, &argv[0]);
|
2450
|
+
if (JSVAL_IS_NULL(argv[0]))
|
2451
|
+
return JS_FALSE;
|
2452
|
+
d = js_DoubleToInteger(d);
|
2453
|
+
if (d < 0) {
|
2454
|
+
d += length;
|
2455
|
+
if (d < 0)
|
2456
|
+
d = 0;
|
2457
|
+
} else if (d > length) {
|
2458
|
+
d = length;
|
2459
|
+
}
|
2460
|
+
begin = (jsuint)d;
|
2461
|
+
|
2462
|
+
if (argc > 1) {
|
2463
|
+
d = js_ValueToNumber(cx, &argv[1]);
|
2464
|
+
if (JSVAL_IS_NULL(argv[1]))
|
2465
|
+
return JS_FALSE;
|
2466
|
+
d = js_DoubleToInteger(d);
|
2467
|
+
if (d < 0) {
|
2468
|
+
d += length;
|
2469
|
+
if (d < 0)
|
2470
|
+
d = 0;
|
2471
|
+
} else if (d > length) {
|
2472
|
+
d = length;
|
2473
|
+
}
|
2474
|
+
end = (jsuint)d;
|
2475
|
+
}
|
2476
|
+
}
|
2477
|
+
|
2478
|
+
if (begin > end)
|
2479
|
+
begin = end;
|
2480
|
+
|
2481
|
+
if (OBJ_IS_DENSE_ARRAY(cx, obj) && end <= ARRAY_DENSE_LENGTH(obj)) {
|
2482
|
+
nobj = js_NewArrayObject(cx, end - begin, obj->dslots + begin);
|
2483
|
+
if (!nobj)
|
2484
|
+
return JS_FALSE;
|
2485
|
+
*vp = OBJECT_TO_JSVAL(nobj);
|
2486
|
+
return JS_TRUE;
|
2487
|
+
}
|
2488
|
+
|
2489
|
+
/* Create a new Array object and root it using *vp. */
|
2490
|
+
nobj = js_NewArrayObject(cx, 0, NULL);
|
2491
|
+
if (!nobj)
|
2492
|
+
return JS_FALSE;
|
2493
|
+
*vp = OBJECT_TO_JSVAL(nobj);
|
2494
|
+
|
2495
|
+
/* After this, control must flow through label out: to exit. */
|
2496
|
+
JS_PUSH_SINGLE_TEMP_ROOT(cx, JSVAL_NULL, &tvr);
|
2497
|
+
|
2498
|
+
for (slot = begin; slot < end; slot++) {
|
2499
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2500
|
+
GetArrayElement(cx, obj, slot, &hole, &tvr.u.value);
|
2501
|
+
if (!ok)
|
2502
|
+
goto out;
|
2503
|
+
if (!hole) {
|
2504
|
+
ok = SetArrayElement(cx, nobj, slot - begin, tvr.u.value);
|
2505
|
+
if (!ok)
|
2506
|
+
goto out;
|
2507
|
+
}
|
2508
|
+
}
|
2509
|
+
ok = js_SetLengthProperty(cx, nobj, end - begin);
|
2510
|
+
|
2511
|
+
out:
|
2512
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
2513
|
+
return ok;
|
2514
|
+
}
|
2515
|
+
|
2516
|
+
#if JS_HAS_ARRAY_EXTRAS
|
2517
|
+
|
2518
|
+
static JSBool
|
2519
|
+
array_indexOfHelper(JSContext *cx, JSBool isLast, uintN argc, jsval *vp)
|
2520
|
+
{
|
2521
|
+
JSObject *obj;
|
2522
|
+
jsuint length, i, stop;
|
2523
|
+
jsint direction;
|
2524
|
+
JSBool hole;
|
2525
|
+
|
2526
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2527
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &length))
|
2528
|
+
return JS_FALSE;
|
2529
|
+
if (length == 0)
|
2530
|
+
goto not_found;
|
2531
|
+
|
2532
|
+
if (argc <= 1) {
|
2533
|
+
i = isLast ? length - 1 : 0;
|
2534
|
+
} else {
|
2535
|
+
jsdouble start;
|
2536
|
+
|
2537
|
+
start = js_ValueToNumber(cx, &vp[3]);
|
2538
|
+
if (JSVAL_IS_NULL(vp[3]))
|
2539
|
+
return JS_FALSE;
|
2540
|
+
start = js_DoubleToInteger(start);
|
2541
|
+
if (start < 0) {
|
2542
|
+
start += length;
|
2543
|
+
if (start < 0) {
|
2544
|
+
if (isLast)
|
2545
|
+
goto not_found;
|
2546
|
+
i = 0;
|
2547
|
+
} else {
|
2548
|
+
i = (jsuint)start;
|
2549
|
+
}
|
2550
|
+
} else if (start >= length) {
|
2551
|
+
if (!isLast)
|
2552
|
+
goto not_found;
|
2553
|
+
i = length - 1;
|
2554
|
+
} else {
|
2555
|
+
i = (jsuint)start;
|
2556
|
+
}
|
2557
|
+
}
|
2558
|
+
|
2559
|
+
if (isLast) {
|
2560
|
+
stop = 0;
|
2561
|
+
direction = -1;
|
2562
|
+
} else {
|
2563
|
+
stop = length - 1;
|
2564
|
+
direction = 1;
|
2565
|
+
}
|
2566
|
+
|
2567
|
+
for (;;) {
|
2568
|
+
if (!JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) ||
|
2569
|
+
!GetArrayElement(cx, obj, (jsuint)i, &hole, vp)) {
|
2570
|
+
return JS_FALSE;
|
2571
|
+
}
|
2572
|
+
if (!hole && js_StrictlyEqual(cx, *vp, vp[2]))
|
2573
|
+
return js_NewNumberInRootedValue(cx, i, vp);
|
2574
|
+
if (i == stop)
|
2575
|
+
goto not_found;
|
2576
|
+
i += direction;
|
2577
|
+
}
|
2578
|
+
|
2579
|
+
not_found:
|
2580
|
+
*vp = INT_TO_JSVAL(-1);
|
2581
|
+
return JS_TRUE;
|
2582
|
+
}
|
2583
|
+
|
2584
|
+
static JSBool
|
2585
|
+
array_indexOf(JSContext *cx, uintN argc, jsval *vp)
|
2586
|
+
{
|
2587
|
+
return array_indexOfHelper(cx, JS_FALSE, argc, vp);
|
2588
|
+
}
|
2589
|
+
|
2590
|
+
static JSBool
|
2591
|
+
array_lastIndexOf(JSContext *cx, uintN argc, jsval *vp)
|
2592
|
+
{
|
2593
|
+
return array_indexOfHelper(cx, JS_TRUE, argc, vp);
|
2594
|
+
}
|
2595
|
+
|
2596
|
+
/* Order is important; extras that take a predicate funarg must follow MAP. */
|
2597
|
+
typedef enum ArrayExtraMode {
|
2598
|
+
FOREACH,
|
2599
|
+
REDUCE,
|
2600
|
+
REDUCE_RIGHT,
|
2601
|
+
MAP,
|
2602
|
+
FILTER,
|
2603
|
+
SOME,
|
2604
|
+
EVERY
|
2605
|
+
} ArrayExtraMode;
|
2606
|
+
|
2607
|
+
#define REDUCE_MODE(mode) ((mode) == REDUCE || (mode) == REDUCE_RIGHT)
|
2608
|
+
|
2609
|
+
static JSBool
|
2610
|
+
array_extra(JSContext *cx, ArrayExtraMode mode, uintN argc, jsval *vp)
|
2611
|
+
{
|
2612
|
+
JSObject *obj;
|
2613
|
+
jsuint length, newlen;
|
2614
|
+
jsval *argv, *elemroot, *invokevp, *sp;
|
2615
|
+
JSBool ok, cond, hole;
|
2616
|
+
JSObject *callable, *thisp, *newarr;
|
2617
|
+
jsint start, end, step, i;
|
2618
|
+
void *mark;
|
2619
|
+
|
2620
|
+
obj = JS_THIS_OBJECT(cx, vp);
|
2621
|
+
if (!obj || !js_GetLengthProperty(cx, obj, &length))
|
2622
|
+
return JS_FALSE;
|
2623
|
+
|
2624
|
+
/*
|
2625
|
+
* First, get or compute our callee, so that we error out consistently
|
2626
|
+
* when passed a non-callable object.
|
2627
|
+
*/
|
2628
|
+
argv = vp + 2;
|
2629
|
+
callable = js_ValueToCallableObject(cx, &argv[0], JSV2F_SEARCH_STACK);
|
2630
|
+
if (!callable)
|
2631
|
+
return JS_FALSE;
|
2632
|
+
|
2633
|
+
/*
|
2634
|
+
* Set our initial return condition, used for zero-length array cases
|
2635
|
+
* (and pre-size our map return to match our known length, for all cases).
|
2636
|
+
*/
|
2637
|
+
#ifdef __GNUC__ /* quell GCC overwarning */
|
2638
|
+
newlen = 0;
|
2639
|
+
newarr = NULL;
|
2640
|
+
#endif
|
2641
|
+
start = 0, end = length, step = 1;
|
2642
|
+
|
2643
|
+
switch (mode) {
|
2644
|
+
case REDUCE_RIGHT:
|
2645
|
+
start = length - 1, end = -1, step = -1;
|
2646
|
+
/* FALL THROUGH */
|
2647
|
+
case REDUCE:
|
2648
|
+
if (length == 0 && argc == 1) {
|
2649
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
2650
|
+
JSMSG_EMPTY_ARRAY_REDUCE);
|
2651
|
+
return JS_FALSE;
|
2652
|
+
}
|
2653
|
+
if (argc >= 2) {
|
2654
|
+
*vp = argv[1];
|
2655
|
+
} else {
|
2656
|
+
do {
|
2657
|
+
if (!GetArrayElement(cx, obj, start, &hole, vp))
|
2658
|
+
return JS_FALSE;
|
2659
|
+
start += step;
|
2660
|
+
} while (hole && start != end);
|
2661
|
+
|
2662
|
+
if (hole && start == end) {
|
2663
|
+
JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL,
|
2664
|
+
JSMSG_EMPTY_ARRAY_REDUCE);
|
2665
|
+
return JS_FALSE;
|
2666
|
+
}
|
2667
|
+
}
|
2668
|
+
break;
|
2669
|
+
case MAP:
|
2670
|
+
case FILTER:
|
2671
|
+
newlen = (mode == MAP) ? length : 0;
|
2672
|
+
newarr = js_NewArrayObject(cx, newlen, NULL);
|
2673
|
+
if (!newarr)
|
2674
|
+
return JS_FALSE;
|
2675
|
+
*vp = OBJECT_TO_JSVAL(newarr);
|
2676
|
+
break;
|
2677
|
+
case SOME:
|
2678
|
+
*vp = JSVAL_FALSE;
|
2679
|
+
break;
|
2680
|
+
case EVERY:
|
2681
|
+
*vp = JSVAL_TRUE;
|
2682
|
+
break;
|
2683
|
+
case FOREACH:
|
2684
|
+
*vp = JSVAL_VOID;
|
2685
|
+
break;
|
2686
|
+
}
|
2687
|
+
|
2688
|
+
if (length == 0)
|
2689
|
+
return JS_TRUE;
|
2690
|
+
|
2691
|
+
if (argc > 1 && !REDUCE_MODE(mode)) {
|
2692
|
+
if (!js_ValueToObject(cx, argv[1], &thisp))
|
2693
|
+
return JS_FALSE;
|
2694
|
+
argv[1] = OBJECT_TO_JSVAL(thisp);
|
2695
|
+
} else {
|
2696
|
+
thisp = NULL;
|
2697
|
+
}
|
2698
|
+
|
2699
|
+
/*
|
2700
|
+
* For all but REDUCE, we call with 3 args (value, index, array). REDUCE
|
2701
|
+
* requires 4 args (accum, value, index, array).
|
2702
|
+
*/
|
2703
|
+
argc = 3 + REDUCE_MODE(mode);
|
2704
|
+
elemroot = js_AllocStack(cx, 1 + 2 + argc, &mark);
|
2705
|
+
if (!elemroot)
|
2706
|
+
return JS_FALSE;
|
2707
|
+
|
2708
|
+
/* From this point the control must flow through out:. */
|
2709
|
+
ok = JS_TRUE;
|
2710
|
+
invokevp = elemroot + 1;
|
2711
|
+
|
2712
|
+
for (i = start; i != end; i += step) {
|
2713
|
+
ok = JS_CHECK_OPERATION_LIMIT(cx, JSOW_JUMP) &&
|
2714
|
+
GetArrayElement(cx, obj, i, &hole, elemroot);
|
2715
|
+
if (!ok)
|
2716
|
+
goto out;
|
2717
|
+
if (hole)
|
2718
|
+
continue;
|
2719
|
+
|
2720
|
+
/*
|
2721
|
+
* Push callable and 'this', then args. We must do this for every
|
2722
|
+
* iteration around the loop since js_Invoke uses spbase[0] for return
|
2723
|
+
* value storage, while some native functions use spbase[1] for local
|
2724
|
+
* rooting.
|
2725
|
+
*/
|
2726
|
+
sp = invokevp;
|
2727
|
+
*sp++ = OBJECT_TO_JSVAL(callable);
|
2728
|
+
*sp++ = OBJECT_TO_JSVAL(thisp);
|
2729
|
+
if (REDUCE_MODE(mode))
|
2730
|
+
*sp++ = *vp;
|
2731
|
+
*sp++ = *elemroot;
|
2732
|
+
*sp++ = INT_TO_JSVAL(i);
|
2733
|
+
*sp++ = OBJECT_TO_JSVAL(obj);
|
2734
|
+
|
2735
|
+
/* Do the call. */
|
2736
|
+
ok = js_Invoke(cx, argc, invokevp, 0);
|
2737
|
+
if (!ok)
|
2738
|
+
break;
|
2739
|
+
|
2740
|
+
if (mode > MAP)
|
2741
|
+
cond = js_ValueToBoolean(*invokevp);
|
2742
|
+
#ifdef __GNUC__ /* quell GCC overwarning */
|
2743
|
+
else
|
2744
|
+
cond = JS_FALSE;
|
2745
|
+
#endif
|
2746
|
+
|
2747
|
+
switch (mode) {
|
2748
|
+
case FOREACH:
|
2749
|
+
break;
|
2750
|
+
case REDUCE:
|
2751
|
+
case REDUCE_RIGHT:
|
2752
|
+
*vp = *invokevp;
|
2753
|
+
break;
|
2754
|
+
case MAP:
|
2755
|
+
ok = SetArrayElement(cx, newarr, i, *invokevp);
|
2756
|
+
if (!ok)
|
2757
|
+
goto out;
|
2758
|
+
break;
|
2759
|
+
case FILTER:
|
2760
|
+
if (!cond)
|
2761
|
+
break;
|
2762
|
+
/* The filter passed *elemroot, so push it onto our result. */
|
2763
|
+
ok = SetArrayElement(cx, newarr, newlen++, *elemroot);
|
2764
|
+
if (!ok)
|
2765
|
+
goto out;
|
2766
|
+
break;
|
2767
|
+
case SOME:
|
2768
|
+
if (cond) {
|
2769
|
+
*vp = JSVAL_TRUE;
|
2770
|
+
goto out;
|
2771
|
+
}
|
2772
|
+
break;
|
2773
|
+
case EVERY:
|
2774
|
+
if (!cond) {
|
2775
|
+
*vp = JSVAL_FALSE;
|
2776
|
+
goto out;
|
2777
|
+
}
|
2778
|
+
break;
|
2779
|
+
}
|
2780
|
+
}
|
2781
|
+
|
2782
|
+
out:
|
2783
|
+
js_FreeStack(cx, mark);
|
2784
|
+
if (ok && mode == FILTER)
|
2785
|
+
ok = js_SetLengthProperty(cx, newarr, newlen);
|
2786
|
+
return ok;
|
2787
|
+
}
|
2788
|
+
|
2789
|
+
static JSBool
|
2790
|
+
array_forEach(JSContext *cx, uintN argc, jsval *vp)
|
2791
|
+
{
|
2792
|
+
return array_extra(cx, FOREACH, argc, vp);
|
2793
|
+
}
|
2794
|
+
|
2795
|
+
static JSBool
|
2796
|
+
array_map(JSContext *cx, uintN argc, jsval *vp)
|
2797
|
+
{
|
2798
|
+
return array_extra(cx, MAP, argc, vp);
|
2799
|
+
}
|
2800
|
+
|
2801
|
+
static JSBool
|
2802
|
+
array_reduce(JSContext *cx, uintN argc, jsval *vp)
|
2803
|
+
{
|
2804
|
+
return array_extra(cx, REDUCE, argc, vp);
|
2805
|
+
}
|
2806
|
+
|
2807
|
+
static JSBool
|
2808
|
+
array_reduceRight(JSContext *cx, uintN argc, jsval *vp)
|
2809
|
+
{
|
2810
|
+
return array_extra(cx, REDUCE_RIGHT, argc, vp);
|
2811
|
+
}
|
2812
|
+
|
2813
|
+
static JSBool
|
2814
|
+
array_filter(JSContext *cx, uintN argc, jsval *vp)
|
2815
|
+
{
|
2816
|
+
return array_extra(cx, FILTER, argc, vp);
|
2817
|
+
}
|
2818
|
+
|
2819
|
+
static JSBool
|
2820
|
+
array_some(JSContext *cx, uintN argc, jsval *vp)
|
2821
|
+
{
|
2822
|
+
return array_extra(cx, SOME, argc, vp);
|
2823
|
+
}
|
2824
|
+
|
2825
|
+
static JSBool
|
2826
|
+
array_every(JSContext *cx, uintN argc, jsval *vp)
|
2827
|
+
{
|
2828
|
+
return array_extra(cx, EVERY, argc, vp);
|
2829
|
+
}
|
2830
|
+
#endif
|
2831
|
+
|
2832
|
+
static JSPropertySpec array_props[] = {
|
2833
|
+
{js_length_str, -1, JSPROP_SHARED | JSPROP_PERMANENT,
|
2834
|
+
array_length_getter, array_length_setter},
|
2835
|
+
{0,0,0,0,0}
|
2836
|
+
};
|
2837
|
+
|
2838
|
+
static JSFunctionSpec array_methods[] = {
|
2839
|
+
#if JS_HAS_TOSOURCE
|
2840
|
+
JS_FN(js_toSource_str, array_toSource, 0,0,0),
|
2841
|
+
#endif
|
2842
|
+
JS_FN(js_toString_str, array_toString, 0,0,0),
|
2843
|
+
JS_FN(js_toLocaleString_str,array_toLocaleString,0,0,0),
|
2844
|
+
|
2845
|
+
/* Perl-ish methods. */
|
2846
|
+
JS_FN("join", array_join, 1,1,JSFUN_GENERIC_NATIVE),
|
2847
|
+
JS_FN("reverse", array_reverse, 0,0,JSFUN_GENERIC_NATIVE),
|
2848
|
+
JS_FN("sort", array_sort, 0,1,JSFUN_GENERIC_NATIVE),
|
2849
|
+
JS_FN("push", array_push, 1,1,JSFUN_GENERIC_NATIVE),
|
2850
|
+
JS_FN("pop", array_pop, 0,0,JSFUN_GENERIC_NATIVE),
|
2851
|
+
JS_FN("shift", array_shift, 0,0,JSFUN_GENERIC_NATIVE),
|
2852
|
+
JS_FN("unshift", array_unshift, 0,1,JSFUN_GENERIC_NATIVE),
|
2853
|
+
JS_FN("splice", array_splice, 0,2,JSFUN_GENERIC_NATIVE),
|
2854
|
+
|
2855
|
+
/* Pythonic sequence methods. */
|
2856
|
+
JS_FN("concat", array_concat, 0,1,JSFUN_GENERIC_NATIVE),
|
2857
|
+
JS_FN("slice", array_slice, 0,2,JSFUN_GENERIC_NATIVE),
|
2858
|
+
|
2859
|
+
#if JS_HAS_ARRAY_EXTRAS
|
2860
|
+
JS_FN("indexOf", array_indexOf, 1,1,JSFUN_GENERIC_NATIVE),
|
2861
|
+
JS_FN("lastIndexOf", array_lastIndexOf, 1,1,JSFUN_GENERIC_NATIVE),
|
2862
|
+
JS_FN("forEach", array_forEach, 1,1,JSFUN_GENERIC_NATIVE),
|
2863
|
+
JS_FN("map", array_map, 1,1,JSFUN_GENERIC_NATIVE),
|
2864
|
+
JS_FN("reduce", array_reduce, 1,1,JSFUN_GENERIC_NATIVE),
|
2865
|
+
JS_FN("reduceRight", array_reduceRight, 1,1,JSFUN_GENERIC_NATIVE),
|
2866
|
+
JS_FN("filter", array_filter, 1,1,JSFUN_GENERIC_NATIVE),
|
2867
|
+
JS_FN("some", array_some, 1,1,JSFUN_GENERIC_NATIVE),
|
2868
|
+
JS_FN("every", array_every, 1,1,JSFUN_GENERIC_NATIVE),
|
2869
|
+
#endif
|
2870
|
+
|
2871
|
+
JS_FS_END
|
2872
|
+
};
|
2873
|
+
|
2874
|
+
static JSBool
|
2875
|
+
Array(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
2876
|
+
{
|
2877
|
+
jsuint length;
|
2878
|
+
jsval *vector;
|
2879
|
+
|
2880
|
+
/* If called without new, replace obj with a new Array object. */
|
2881
|
+
if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) {
|
2882
|
+
obj = js_NewObject(cx, &js_ArrayClass, NULL, NULL, 0);
|
2883
|
+
if (!obj)
|
2884
|
+
return JS_FALSE;
|
2885
|
+
*rval = OBJECT_TO_JSVAL(obj);
|
2886
|
+
}
|
2887
|
+
|
2888
|
+
if (argc == 0) {
|
2889
|
+
length = 0;
|
2890
|
+
vector = NULL;
|
2891
|
+
} else if (argc > 1) {
|
2892
|
+
length = (jsuint) argc;
|
2893
|
+
vector = argv;
|
2894
|
+
} else if (!JSVAL_IS_NUMBER(argv[0])) {
|
2895
|
+
length = 1;
|
2896
|
+
vector = argv;
|
2897
|
+
} else {
|
2898
|
+
length = ValueIsLength(cx, &argv[0]);
|
2899
|
+
if (JSVAL_IS_NULL(argv[0]))
|
2900
|
+
return JS_FALSE;
|
2901
|
+
vector = NULL;
|
2902
|
+
}
|
2903
|
+
return InitArrayObject(cx, obj, length, vector);
|
2904
|
+
}
|
2905
|
+
|
2906
|
+
JSObject *
|
2907
|
+
js_InitArrayClass(JSContext *cx, JSObject *obj)
|
2908
|
+
{
|
2909
|
+
JSObject *proto;
|
2910
|
+
|
2911
|
+
/* Initialize the ops structure used by slow arrays */
|
2912
|
+
memcpy(&js_SlowArrayObjectOps, &js_ObjectOps, sizeof(JSObjectOps));
|
2913
|
+
js_SlowArrayObjectOps.trace = slowarray_trace;
|
2914
|
+
js_SlowArrayObjectOps.enumerate = slowarray_enumerate;
|
2915
|
+
js_SlowArrayObjectOps.call = NULL;
|
2916
|
+
|
2917
|
+
proto = JS_InitClass(cx, obj, NULL, &js_ArrayClass, Array, 1,
|
2918
|
+
array_props, array_methods, NULL, NULL);
|
2919
|
+
|
2920
|
+
/* Initialize the Array prototype object so it gets a length property. */
|
2921
|
+
if (!proto || !InitArrayObject(cx, proto, 0, NULL))
|
2922
|
+
return NULL;
|
2923
|
+
return proto;
|
2924
|
+
}
|
2925
|
+
|
2926
|
+
JSObject *
|
2927
|
+
js_NewArrayObject(JSContext *cx, jsuint length, jsval *vector)
|
2928
|
+
{
|
2929
|
+
JSTempValueRooter tvr;
|
2930
|
+
JSObject *obj;
|
2931
|
+
|
2932
|
+
obj = js_NewObject(cx, &js_ArrayClass, NULL, NULL, 0);
|
2933
|
+
if (!obj)
|
2934
|
+
return NULL;
|
2935
|
+
|
2936
|
+
JS_PUSH_TEMP_ROOT_OBJECT(cx, obj, &tvr);
|
2937
|
+
if (!InitArrayObject(cx, obj, length, vector))
|
2938
|
+
obj = NULL;
|
2939
|
+
JS_POP_TEMP_ROOT(cx, &tvr);
|
2940
|
+
|
2941
|
+
/* Set/clear newborn root, in case we lost it. */
|
2942
|
+
cx->weakRoots.newborn[GCX_OBJECT] = obj;
|
2943
|
+
return obj;
|
2944
|
+
}
|
2945
|
+
|
2946
|
+
JSObject *
|
2947
|
+
js_NewSlowArrayObject(JSContext *cx)
|
2948
|
+
{
|
2949
|
+
JSObject *obj = js_NewObject(cx, &js_SlowArrayClass, NULL, NULL, 0);
|
2950
|
+
if (obj)
|
2951
|
+
obj->fslots[JSSLOT_ARRAY_LENGTH] = 0;
|
2952
|
+
return obj;
|
2953
|
+
}
|
2954
|
+
|
2955
|
+
#ifdef DEBUG_ARRAYS
|
2956
|
+
JSBool
|
2957
|
+
js_ArrayInfo(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
|
2958
|
+
{
|
2959
|
+
uintN i;
|
2960
|
+
JSObject *array;
|
2961
|
+
|
2962
|
+
for (i = 0; i < argc; i++) {
|
2963
|
+
char *bytes;
|
2964
|
+
|
2965
|
+
bytes = js_DecompileValueGenerator(cx, JSDVG_SEARCH_STACK, argv[i],
|
2966
|
+
NULL);
|
2967
|
+
if (!bytes)
|
2968
|
+
return JS_FALSE;
|
2969
|
+
if (JSVAL_IS_PRIMITIVE(argv[i]) ||
|
2970
|
+
!OBJ_IS_ARRAY(cx, (array = JSVAL_TO_OBJECT(argv[i])))) {
|
2971
|
+
fprintf(stderr, "%s: not array\n", bytes);
|
2972
|
+
JS_free(cx, bytes);
|
2973
|
+
continue;
|
2974
|
+
}
|
2975
|
+
fprintf(stderr, "%s: %s (len %lu", bytes,
|
2976
|
+
OBJ_IS_DENSE_ARRAY(cx, array) ? "dense" : "sparse",
|
2977
|
+
array->fslots[JSSLOT_ARRAY_LENGTH]);
|
2978
|
+
if (OBJ_IS_DENSE_ARRAY(cx, array)) {
|
2979
|
+
fprintf(stderr, ", count %lu, denselen %lu",
|
2980
|
+
array->fslots[JSSLOT_ARRAY_COUNT],
|
2981
|
+
ARRAY_DENSE_LENGTH(array));
|
2982
|
+
}
|
2983
|
+
fputs(")\n", stderr);
|
2984
|
+
JS_free(cx, bytes);
|
2985
|
+
}
|
2986
|
+
return JS_TRUE;
|
2987
|
+
}
|
2988
|
+
#endif
|