Make the runtime entry for setting/changing accessors "atomic".
Previously, there were 1 or 2 calls to the runtime when accessors were changed
or set. This doesn't really work well with property attributes, leading to some
hacks and complicates things even further when trying to share maps in presence
of accessors. Therefore, the runtime entry now takes the full triple (getter,
setter, attributes), where the getter and/or the setter can be null in case they
shouldn't be changed.
For now, we do basically the same on the native side as we did before on the
JavaScript side, but this will change in future CLs, the current CL is already
large enough.
Note that object literals with a getter and a setter for the same property still
do 2 calls, but this is a little bit more tricky to fix and will be handled in a
separate CL.
Review URL: https://chromiumcodereview.appspot.com/9616016
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@10956 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-03-07 13:24:44 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2008-07-03 15:10:15 +00:00
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are
|
|
|
|
// met:
|
|
|
|
//
|
|
|
|
// * Redistributions of source code must retain the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above
|
|
|
|
// copyright notice, this list of conditions and the following
|
|
|
|
// disclaimer in the documentation and/or other materials provided
|
|
|
|
// with the distribution.
|
|
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
|
|
// contributors may be used to endorse or promote products derived
|
|
|
|
// from this software without specific prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
// This file relies on the fact that the following declarations have been made
|
2008-10-03 07:14:31 +00:00
|
|
|
//
|
2008-07-03 15:10:15 +00:00
|
|
|
// in runtime.js:
|
2012-02-20 13:48:24 +00:00
|
|
|
// var $Object = global.Object;
|
|
|
|
// var $Boolean = global.Boolean;
|
|
|
|
// var $Number = global.Number;
|
|
|
|
// var $Function = global.Function;
|
|
|
|
// var $Array = global.Array;
|
|
|
|
// var $NaN = 0/0;
|
2008-10-03 07:14:31 +00:00
|
|
|
//
|
|
|
|
// in math.js:
|
2012-02-20 13:48:24 +00:00
|
|
|
// var $floor = MathFloor
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-02-20 13:48:24 +00:00
|
|
|
var $isNaN = GlobalIsNaN;
|
|
|
|
var $isFinite = GlobalIsFinite;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// Helper function used to install functions on objects.
|
|
|
|
function InstallFunctions(object, attributes, functions) {
|
2009-08-13 12:35:59 +00:00
|
|
|
if (functions.length >= 8) {
|
2009-08-18 09:47:45 +00:00
|
|
|
%OptimizeObjectForAddingMultipleProperties(object, functions.length >> 1);
|
2009-08-13 12:35:59 +00:00
|
|
|
}
|
2008-10-03 18:00:28 +00:00
|
|
|
for (var i = 0; i < functions.length; i += 2) {
|
|
|
|
var key = functions[i];
|
|
|
|
var f = functions[i + 1];
|
|
|
|
%FunctionSetName(f, key);
|
2010-04-28 12:05:40 +00:00
|
|
|
%FunctionRemovePrototype(f);
|
2008-10-03 18:00:28 +00:00
|
|
|
%SetProperty(object, key, f, attributes);
|
2011-05-30 13:49:22 +00:00
|
|
|
%SetNativeFlag(f);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2010-02-17 11:26:00 +00:00
|
|
|
%ToFastProperties(object);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-11-06 18:14:45 +00:00
|
|
|
|
|
|
|
// Helper function to install a getter only property.
|
|
|
|
function InstallGetter(object, name, getter) {
|
|
|
|
%FunctionSetName(getter, name);
|
|
|
|
%FunctionRemovePrototype(getter);
|
|
|
|
%DefineOrRedefineAccessorProperty(object, name, getter, null, DONT_ENUM);
|
|
|
|
%SetNativeFlag(getter);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-06 12:32:36 +00:00
|
|
|
// Prevents changes to the prototype of a built-in function.
|
2011-09-05 07:30:35 +00:00
|
|
|
// The "prototype" property of the function object is made non-configurable,
|
|
|
|
// and the prototype object is made non-extensible. The latter prevents
|
|
|
|
// changing the __proto__ property.
|
|
|
|
function SetUpLockedPrototype(constructor, fields, methods) {
|
|
|
|
%CheckIsBootstrapping();
|
|
|
|
var prototype = constructor.prototype;
|
|
|
|
// Install functions first, because this function is used to initialize
|
|
|
|
// PropertyDescriptor itself.
|
|
|
|
var property_count = (methods.length >> 1) + (fields ? fields.length : 0);
|
|
|
|
if (property_count >= 4) {
|
|
|
|
%OptimizeObjectForAddingMultipleProperties(prototype, property_count);
|
|
|
|
}
|
|
|
|
if (fields) {
|
|
|
|
for (var i = 0; i < fields.length; i++) {
|
|
|
|
%SetProperty(prototype, fields[i], void 0, DONT_ENUM | DONT_DELETE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (var i = 0; i < methods.length; i += 2) {
|
|
|
|
var key = methods[i];
|
|
|
|
var f = methods[i + 1];
|
|
|
|
%SetProperty(prototype, key, f, DONT_ENUM | DONT_DELETE | READ_ONLY);
|
|
|
|
%SetNativeFlag(f);
|
|
|
|
}
|
|
|
|
prototype.__proto__ = null;
|
|
|
|
%ToFastProperties(prototype);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA 262 - 15.1.4
|
2008-10-03 18:00:28 +00:00
|
|
|
function GlobalIsNaN(number) {
|
2011-09-05 11:08:57 +00:00
|
|
|
if (!IS_NUMBER(number)) number = NonNumberToNumber(number);
|
|
|
|
return NUMBER_IS_NAN(number);
|
2008-10-03 07:14:31 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA 262 - 15.1.5
|
2008-10-03 18:00:28 +00:00
|
|
|
function GlobalIsFinite(number) {
|
2011-01-05 13:52:00 +00:00
|
|
|
if (!IS_NUMBER(number)) number = NonNumberToNumber(number);
|
2011-09-05 11:08:57 +00:00
|
|
|
return NUMBER_IS_FINITE(number);
|
2008-10-03 07:14:31 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 - 15.1.2.2
|
2008-10-03 18:00:28 +00:00
|
|
|
function GlobalParseInt(string, radix) {
|
2011-02-22 11:21:15 +00:00
|
|
|
if (IS_UNDEFINED(radix) || radix === 10 || radix === 0) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Some people use parseInt instead of Math.floor. This
|
|
|
|
// optimization makes parseInt on a Smi 12 times faster (60ns
|
|
|
|
// vs 800ns). The following optimization makes parseInt on a
|
|
|
|
// non-Smi number 9 times faster (230ns vs 2070ns). Together
|
|
|
|
// they make parseInt on a string 1.4% slower (274ns vs 270ns).
|
|
|
|
if (%_IsSmi(string)) return string;
|
2009-02-16 10:18:34 +00:00
|
|
|
if (IS_NUMBER(string) &&
|
2009-11-17 13:54:05 +00:00
|
|
|
((0.01 < string && string < 1e9) ||
|
|
|
|
(-1e9 < string && string < -0.01))) {
|
2009-02-16 10:18:34 +00:00
|
|
|
// Truncate number.
|
|
|
|
return string | 0;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-09-02 11:38:40 +00:00
|
|
|
string = TO_STRING_INLINE(string);
|
2011-06-28 12:31:42 +00:00
|
|
|
radix = radix | 0;
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
2011-09-02 11:38:40 +00:00
|
|
|
// The spec says ToString should be evaluated before ToInt32.
|
|
|
|
string = TO_STRING_INLINE(string);
|
2008-07-03 15:10:15 +00:00
|
|
|
radix = TO_INT32(radix);
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!(radix == 0 || (2 <= radix && radix <= 36))) {
|
2008-07-03 15:10:15 +00:00
|
|
|
return $NaN;
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-09-02 11:38:40 +00:00
|
|
|
|
2010-08-27 11:47:12 +00:00
|
|
|
if (%_HasCachedArrayIndex(string) &&
|
|
|
|
(radix == 0 || radix == 10)) {
|
|
|
|
return %_GetCachedArrayIndex(string);
|
|
|
|
}
|
|
|
|
return %StringParseInt(string, radix);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 - 15.1.2.3
|
2008-10-03 18:00:28 +00:00
|
|
|
function GlobalParseFloat(string) {
|
2010-08-27 11:47:12 +00:00
|
|
|
string = TO_STRING_INLINE(string);
|
|
|
|
if (%_HasCachedArrayIndex(string)) return %_GetCachedArrayIndex(string);
|
|
|
|
return %StringParseFloat(string);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
|
|
|
|
2008-11-21 15:01:41 +00:00
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function GlobalEval(x) {
|
|
|
|
if (!IS_STRING(x)) return x;
|
|
|
|
|
2009-06-08 09:46:09 +00:00
|
|
|
var global_receiver = %GlobalReceiver(global);
|
|
|
|
var global_is_detached = (global === global_receiver);
|
|
|
|
|
2011-05-30 13:49:22 +00:00
|
|
|
// For consistency with JSC we require the global object passed to
|
|
|
|
// eval to be the global object from which 'eval' originated. This
|
|
|
|
// is not mandated by the spec.
|
2011-10-31 09:38:52 +00:00
|
|
|
// We only throw if the global has been detached, since we need the
|
|
|
|
// receiver as this-value for the call.
|
|
|
|
if (global_is_detached) {
|
|
|
|
throw new $EvalError('The "this" value passed to eval must ' +
|
2008-11-14 14:36:46 +00:00
|
|
|
'be the global object from which eval originated');
|
2008-11-14 13:14:49 +00:00
|
|
|
}
|
2009-06-08 09:46:09 +00:00
|
|
|
|
2010-10-27 09:19:43 +00:00
|
|
|
var f = %CompileString(x);
|
2008-10-03 18:00:28 +00:00
|
|
|
if (!IS_FUNCTION(f)) return f;
|
|
|
|
|
2011-10-31 09:38:52 +00:00
|
|
|
return %_CallFunction(global_receiver, f);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
// Set up global object.
|
|
|
|
function SetUpGlobal() {
|
|
|
|
%CheckIsBootstrapping();
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA 262 - 15.1.1.1.
|
2011-09-15 12:00:30 +00:00
|
|
|
%SetProperty(global, "NaN", $NaN, DONT_ENUM | DONT_DELETE | READ_ONLY);
|
2008-10-03 18:00:28 +00:00
|
|
|
|
|
|
|
// ECMA-262 - 15.1.1.2.
|
2011-09-15 12:00:30 +00:00
|
|
|
%SetProperty(global, "Infinity", 1/0, DONT_ENUM | DONT_DELETE | READ_ONLY);
|
2008-10-03 18:00:28 +00:00
|
|
|
|
|
|
|
// ECMA-262 - 15.1.1.3.
|
2011-09-15 12:00:30 +00:00
|
|
|
%SetProperty(global, "undefined", void 0,
|
|
|
|
DONT_ENUM | DONT_DELETE | READ_ONLY);
|
2009-06-29 08:14:06 +00:00
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
// Set up non-enumerable function on the global object.
|
2008-10-03 18:00:28 +00:00
|
|
|
InstallFunctions(global, DONT_ENUM, $Array(
|
|
|
|
"isNaN", GlobalIsNaN,
|
|
|
|
"isFinite", GlobalIsFinite,
|
|
|
|
"parseInt", GlobalParseInt,
|
|
|
|
"parseFloat", GlobalParseFloat,
|
2011-05-19 08:10:27 +00:00
|
|
|
"eval", GlobalEval
|
2008-10-03 18:00:28 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
SetUpGlobal();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Boolean (first part of definition)
|
|
|
|
|
|
|
|
|
|
|
|
%SetCode($Boolean, function(x) {
|
2009-06-29 08:14:06 +00:00
|
|
|
if (%_IsConstructCall()) {
|
2008-07-03 15:10:15 +00:00
|
|
|
%_SetValueOf(this, ToBoolean(x));
|
|
|
|
} else {
|
|
|
|
return ToBoolean(x);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
%FunctionSetPrototype($Boolean, new $Boolean(false));
|
|
|
|
|
2008-10-03 12:14:29 +00:00
|
|
|
%SetProperty($Boolean.prototype, "constructor", $Boolean, DONT_ENUM);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Object
|
|
|
|
|
|
|
|
$Object.prototype.constructor = $Object;
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 - 15.2.4.2
|
|
|
|
function ObjectToString() {
|
2013-03-01 13:28:55 +00:00
|
|
|
if (IS_UNDEFINED(this) && !IS_UNDETECTABLE(this)) return "[object Undefined]";
|
|
|
|
if (IS_NULL(this)) return "[object Null]";
|
|
|
|
if (IS_SYMBOL(this)) return "[object Symbol]";
|
2010-01-19 14:15:47 +00:00
|
|
|
return "[object " + %_ClassOf(ToObject(this)) + "]";
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 - 15.2.4.3
|
|
|
|
function ObjectToLocaleString() {
|
2011-05-05 05:21:30 +00:00
|
|
|
if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) {
|
|
|
|
throw MakeTypeError("called_on_null_or_undefined",
|
|
|
|
["Object.prototype.toLocaleString"]);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
return this.toString();
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 - 15.2.4.4
|
|
|
|
function ObjectValueOf() {
|
2010-01-19 14:15:47 +00:00
|
|
|
return ToObject(this);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 - 15.2.4.5
|
|
|
|
function ObjectHasOwnProperty(V) {
|
2011-07-21 11:20:27 +00:00
|
|
|
if (%IsJSProxy(this)) {
|
|
|
|
var handler = %GetHandler(this);
|
|
|
|
return CallTrap1(handler, "hasOwn", DerivedHasOwnTrap, TO_STRING_INLINE(V));
|
|
|
|
}
|
2011-05-12 16:19:03 +00:00
|
|
|
return %HasLocalProperty(TO_OBJECT_INLINE(this), TO_STRING_INLINE(V));
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 - 15.2.4.6
|
|
|
|
function ObjectIsPrototypeOf(V) {
|
2011-05-05 05:21:30 +00:00
|
|
|
if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) {
|
|
|
|
throw MakeTypeError("called_on_null_or_undefined",
|
|
|
|
["Object.prototype.isPrototypeOf"]);
|
|
|
|
}
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(V)) return false;
|
2008-07-03 15:10:15 +00:00
|
|
|
return %IsInPrototypeChain(this, V);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 - 15.2.4.6
|
|
|
|
function ObjectPropertyIsEnumerable(V) {
|
2011-07-21 11:20:27 +00:00
|
|
|
var P = ToString(V);
|
|
|
|
if (%IsJSProxy(this)) {
|
|
|
|
var desc = GetOwnProperty(this, P);
|
|
|
|
return IS_UNDEFINED(desc) ? false : desc.isEnumerable();
|
|
|
|
}
|
|
|
|
return %IsPropertyEnumerable(ToObject(this), P);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Extensions for providing property getters and setters.
|
2008-10-03 18:00:28 +00:00
|
|
|
function ObjectDefineGetter(name, fun) {
|
2011-05-30 13:49:22 +00:00
|
|
|
var receiver = this;
|
|
|
|
if (receiver == null && !IS_UNDETECTABLE(receiver)) {
|
|
|
|
receiver = %GlobalReceiver(global);
|
2008-10-03 12:14:29 +00:00
|
|
|
}
|
2011-09-13 11:42:57 +00:00
|
|
|
if (!IS_SPEC_FUNCTION(fun)) {
|
2011-11-28 12:11:00 +00:00
|
|
|
throw new $TypeError(
|
|
|
|
'Object.prototype.__defineGetter__: Expecting function');
|
2008-10-03 12:14:29 +00:00
|
|
|
}
|
2011-03-15 12:25:13 +00:00
|
|
|
var desc = new PropertyDescriptor();
|
|
|
|
desc.setGet(fun);
|
|
|
|
desc.setEnumerable(true);
|
|
|
|
desc.setConfigurable(true);
|
2011-05-30 13:49:22 +00:00
|
|
|
DefineOwnProperty(ToObject(receiver), ToString(name), desc, false);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function ObjectLookupGetter(name) {
|
2011-05-30 13:49:22 +00:00
|
|
|
var receiver = this;
|
|
|
|
if (receiver == null && !IS_UNDETECTABLE(receiver)) {
|
|
|
|
receiver = %GlobalReceiver(global);
|
2008-10-03 12:14:29 +00:00
|
|
|
}
|
2011-05-30 13:49:22 +00:00
|
|
|
return %LookupAccessor(ToObject(receiver), ToString(name), GETTER);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function ObjectDefineSetter(name, fun) {
|
2011-05-30 13:49:22 +00:00
|
|
|
var receiver = this;
|
|
|
|
if (receiver == null && !IS_UNDETECTABLE(receiver)) {
|
|
|
|
receiver = %GlobalReceiver(global);
|
2008-10-03 12:14:29 +00:00
|
|
|
}
|
2011-09-13 11:42:57 +00:00
|
|
|
if (!IS_SPEC_FUNCTION(fun)) {
|
2008-10-03 12:14:29 +00:00
|
|
|
throw new $TypeError(
|
|
|
|
'Object.prototype.__defineSetter__: Expecting function');
|
|
|
|
}
|
2011-03-15 12:25:13 +00:00
|
|
|
var desc = new PropertyDescriptor();
|
|
|
|
desc.setSet(fun);
|
|
|
|
desc.setEnumerable(true);
|
|
|
|
desc.setConfigurable(true);
|
2011-05-30 13:49:22 +00:00
|
|
|
DefineOwnProperty(ToObject(receiver), ToString(name), desc, false);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function ObjectLookupSetter(name) {
|
2011-05-30 13:49:22 +00:00
|
|
|
var receiver = this;
|
|
|
|
if (receiver == null && !IS_UNDETECTABLE(receiver)) {
|
|
|
|
receiver = %GlobalReceiver(global);
|
2008-10-03 12:14:29 +00:00
|
|
|
}
|
2011-05-30 13:49:22 +00:00
|
|
|
return %LookupAccessor(ToObject(receiver), ToString(name), SETTER);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2009-09-15 11:51:40 +00:00
|
|
|
function ObjectKeys(obj) {
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.keys"]);
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2011-07-13 11:01:17 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
var handler = %GetHandler(obj);
|
2011-07-21 11:20:27 +00:00
|
|
|
var names = CallTrap0(handler, "keys", DerivedKeysTrap);
|
2012-05-29 13:57:55 +00:00
|
|
|
return ToStringArray(names, "keys");
|
2011-07-13 11:01:17 +00:00
|
|
|
}
|
2009-09-15 11:51:40 +00:00
|
|
|
return %LocalKeys(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-09 13:56:58 +00:00
|
|
|
// ES5 8.10.1.
|
|
|
|
function IsAccessorDescriptor(desc) {
|
|
|
|
if (IS_UNDEFINED(desc)) return false;
|
2011-07-07 12:41:20 +00:00
|
|
|
return desc.hasGetter() || desc.hasSetter();
|
2009-12-09 13:56:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ES5 8.10.2.
|
|
|
|
function IsDataDescriptor(desc) {
|
|
|
|
if (IS_UNDEFINED(desc)) return false;
|
2011-07-07 12:41:20 +00:00
|
|
|
return desc.hasValue() || desc.hasWritable();
|
2009-12-09 13:56:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ES5 8.10.3.
|
|
|
|
function IsGenericDescriptor(desc) {
|
2011-10-20 07:55:30 +00:00
|
|
|
if (IS_UNDEFINED(desc)) return false;
|
2009-12-09 13:56:58 +00:00
|
|
|
return !(IsAccessorDescriptor(desc) || IsDataDescriptor(desc));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function IsInconsistentDescriptor(desc) {
|
|
|
|
return IsAccessorDescriptor(desc) && IsDataDescriptor(desc);
|
|
|
|
}
|
|
|
|
|
2011-06-03 10:15:49 +00:00
|
|
|
|
2010-01-13 12:10:57 +00:00
|
|
|
// ES5 8.10.4
|
|
|
|
function FromPropertyDescriptor(desc) {
|
2010-02-03 13:10:03 +00:00
|
|
|
if (IS_UNDEFINED(desc)) return desc;
|
2011-08-25 09:07:43 +00:00
|
|
|
|
2010-01-13 12:10:57 +00:00
|
|
|
if (IsDataDescriptor(desc)) {
|
2011-08-25 09:07:43 +00:00
|
|
|
return { value: desc.getValue(),
|
|
|
|
writable: desc.isWritable(),
|
|
|
|
enumerable: desc.isEnumerable(),
|
|
|
|
configurable: desc.isConfigurable() };
|
2010-01-13 12:10:57 +00:00
|
|
|
}
|
2011-08-25 09:07:43 +00:00
|
|
|
// Must be an AccessorDescriptor then. We never return a generic descriptor.
|
|
|
|
return { get: desc.getGet(),
|
|
|
|
set: desc.getSet(),
|
|
|
|
enumerable: desc.isEnumerable(),
|
|
|
|
configurable: desc.isConfigurable() };
|
2010-01-13 12:10:57 +00:00
|
|
|
}
|
2009-12-09 13:56:58 +00:00
|
|
|
|
2011-08-25 09:07:43 +00:00
|
|
|
|
2011-07-07 12:41:20 +00:00
|
|
|
// Harmony Proxies
|
|
|
|
function FromGenericPropertyDescriptor(desc) {
|
|
|
|
if (IS_UNDEFINED(desc)) return desc;
|
|
|
|
var obj = new $Object();
|
2011-08-25 09:07:43 +00:00
|
|
|
|
|
|
|
if (desc.hasValue()) {
|
|
|
|
%IgnoreAttributesAndSetProperty(obj, "value", desc.getValue(), NONE);
|
|
|
|
}
|
|
|
|
if (desc.hasWritable()) {
|
|
|
|
%IgnoreAttributesAndSetProperty(obj, "writable", desc.isWritable(), NONE);
|
|
|
|
}
|
|
|
|
if (desc.hasGetter()) {
|
|
|
|
%IgnoreAttributesAndSetProperty(obj, "get", desc.getGet(), NONE);
|
|
|
|
}
|
|
|
|
if (desc.hasSetter()) {
|
|
|
|
%IgnoreAttributesAndSetProperty(obj, "set", desc.getSet(), NONE);
|
|
|
|
}
|
|
|
|
if (desc.hasEnumerable()) {
|
|
|
|
%IgnoreAttributesAndSetProperty(obj, "enumerable",
|
|
|
|
desc.isEnumerable(), NONE);
|
|
|
|
}
|
|
|
|
if (desc.hasConfigurable()) {
|
|
|
|
%IgnoreAttributesAndSetProperty(obj, "configurable",
|
|
|
|
desc.isConfigurable(), NONE);
|
|
|
|
}
|
2011-07-07 12:41:20 +00:00
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2011-08-25 09:07:43 +00:00
|
|
|
|
2009-12-09 13:56:58 +00:00
|
|
|
// ES5 8.10.5.
|
|
|
|
function ToPropertyDescriptor(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2009-12-09 13:56:58 +00:00
|
|
|
throw MakeTypeError("property_desc_object", [obj]);
|
|
|
|
}
|
|
|
|
var desc = new PropertyDescriptor();
|
|
|
|
|
|
|
|
if ("enumerable" in obj) {
|
|
|
|
desc.setEnumerable(ToBoolean(obj.enumerable));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ("configurable" in obj) {
|
|
|
|
desc.setConfigurable(ToBoolean(obj.configurable));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ("value" in obj) {
|
|
|
|
desc.setValue(obj.value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ("writable" in obj) {
|
|
|
|
desc.setWritable(ToBoolean(obj.writable));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ("get" in obj) {
|
|
|
|
var get = obj.get;
|
2011-09-13 11:42:57 +00:00
|
|
|
if (!IS_UNDEFINED(get) && !IS_SPEC_FUNCTION(get)) {
|
2009-12-09 13:56:58 +00:00
|
|
|
throw MakeTypeError("getter_must_be_callable", [get]);
|
|
|
|
}
|
|
|
|
desc.setGet(get);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ("set" in obj) {
|
|
|
|
var set = obj.set;
|
2011-09-13 11:42:57 +00:00
|
|
|
if (!IS_UNDEFINED(set) && !IS_SPEC_FUNCTION(set)) {
|
2009-12-09 13:56:58 +00:00
|
|
|
throw MakeTypeError("setter_must_be_callable", [set]);
|
|
|
|
}
|
|
|
|
desc.setSet(set);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IsInconsistentDescriptor(desc)) {
|
|
|
|
throw MakeTypeError("value_and_accessor", [obj]);
|
|
|
|
}
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-03 10:15:49 +00:00
|
|
|
// For Harmony proxies.
|
|
|
|
function ToCompletePropertyDescriptor(obj) {
|
2011-11-28 12:11:00 +00:00
|
|
|
var desc = ToPropertyDescriptor(obj);
|
2011-06-03 10:15:49 +00:00
|
|
|
if (IsGenericDescriptor(desc) || IsDataDescriptor(desc)) {
|
2011-07-07 12:41:20 +00:00
|
|
|
if (!desc.hasValue()) desc.setValue(void 0);
|
|
|
|
if (!desc.hasWritable()) desc.setWritable(false);
|
2011-06-03 10:15:49 +00:00
|
|
|
} else {
|
|
|
|
// Is accessor descriptor.
|
2011-07-07 12:41:20 +00:00
|
|
|
if (!desc.hasGetter()) desc.setGet(void 0);
|
|
|
|
if (!desc.hasSetter()) desc.setSet(void 0);
|
2011-06-03 10:15:49 +00:00
|
|
|
}
|
2011-07-07 12:41:20 +00:00
|
|
|
if (!desc.hasEnumerable()) desc.setEnumerable(false);
|
|
|
|
if (!desc.hasConfigurable()) desc.setConfigurable(false);
|
2011-06-03 10:15:49 +00:00
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-09 13:56:58 +00:00
|
|
|
function PropertyDescriptor() {
|
|
|
|
// Initialize here so they are all in-object and have the same map.
|
|
|
|
// Default values from ES5 8.6.1.
|
|
|
|
this.value_ = void 0;
|
|
|
|
this.hasValue_ = false;
|
|
|
|
this.writable_ = false;
|
|
|
|
this.hasWritable_ = false;
|
|
|
|
this.enumerable_ = false;
|
2010-02-03 13:10:03 +00:00
|
|
|
this.hasEnumerable_ = false;
|
2009-12-09 13:56:58 +00:00
|
|
|
this.configurable_ = false;
|
2010-02-03 13:10:03 +00:00
|
|
|
this.hasConfigurable_ = false;
|
2009-12-09 13:56:58 +00:00
|
|
|
this.get_ = void 0;
|
|
|
|
this.hasGetter_ = false;
|
|
|
|
this.set_ = void 0;
|
|
|
|
this.hasSetter_ = false;
|
|
|
|
}
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
SetUpLockedPrototype(PropertyDescriptor, $Array(
|
|
|
|
"value_",
|
|
|
|
"hasValue_",
|
|
|
|
"writable_",
|
|
|
|
"hasWritable_",
|
|
|
|
"enumerable_",
|
|
|
|
"hasEnumerable_",
|
|
|
|
"configurable_",
|
|
|
|
"hasConfigurable_",
|
|
|
|
"get_",
|
|
|
|
"hasGetter_",
|
|
|
|
"set_",
|
|
|
|
"hasSetter_"
|
|
|
|
), $Array(
|
|
|
|
"toString", function() {
|
|
|
|
return "[object PropertyDescriptor]";
|
|
|
|
},
|
|
|
|
"setValue", function(value) {
|
|
|
|
this.value_ = value;
|
|
|
|
this.hasValue_ = true;
|
|
|
|
},
|
|
|
|
"getValue", function() {
|
|
|
|
return this.value_;
|
|
|
|
},
|
|
|
|
"hasValue", function() {
|
|
|
|
return this.hasValue_;
|
|
|
|
},
|
|
|
|
"setEnumerable", function(enumerable) {
|
|
|
|
this.enumerable_ = enumerable;
|
|
|
|
this.hasEnumerable_ = true;
|
|
|
|
},
|
|
|
|
"isEnumerable", function () {
|
|
|
|
return this.enumerable_;
|
|
|
|
},
|
|
|
|
"hasEnumerable", function() {
|
|
|
|
return this.hasEnumerable_;
|
|
|
|
},
|
|
|
|
"setWritable", function(writable) {
|
|
|
|
this.writable_ = writable;
|
|
|
|
this.hasWritable_ = true;
|
|
|
|
},
|
|
|
|
"isWritable", function() {
|
|
|
|
return this.writable_;
|
|
|
|
},
|
|
|
|
"hasWritable", function() {
|
|
|
|
return this.hasWritable_;
|
|
|
|
},
|
|
|
|
"setConfigurable", function(configurable) {
|
|
|
|
this.configurable_ = configurable;
|
|
|
|
this.hasConfigurable_ = true;
|
|
|
|
},
|
|
|
|
"hasConfigurable", function() {
|
|
|
|
return this.hasConfigurable_;
|
|
|
|
},
|
|
|
|
"isConfigurable", function() {
|
|
|
|
return this.configurable_;
|
|
|
|
},
|
|
|
|
"setGet", function(get) {
|
|
|
|
this.get_ = get;
|
|
|
|
this.hasGetter_ = true;
|
|
|
|
},
|
|
|
|
"getGet", function() {
|
|
|
|
return this.get_;
|
|
|
|
},
|
|
|
|
"hasGetter", function() {
|
|
|
|
return this.hasGetter_;
|
|
|
|
},
|
|
|
|
"setSet", function(set) {
|
|
|
|
this.set_ = set;
|
|
|
|
this.hasSetter_ = true;
|
|
|
|
},
|
|
|
|
"getSet", function() {
|
|
|
|
return this.set_;
|
|
|
|
},
|
|
|
|
"hasSetter", function() {
|
|
|
|
return this.hasSetter_;
|
|
|
|
}));
|
2010-02-03 13:10:03 +00:00
|
|
|
|
|
|
|
|
2011-02-04 12:14:56 +00:00
|
|
|
// Converts an array returned from Runtime_GetOwnProperty to an actual
|
|
|
|
// property descriptor. For a description of the array layout please
|
|
|
|
// see the runtime.cc file.
|
|
|
|
function ConvertDescriptorArrayToDescriptor(desc_array) {
|
2011-03-11 13:57:20 +00:00
|
|
|
if (desc_array === false) {
|
2011-02-04 12:14:56 +00:00
|
|
|
throw 'Internal error: invalid desc_array';
|
|
|
|
}
|
2010-02-03 13:10:03 +00:00
|
|
|
|
2011-02-04 12:14:56 +00:00
|
|
|
if (IS_UNDEFINED(desc_array)) {
|
|
|
|
return void 0;
|
|
|
|
}
|
2010-01-13 12:10:57 +00:00
|
|
|
|
2011-02-04 12:14:56 +00:00
|
|
|
var desc = new PropertyDescriptor();
|
|
|
|
// This is an accessor.
|
|
|
|
if (desc_array[IS_ACCESSOR_INDEX]) {
|
|
|
|
desc.setGet(desc_array[GETTER_INDEX]);
|
|
|
|
desc.setSet(desc_array[SETTER_INDEX]);
|
2010-01-13 12:10:57 +00:00
|
|
|
} else {
|
2011-02-04 12:14:56 +00:00
|
|
|
desc.setValue(desc_array[VALUE_INDEX]);
|
|
|
|
desc.setWritable(desc_array[WRITABLE_INDEX]);
|
2010-01-13 12:10:57 +00:00
|
|
|
}
|
2011-02-04 12:14:56 +00:00
|
|
|
desc.setEnumerable(desc_array[ENUMERABLE_INDEX]);
|
|
|
|
desc.setConfigurable(desc_array[CONFIGURABLE_INDEX]);
|
2010-01-13 12:10:57 +00:00
|
|
|
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-21 11:20:27 +00:00
|
|
|
// For Harmony proxies.
|
|
|
|
function GetTrap(handler, name, defaultTrap) {
|
|
|
|
var trap = handler[name];
|
|
|
|
if (IS_UNDEFINED(trap)) {
|
|
|
|
if (IS_UNDEFINED(defaultTrap)) {
|
|
|
|
throw MakeTypeError("handler_trap_missing", [handler, name]);
|
|
|
|
}
|
|
|
|
trap = defaultTrap;
|
2011-09-13 11:42:57 +00:00
|
|
|
} else if (!IS_SPEC_FUNCTION(trap)) {
|
2011-07-21 11:20:27 +00:00
|
|
|
throw MakeTypeError("handler_trap_must_be_callable", [handler, name]);
|
|
|
|
}
|
|
|
|
return trap;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function CallTrap0(handler, name, defaultTrap) {
|
|
|
|
return %_CallFunction(handler, GetTrap(handler, name, defaultTrap));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function CallTrap1(handler, name, defaultTrap, x) {
|
|
|
|
return %_CallFunction(handler, x, GetTrap(handler, name, defaultTrap));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function CallTrap2(handler, name, defaultTrap, x, y) {
|
|
|
|
return %_CallFunction(handler, x, y, GetTrap(handler, name, defaultTrap));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-04 12:14:56 +00:00
|
|
|
// ES5 section 8.12.1.
|
2011-07-21 11:20:27 +00:00
|
|
|
function GetOwnProperty(obj, v) {
|
|
|
|
var p = ToString(v);
|
2011-07-13 11:57:15 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
var handler = %GetHandler(obj);
|
2011-07-21 11:20:27 +00:00
|
|
|
var descriptor = CallTrap1(handler, "getOwnPropertyDescriptor", void 0, p);
|
2011-07-13 11:57:15 +00:00
|
|
|
if (IS_UNDEFINED(descriptor)) return descriptor;
|
|
|
|
var desc = ToCompletePropertyDescriptor(descriptor);
|
|
|
|
if (!desc.isConfigurable()) {
|
|
|
|
throw MakeTypeError("proxy_prop_not_configurable",
|
|
|
|
[handler, "getOwnPropertyDescriptor", p, descriptor]);
|
|
|
|
}
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
|
2011-02-04 12:14:56 +00:00
|
|
|
// GetOwnProperty returns an array indexed by the constants
|
|
|
|
// defined in macros.py.
|
|
|
|
// If p is not a property on obj undefined is returned.
|
2011-07-21 11:20:27 +00:00
|
|
|
var props = %GetOwnProperty(ToObject(obj), ToString(v));
|
2011-02-04 12:14:56 +00:00
|
|
|
|
|
|
|
// A false value here means that access checks failed.
|
2011-03-11 13:57:20 +00:00
|
|
|
if (props === false) return void 0;
|
2011-02-04 12:14:56 +00:00
|
|
|
|
|
|
|
return ConvertDescriptorArrayToDescriptor(props);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-05 09:53:43 +00:00
|
|
|
// ES5 section 8.12.7.
|
|
|
|
function Delete(obj, p, should_throw) {
|
|
|
|
var desc = GetOwnProperty(obj, p);
|
|
|
|
if (IS_UNDEFINED(desc)) return true;
|
|
|
|
if (desc.isConfigurable()) {
|
|
|
|
%DeleteProperty(obj, p, 0);
|
|
|
|
return true;
|
|
|
|
} else if (should_throw) {
|
|
|
|
throw MakeTypeError("define_disallowed", [p]);
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-07 12:41:20 +00:00
|
|
|
// Harmony proxies.
|
|
|
|
function DefineProxyProperty(obj, p, attributes, should_throw) {
|
|
|
|
var handler = %GetHandler(obj);
|
2011-07-21 11:20:27 +00:00
|
|
|
var result = CallTrap2(handler, "defineProperty", void 0, p, attributes);
|
2011-07-07 12:41:20 +00:00
|
|
|
if (!ToBoolean(result)) {
|
|
|
|
if (should_throw) {
|
2011-07-18 13:04:52 +00:00
|
|
|
throw MakeTypeError("handler_returned_false",
|
|
|
|
[handler, "defineProperty"]);
|
2011-07-07 12:41:20 +00:00
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-30 07:15:23 +00:00
|
|
|
// ES5 8.12.9.
|
2011-12-05 09:53:43 +00:00
|
|
|
function DefineObjectProperty(obj, p, desc, should_throw) {
|
2011-02-04 12:14:56 +00:00
|
|
|
var current_or_access = %GetOwnProperty(ToObject(obj), ToString(p));
|
|
|
|
// A false value here means that access checks failed.
|
2011-03-11 13:57:20 +00:00
|
|
|
if (current_or_access === false) return void 0;
|
2011-02-04 12:14:56 +00:00
|
|
|
|
|
|
|
var current = ConvertDescriptorArrayToDescriptor(current_or_access);
|
2010-02-03 13:10:03 +00:00
|
|
|
var extensible = %IsExtensible(ToObject(obj));
|
|
|
|
|
|
|
|
// Error handling according to spec.
|
|
|
|
// Step 3
|
2011-03-15 14:19:18 +00:00
|
|
|
if (IS_UNDEFINED(current) && !extensible) {
|
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("define_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
|
|
|
}
|
2010-02-03 13:10:03 +00:00
|
|
|
|
2011-02-03 19:29:10 +00:00
|
|
|
if (!IS_UNDEFINED(current)) {
|
2010-07-23 10:08:55 +00:00
|
|
|
// Step 5 and 6
|
2011-02-03 19:29:10 +00:00
|
|
|
if ((IsGenericDescriptor(desc) ||
|
|
|
|
IsDataDescriptor(desc) == IsDataDescriptor(current)) &&
|
|
|
|
(!desc.hasEnumerable() ||
|
|
|
|
SameValue(desc.isEnumerable(), current.isEnumerable())) &&
|
2010-10-15 13:03:59 +00:00
|
|
|
(!desc.hasConfigurable() ||
|
2010-07-23 10:08:55 +00:00
|
|
|
SameValue(desc.isConfigurable(), current.isConfigurable())) &&
|
2010-10-15 13:03:59 +00:00
|
|
|
(!desc.hasWritable() ||
|
2010-07-23 10:08:55 +00:00
|
|
|
SameValue(desc.isWritable(), current.isWritable())) &&
|
|
|
|
(!desc.hasValue() ||
|
|
|
|
SameValue(desc.getValue(), current.getValue())) &&
|
|
|
|
(!desc.hasGetter() ||
|
|
|
|
SameValue(desc.getGet(), current.getGet())) &&
|
|
|
|
(!desc.hasSetter() ||
|
|
|
|
SameValue(desc.getSet(), current.getSet()))) {
|
|
|
|
return true;
|
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
if (!current.isConfigurable()) {
|
|
|
|
// Step 7
|
|
|
|
if (desc.isConfigurable() ||
|
|
|
|
(desc.hasEnumerable() &&
|
2011-02-14 10:43:21 +00:00
|
|
|
desc.isEnumerable() != current.isEnumerable())) {
|
2011-03-15 14:19:18 +00:00
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
2011-02-14 10:43:21 +00:00
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
// Step 8
|
|
|
|
if (!IsGenericDescriptor(desc)) {
|
|
|
|
// Step 9a
|
2011-02-14 10:43:21 +00:00
|
|
|
if (IsDataDescriptor(current) != IsDataDescriptor(desc)) {
|
2011-03-15 14:19:18 +00:00
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
2011-02-14 10:43:21 +00:00
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
// Step 10a
|
|
|
|
if (IsDataDescriptor(current) && IsDataDescriptor(desc)) {
|
2011-02-14 10:43:21 +00:00
|
|
|
if (!current.isWritable() && desc.isWritable()) {
|
2011-03-15 14:19:18 +00:00
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
2011-02-14 10:43:21 +00:00
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
if (!current.isWritable() && desc.hasValue() &&
|
|
|
|
!SameValue(desc.getValue(), current.getValue())) {
|
2011-03-15 14:19:18 +00:00
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Step 11
|
|
|
|
if (IsAccessorDescriptor(desc) && IsAccessorDescriptor(current)) {
|
2011-02-14 10:43:21 +00:00
|
|
|
if (desc.hasSetter() && !SameValue(desc.getSet(), current.getSet())) {
|
2011-03-15 14:19:18 +00:00
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
}
|
2011-02-14 10:43:21 +00:00
|
|
|
if (desc.hasGetter() && !SameValue(desc.getGet(),current.getGet())) {
|
2011-03-15 14:19:18 +00:00
|
|
|
if (should_throw) {
|
2011-06-16 06:37:49 +00:00
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
2011-03-15 14:19:18 +00:00
|
|
|
} else {
|
2011-10-19 13:29:37 +00:00
|
|
|
return false;
|
2011-03-15 14:19:18 +00:00
|
|
|
}
|
2011-02-14 10:43:21 +00:00
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
}
|
2010-02-03 13:10:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-30 07:15:23 +00:00
|
|
|
// Send flags - enumerable and configurable are common - writable is
|
2010-02-03 13:10:03 +00:00
|
|
|
// only send to the data descriptor.
|
|
|
|
// Take special care if enumerable and configurable is not defined on
|
|
|
|
// desc (we need to preserve the existing values from current).
|
|
|
|
var flag = NONE;
|
|
|
|
if (desc.hasEnumerable()) {
|
|
|
|
flag |= desc.isEnumerable() ? 0 : DONT_ENUM;
|
|
|
|
} else if (!IS_UNDEFINED(current)) {
|
|
|
|
flag |= current.isEnumerable() ? 0 : DONT_ENUM;
|
2009-12-09 13:56:58 +00:00
|
|
|
} else {
|
2010-02-03 13:10:03 +00:00
|
|
|
flag |= DONT_ENUM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (desc.hasConfigurable()) {
|
|
|
|
flag |= desc.isConfigurable() ? 0 : DONT_DELETE;
|
|
|
|
} else if (!IS_UNDEFINED(current)) {
|
|
|
|
flag |= current.isConfigurable() ? 0 : DONT_DELETE;
|
|
|
|
} else
|
|
|
|
flag |= DONT_DELETE;
|
|
|
|
|
2011-02-03 19:29:10 +00:00
|
|
|
if (IsDataDescriptor(desc) ||
|
|
|
|
(IsGenericDescriptor(desc) &&
|
|
|
|
(IS_UNDEFINED(current) || IsDataDescriptor(current)))) {
|
|
|
|
// There are 3 cases that lead here:
|
|
|
|
// Step 4a - defining a new data property.
|
|
|
|
// Steps 9b & 12 - replacing an existing accessor property with a data
|
|
|
|
// property.
|
|
|
|
// Step 12 - updating an existing data property with a data or generic
|
|
|
|
// descriptor.
|
|
|
|
|
2010-05-26 08:31:57 +00:00
|
|
|
if (desc.hasWritable()) {
|
|
|
|
flag |= desc.isWritable() ? 0 : READ_ONLY;
|
|
|
|
} else if (!IS_UNDEFINED(current)) {
|
|
|
|
flag |= current.isWritable() ? 0 : READ_ONLY;
|
|
|
|
} else {
|
|
|
|
flag |= READ_ONLY;
|
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
|
2010-12-10 13:07:52 +00:00
|
|
|
var value = void 0; // Default value is undefined.
|
2010-12-10 11:27:15 +00:00
|
|
|
if (desc.hasValue()) {
|
|
|
|
value = desc.getValue();
|
2011-02-03 19:29:10 +00:00
|
|
|
} else if (!IS_UNDEFINED(current) && IsDataDescriptor(current)) {
|
2010-12-10 11:27:15 +00:00
|
|
|
value = current.getValue();
|
|
|
|
}
|
2011-02-03 19:29:10 +00:00
|
|
|
|
2010-12-10 11:27:15 +00:00
|
|
|
%DefineOrRedefineDataProperty(obj, p, value, flag);
|
2010-02-03 13:10:03 +00:00
|
|
|
} else {
|
2011-02-03 19:29:10 +00:00
|
|
|
// There are 3 cases that lead here:
|
|
|
|
// Step 4b - defining a new accessor property.
|
|
|
|
// Steps 9c & 12 - replacing an existing data property with an accessor
|
|
|
|
// property.
|
|
|
|
// Step 12 - updating an existing accessor property with an accessor
|
|
|
|
// descriptor.
|
Make the runtime entry for setting/changing accessors "atomic".
Previously, there were 1 or 2 calls to the runtime when accessors were changed
or set. This doesn't really work well with property attributes, leading to some
hacks and complicates things even further when trying to share maps in presence
of accessors. Therefore, the runtime entry now takes the full triple (getter,
setter, attributes), where the getter and/or the setter can be null in case they
shouldn't be changed.
For now, we do basically the same on the native side as we did before on the
JavaScript side, but this will change in future CLs, the current CL is already
large enough.
Note that object literals with a getter and a setter for the same property still
do 2 calls, but this is a little bit more tricky to fix and will be handled in a
separate CL.
Review URL: https://chromiumcodereview.appspot.com/9616016
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@10956 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-03-07 13:24:44 +00:00
|
|
|
var getter = desc.hasGetter() ? desc.getGet() : null;
|
|
|
|
var setter = desc.hasSetter() ? desc.getSet() : null;
|
|
|
|
%DefineOrRedefineAccessorProperty(obj, p, getter, setter, flag);
|
2009-12-09 13:56:58 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-05 09:53:43 +00:00
|
|
|
// ES5 section 15.4.5.1.
|
|
|
|
function DefineArrayProperty(obj, p, desc, should_throw) {
|
|
|
|
// Note that the length of an array is not actually stored as part of the
|
|
|
|
// property, hence we use generated code throughout this function instead of
|
|
|
|
// DefineObjectProperty() to modify its value.
|
|
|
|
|
|
|
|
// Step 3 - Special handling for length property.
|
|
|
|
if (p == "length") {
|
|
|
|
var length = obj.length;
|
|
|
|
if (!desc.hasValue()) {
|
|
|
|
return DefineObjectProperty(obj, "length", desc, should_throw);
|
|
|
|
}
|
|
|
|
var new_length = ToUint32(desc.getValue());
|
|
|
|
if (new_length != ToNumber(desc.getValue())) {
|
|
|
|
throw new $RangeError('defineProperty() array length out of range');
|
|
|
|
}
|
|
|
|
var length_desc = GetOwnProperty(obj, "length");
|
2011-12-12 10:20:46 +00:00
|
|
|
if (new_length != length && !length_desc.isWritable()) {
|
|
|
|
if (should_throw) {
|
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var threw = false;
|
|
|
|
while (new_length < length--) {
|
|
|
|
if (!Delete(obj, ToString(length), false)) {
|
|
|
|
new_length = length + 1;
|
|
|
|
threw = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-12-05 09:53:43 +00:00
|
|
|
// Make sure the below call to DefineObjectProperty() doesn't overwrite
|
|
|
|
// any magic "length" property by removing the value.
|
2012-12-13 09:31:44 +00:00
|
|
|
// TODO(mstarzinger): This hack should be removed once we have addressed the
|
|
|
|
// respective TODO in Runtime_DefineOrRedefineDataProperty.
|
|
|
|
// For the time being, we need a hack to prevent Object.observe from
|
|
|
|
// generating two change records.
|
|
|
|
var isObserved = %IsObserved(obj);
|
|
|
|
if (isObserved) %SetIsObserved(obj, false);
|
2011-12-12 10:20:46 +00:00
|
|
|
obj.length = new_length;
|
2011-12-05 09:53:43 +00:00
|
|
|
desc.value_ = void 0;
|
|
|
|
desc.hasValue_ = false;
|
2012-12-13 09:31:44 +00:00
|
|
|
threw = !DefineObjectProperty(obj, "length", desc, should_throw) || threw;
|
|
|
|
if (isObserved) %SetIsObserved(obj, true);
|
|
|
|
if (threw) {
|
2011-12-05 09:53:43 +00:00
|
|
|
if (should_throw) {
|
|
|
|
throw MakeTypeError("redefine_disallowed", [p]);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2012-12-13 09:31:44 +00:00
|
|
|
if (isObserved) {
|
|
|
|
var new_desc = GetOwnProperty(obj, "length");
|
|
|
|
var updated = length_desc.value_ !== new_desc.value_;
|
|
|
|
var reconfigured = length_desc.writable_ !== new_desc.writable_ ||
|
|
|
|
length_desc.configurable_ !== new_desc.configurable_ ||
|
|
|
|
length_desc.enumerable_ !== new_desc.configurable_;
|
|
|
|
if (updated || reconfigured) {
|
|
|
|
NotifyChange(reconfigured ? "reconfigured" : "updated",
|
|
|
|
obj, "length", length_desc.value_);
|
|
|
|
}
|
|
|
|
}
|
2011-12-05 09:53:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Step 4 - Special handling for array index.
|
|
|
|
var index = ToUint32(p);
|
|
|
|
if (index == ToNumber(p) && index != 4294967295) {
|
|
|
|
var length = obj.length;
|
|
|
|
var length_desc = GetOwnProperty(obj, "length");
|
|
|
|
if ((index >= length && !length_desc.isWritable()) ||
|
|
|
|
!DefineObjectProperty(obj, p, desc, true)) {
|
|
|
|
if (should_throw) {
|
|
|
|
throw MakeTypeError("define_disallowed", [p]);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (index >= length) {
|
|
|
|
obj.length = index + 1;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Step 5 - Fallback to default implementation.
|
|
|
|
return DefineObjectProperty(obj, p, desc, should_throw);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ES5 section 8.12.9, ES5 section 15.4.5.1 and Harmony proxies.
|
|
|
|
function DefineOwnProperty(obj, p, desc, should_throw) {
|
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
var attributes = FromGenericPropertyDescriptor(desc);
|
|
|
|
return DefineProxyProperty(obj, p, attributes, should_throw);
|
|
|
|
} else if (IS_ARRAY(obj)) {
|
|
|
|
return DefineArrayProperty(obj, p, desc, should_throw);
|
|
|
|
} else {
|
|
|
|
return DefineObjectProperty(obj, p, desc, should_throw);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-07 10:01:24 +00:00
|
|
|
// ES5 section 15.2.3.2.
|
|
|
|
function ObjectGetPrototypeOf(obj) {
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.getPrototypeOf"]);
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2011-06-03 10:15:49 +00:00
|
|
|
return %GetPrototype(obj);
|
2010-01-07 10:01:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-30 07:15:23 +00:00
|
|
|
// ES5 section 15.2.3.3
|
2010-01-13 12:10:57 +00:00
|
|
|
function ObjectGetOwnPropertyDescriptor(obj, p) {
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object",
|
|
|
|
["Object.getOwnPropertyDescriptor"]);
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2010-01-13 12:10:57 +00:00
|
|
|
var desc = GetOwnProperty(obj, p);
|
|
|
|
return FromPropertyDescriptor(desc);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-03 10:15:49 +00:00
|
|
|
// For Harmony proxies
|
|
|
|
function ToStringArray(obj, trap) {
|
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
|
|
|
throw MakeTypeError("proxy_non_object_prop_names", [obj, trap]);
|
|
|
|
}
|
|
|
|
var n = ToUint32(obj.length);
|
|
|
|
var array = new $Array(n);
|
2012-11-16 09:32:39 +00:00
|
|
|
var names = { __proto__: null }; // TODO(rossberg): use sets once ready.
|
2011-06-03 10:15:49 +00:00
|
|
|
for (var index = 0; index < n; index++) {
|
|
|
|
var s = ToString(obj[index]);
|
2012-05-29 13:57:55 +00:00
|
|
|
if (%HasLocalProperty(names, s)) {
|
2011-11-28 12:11:00 +00:00
|
|
|
throw MakeTypeError("proxy_repeated_prop_name", [obj, trap, s]);
|
2011-06-03 10:15:49 +00:00
|
|
|
}
|
|
|
|
array[index] = s;
|
2011-10-24 15:56:18 +00:00
|
|
|
names[s] = 0;
|
2011-06-03 10:15:49 +00:00
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 15:34:32 +00:00
|
|
|
// ES5 section 15.2.3.4.
|
|
|
|
function ObjectGetOwnPropertyNames(obj) {
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.getOwnPropertyNames"]);
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2011-06-03 10:15:49 +00:00
|
|
|
// Special handling for proxies.
|
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
var handler = %GetHandler(obj);
|
2011-07-21 11:20:27 +00:00
|
|
|
var names = CallTrap0(handler, "getOwnPropertyNames", void 0);
|
2011-06-03 10:15:49 +00:00
|
|
|
return ToStringArray(names, "getOwnPropertyNames");
|
|
|
|
}
|
|
|
|
|
2010-01-15 15:34:32 +00:00
|
|
|
// Find all the indexed properties.
|
|
|
|
|
|
|
|
// Get the local element names.
|
|
|
|
var propertyNames = %GetLocalElementNames(obj);
|
2013-02-25 18:58:47 +00:00
|
|
|
for (var i = 0; i < propertyNames.length; ++i) {
|
|
|
|
propertyNames[i] = %_NumberToString(propertyNames[i]);
|
|
|
|
}
|
2010-01-15 15:34:32 +00:00
|
|
|
|
|
|
|
// Get names for indexed interceptor properties.
|
2013-02-25 18:58:47 +00:00
|
|
|
var interceptorInfo = %GetInterceptorInfo(obj);
|
|
|
|
if ((interceptorInfo & 1) != 0) {
|
2010-01-15 15:34:32 +00:00
|
|
|
var indexedInterceptorNames =
|
|
|
|
%GetIndexedInterceptorElementNames(obj);
|
2011-11-28 12:11:00 +00:00
|
|
|
if (indexedInterceptorNames) {
|
2010-01-15 15:34:32 +00:00
|
|
|
propertyNames = propertyNames.concat(indexedInterceptorNames);
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2010-01-15 15:34:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find all the named properties.
|
|
|
|
|
|
|
|
// Get the local property names.
|
|
|
|
propertyNames = propertyNames.concat(%GetLocalPropertyNames(obj));
|
|
|
|
|
|
|
|
// Get names for named interceptor properties if any.
|
2013-02-25 18:58:47 +00:00
|
|
|
if ((interceptorInfo & 2) != 0) {
|
2010-01-15 15:34:32 +00:00
|
|
|
var namedInterceptorNames =
|
|
|
|
%GetNamedInterceptorPropertyNames(obj);
|
|
|
|
if (namedInterceptorNames) {
|
|
|
|
propertyNames = propertyNames.concat(namedInterceptorNames);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-25 18:58:47 +00:00
|
|
|
// Property names are expected to be unique strings,
|
|
|
|
// but interceptors can interfere with that assumption.
|
|
|
|
if (interceptorInfo != 0) {
|
|
|
|
var propertySet = { __proto__: null };
|
|
|
|
var j = 0;
|
|
|
|
for (var i = 0; i < propertyNames.length; ++i) {
|
|
|
|
var name = ToString(propertyNames[i]);
|
|
|
|
// We need to check for the exact property value since for intrinsic
|
|
|
|
// properties like toString if(propertySet["toString"]) will always
|
|
|
|
// succeed.
|
|
|
|
if (propertySet[name] === true) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
propertySet[name] = true;
|
|
|
|
propertyNames[j++] = name;
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2013-02-25 18:58:47 +00:00
|
|
|
propertyNames.length = j;
|
2010-06-28 08:31:31 +00:00
|
|
|
}
|
2010-02-16 10:08:39 +00:00
|
|
|
|
2010-01-15 15:34:32 +00:00
|
|
|
return propertyNames;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-09 13:56:58 +00:00
|
|
|
// ES5 section 15.2.3.5.
|
|
|
|
function ObjectCreate(proto, properties) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(proto) && proto !== null) {
|
2009-12-09 13:56:58 +00:00
|
|
|
throw MakeTypeError("proto_object_or_null", [proto]);
|
|
|
|
}
|
|
|
|
var obj = new $Object();
|
|
|
|
obj.__proto__ = proto;
|
|
|
|
if (!IS_UNDEFINED(properties)) ObjectDefineProperties(obj, properties);
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-03 13:10:03 +00:00
|
|
|
// ES5 section 15.2.3.6.
|
|
|
|
function ObjectDefineProperty(obj, p, attributes) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.defineProperty"]);
|
2010-05-25 06:25:27 +00:00
|
|
|
}
|
2010-02-03 13:10:03 +00:00
|
|
|
var name = ToString(p);
|
2011-07-07 12:41:20 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
// Clone the attributes object for protection.
|
|
|
|
// TODO(rossberg): not spec'ed yet, so not sure if this should involve
|
|
|
|
// non-own properties as it does (or non-enumerable ones, as it doesn't?).
|
2012-11-16 09:32:39 +00:00
|
|
|
var attributesClone = { __proto__: null };
|
2011-07-07 12:41:20 +00:00
|
|
|
for (var a in attributes) {
|
|
|
|
attributesClone[a] = attributes[a];
|
|
|
|
}
|
|
|
|
DefineProxyProperty(obj, name, attributesClone, true);
|
|
|
|
// The following would implement the spec as in the current proposal,
|
|
|
|
// but after recent comments on es-discuss, is most likely obsolete.
|
|
|
|
/*
|
|
|
|
var defineObj = FromGenericPropertyDescriptor(desc);
|
|
|
|
var names = ObjectGetOwnPropertyNames(attributes);
|
|
|
|
var standardNames =
|
|
|
|
{value: 0, writable: 0, get: 0, set: 0, enumerable: 0, configurable: 0};
|
|
|
|
for (var i = 0; i < names.length; i++) {
|
|
|
|
var N = names[i];
|
|
|
|
if (!(%HasLocalProperty(standardNames, N))) {
|
|
|
|
var attr = GetOwnProperty(attributes, N);
|
|
|
|
DefineOwnProperty(descObj, N, attr, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This is really confusing the types, but it is what the proxies spec
|
|
|
|
// currently requires:
|
|
|
|
desc = descObj;
|
|
|
|
*/
|
|
|
|
} else {
|
|
|
|
var desc = ToPropertyDescriptor(attributes);
|
|
|
|
DefineOwnProperty(obj, name, desc, true);
|
|
|
|
}
|
2010-02-03 13:10:03 +00:00
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-18 08:39:06 +00:00
|
|
|
function GetOwnEnumerablePropertyNames(properties) {
|
2013-02-15 16:21:03 +00:00
|
|
|
var names = new InternalArray();
|
2011-08-18 08:39:06 +00:00
|
|
|
for (var key in properties) {
|
|
|
|
if (%HasLocalProperty(properties, key)) {
|
|
|
|
names.push(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return names;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-03 13:10:03 +00:00
|
|
|
// ES5 section 15.2.3.7.
|
2009-12-09 13:56:58 +00:00
|
|
|
function ObjectDefineProperties(obj, properties) {
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.defineProperties"]);
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2009-12-09 13:56:58 +00:00
|
|
|
var props = ToObject(properties);
|
2011-08-18 08:39:06 +00:00
|
|
|
var names = GetOwnEnumerablePropertyNames(props);
|
2013-02-15 16:21:03 +00:00
|
|
|
var descriptors = new InternalArray();
|
2011-08-18 08:39:06 +00:00
|
|
|
for (var i = 0; i < names.length; i++) {
|
2011-10-19 09:52:08 +00:00
|
|
|
descriptors.push(ToPropertyDescriptor(props[names[i]]));
|
|
|
|
}
|
|
|
|
for (var i = 0; i < names.length; i++) {
|
|
|
|
DefineOwnProperty(obj, names[i], descriptors[i], true);
|
2009-12-09 13:56:58 +00:00
|
|
|
}
|
2010-02-04 11:28:03 +00:00
|
|
|
return obj;
|
2009-12-09 13:56:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-18 13:04:52 +00:00
|
|
|
// Harmony proxies.
|
|
|
|
function ProxyFix(obj) {
|
|
|
|
var handler = %GetHandler(obj);
|
2011-07-21 11:20:27 +00:00
|
|
|
var props = CallTrap0(handler, "fix", void 0);
|
2011-07-18 13:04:52 +00:00
|
|
|
if (IS_UNDEFINED(props)) {
|
|
|
|
throw MakeTypeError("handler_returned_undefined", [handler, "fix"]);
|
|
|
|
}
|
2011-09-13 11:42:57 +00:00
|
|
|
|
2011-09-22 10:45:37 +00:00
|
|
|
if (%IsJSFunctionProxy(obj)) {
|
2011-09-13 11:42:57 +00:00
|
|
|
var callTrap = %GetCallTrap(obj);
|
|
|
|
var constructTrap = %GetConstructTrap(obj);
|
|
|
|
var code = DelegateCallAndConstruct(callTrap, constructTrap);
|
|
|
|
%Fix(obj); // becomes a regular function
|
|
|
|
%SetCode(obj, code);
|
2011-09-16 12:26:29 +00:00
|
|
|
// TODO(rossberg): What about length and other properties? Not specified.
|
|
|
|
// We just put in some half-reasonable defaults for now.
|
|
|
|
var prototype = new $Object();
|
|
|
|
$Object.defineProperty(prototype, "constructor",
|
2011-11-10 15:48:07 +00:00
|
|
|
{value: obj, writable: true, enumerable: false, configurable: true});
|
|
|
|
// TODO(v8:1530): defineProperty does not handle prototype and length.
|
|
|
|
%FunctionSetPrototype(obj, prototype);
|
|
|
|
obj.length = 0;
|
2011-09-13 11:42:57 +00:00
|
|
|
} else {
|
|
|
|
%Fix(obj);
|
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
ObjectDefineProperties(obj, props);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-15 07:51:14 +00:00
|
|
|
// ES5 section 15.2.3.8.
|
|
|
|
function ObjectSeal(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.seal"]);
|
2010-07-15 07:51:14 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
ProxyFix(obj);
|
|
|
|
}
|
2010-07-15 07:51:14 +00:00
|
|
|
var names = ObjectGetOwnPropertyNames(obj);
|
2010-08-26 08:35:49 +00:00
|
|
|
for (var i = 0; i < names.length; i++) {
|
|
|
|
var name = names[i];
|
2010-07-15 07:51:14 +00:00
|
|
|
var desc = GetOwnProperty(obj, name);
|
2011-06-10 09:45:02 +00:00
|
|
|
if (desc.isConfigurable()) {
|
|
|
|
desc.setConfigurable(false);
|
|
|
|
DefineOwnProperty(obj, name, desc, true);
|
|
|
|
}
|
2010-10-15 13:03:59 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
%PreventExtensions(obj);
|
|
|
|
return obj;
|
2010-07-15 07:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-13 12:58:02 +00:00
|
|
|
// ES5 section 15.2.3.9.
|
|
|
|
function ObjectFreeze(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.freeze"]);
|
2010-07-13 12:58:02 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
ProxyFix(obj);
|
|
|
|
}
|
2010-07-13 12:58:02 +00:00
|
|
|
var names = ObjectGetOwnPropertyNames(obj);
|
2010-08-26 08:35:49 +00:00
|
|
|
for (var i = 0; i < names.length; i++) {
|
|
|
|
var name = names[i];
|
2010-07-13 12:58:02 +00:00
|
|
|
var desc = GetOwnProperty(obj, name);
|
2011-06-10 09:45:02 +00:00
|
|
|
if (desc.isWritable() || desc.isConfigurable()) {
|
|
|
|
if (IsDataDescriptor(desc)) desc.setWritable(false);
|
|
|
|
desc.setConfigurable(false);
|
|
|
|
DefineOwnProperty(obj, name, desc, true);
|
|
|
|
}
|
2010-10-15 13:03:59 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
%PreventExtensions(obj);
|
|
|
|
return obj;
|
2010-07-13 12:58:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-02 14:36:34 +00:00
|
|
|
// ES5 section 15.2.3.10
|
|
|
|
function ObjectPreventExtension(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.preventExtension"]);
|
2010-07-02 14:36:34 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
ProxyFix(obj);
|
|
|
|
}
|
2010-07-02 14:36:34 +00:00
|
|
|
%PreventExtensions(obj);
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-15 07:51:14 +00:00
|
|
|
// ES5 section 15.2.3.11
|
|
|
|
function ObjectIsSealed(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.isSealed"]);
|
2010-07-15 07:51:14 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
return false;
|
|
|
|
}
|
2013-02-25 19:44:21 +00:00
|
|
|
if (%IsExtensible(obj)) {
|
|
|
|
return false;
|
|
|
|
}
|
2010-07-15 07:51:14 +00:00
|
|
|
var names = ObjectGetOwnPropertyNames(obj);
|
2010-08-26 08:35:49 +00:00
|
|
|
for (var i = 0; i < names.length; i++) {
|
|
|
|
var name = names[i];
|
2010-07-15 07:51:14 +00:00
|
|
|
var desc = GetOwnProperty(obj, name);
|
2010-08-26 13:03:17 +00:00
|
|
|
if (desc.isConfigurable()) return false;
|
2010-07-15 07:51:14 +00:00
|
|
|
}
|
2013-02-25 19:44:21 +00:00
|
|
|
return true;
|
2010-07-15 07:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-13 12:58:02 +00:00
|
|
|
// ES5 section 15.2.3.12
|
|
|
|
function ObjectIsFrozen(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.isFrozen"]);
|
2010-07-13 12:58:02 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
return false;
|
|
|
|
}
|
2013-02-25 19:44:21 +00:00
|
|
|
if (%IsExtensible(obj)) {
|
|
|
|
return false;
|
|
|
|
}
|
2010-07-13 12:58:02 +00:00
|
|
|
var names = ObjectGetOwnPropertyNames(obj);
|
2010-08-26 08:35:49 +00:00
|
|
|
for (var i = 0; i < names.length; i++) {
|
|
|
|
var name = names[i];
|
2010-07-13 12:58:02 +00:00
|
|
|
var desc = GetOwnProperty(obj, name);
|
2010-07-14 13:15:43 +00:00
|
|
|
if (IsDataDescriptor(desc) && desc.isWritable()) return false;
|
|
|
|
if (desc.isConfigurable()) return false;
|
2010-07-13 12:58:02 +00:00
|
|
|
}
|
2013-02-25 19:44:21 +00:00
|
|
|
return true;
|
2010-07-13 12:58:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-02 14:36:34 +00:00
|
|
|
// ES5 section 15.2.3.13
|
|
|
|
function ObjectIsExtensible(obj) {
|
2010-07-16 11:21:08 +00:00
|
|
|
if (!IS_SPEC_OBJECT(obj)) {
|
2012-03-05 13:57:48 +00:00
|
|
|
throw MakeTypeError("called_on_non_object", ["Object.isExtensible"]);
|
2010-07-02 14:36:34 +00:00
|
|
|
}
|
2011-07-18 13:04:52 +00:00
|
|
|
if (%IsJSProxy(obj)) {
|
|
|
|
return true;
|
|
|
|
}
|
2010-07-02 14:36:34 +00:00
|
|
|
return %IsExtensible(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-08 12:49:24 +00:00
|
|
|
// Harmony egal.
|
|
|
|
function ObjectIs(obj1, obj2) {
|
|
|
|
if (obj1 === obj2) {
|
2012-03-08 16:38:44 +00:00
|
|
|
return (obj1 !== 0) || (1 / obj1 === 1 / obj2);
|
2012-03-08 12:49:24 +00:00
|
|
|
} else {
|
|
|
|
return (obj1 !== obj1) && (obj2 !== obj2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
%SetCode($Object, function(x) {
|
2009-06-29 08:14:06 +00:00
|
|
|
if (%_IsConstructCall()) {
|
2008-07-03 15:10:15 +00:00
|
|
|
if (x == null) return this;
|
|
|
|
return ToObject(x);
|
|
|
|
} else {
|
|
|
|
if (x == null) return { };
|
|
|
|
return ToObject(x);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2010-08-25 17:50:37 +00:00
|
|
|
%SetExpectedNumberOfProperties($Object, 4);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2011-09-05 07:30:35 +00:00
|
|
|
// Object
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
function SetUpObject() {
|
|
|
|
%CheckIsBootstrapping();
|
|
|
|
// Set Up non-enumerable functions on the Object.prototype object.
|
2008-10-03 18:00:28 +00:00
|
|
|
InstallFunctions($Object.prototype, DONT_ENUM, $Array(
|
|
|
|
"toString", ObjectToString,
|
|
|
|
"toLocaleString", ObjectToLocaleString,
|
|
|
|
"valueOf", ObjectValueOf,
|
|
|
|
"hasOwnProperty", ObjectHasOwnProperty,
|
|
|
|
"isPrototypeOf", ObjectIsPrototypeOf,
|
|
|
|
"propertyIsEnumerable", ObjectPropertyIsEnumerable,
|
|
|
|
"__defineGetter__", ObjectDefineGetter,
|
|
|
|
"__lookupGetter__", ObjectLookupGetter,
|
|
|
|
"__defineSetter__", ObjectDefineSetter,
|
|
|
|
"__lookupSetter__", ObjectLookupSetter
|
|
|
|
));
|
2009-09-15 11:51:40 +00:00
|
|
|
InstallFunctions($Object, DONT_ENUM, $Array(
|
2009-12-09 13:56:58 +00:00
|
|
|
"keys", ObjectKeys,
|
2010-01-07 10:01:24 +00:00
|
|
|
"create", ObjectCreate,
|
2010-02-03 13:10:03 +00:00
|
|
|
"defineProperty", ObjectDefineProperty,
|
|
|
|
"defineProperties", ObjectDefineProperties,
|
2010-07-13 12:58:02 +00:00
|
|
|
"freeze", ObjectFreeze,
|
2010-01-13 12:10:57 +00:00
|
|
|
"getPrototypeOf", ObjectGetPrototypeOf,
|
2010-01-15 15:34:32 +00:00
|
|
|
"getOwnPropertyDescriptor", ObjectGetOwnPropertyDescriptor,
|
2010-07-02 14:36:34 +00:00
|
|
|
"getOwnPropertyNames", ObjectGetOwnPropertyNames,
|
2012-03-08 12:49:24 +00:00
|
|
|
"is", ObjectIs,
|
2010-07-02 14:36:34 +00:00
|
|
|
"isExtensible", ObjectIsExtensible,
|
2010-07-13 12:58:02 +00:00
|
|
|
"isFrozen", ObjectIsFrozen,
|
2010-07-15 07:51:14 +00:00
|
|
|
"isSealed", ObjectIsSealed,
|
|
|
|
"preventExtensions", ObjectPreventExtension,
|
|
|
|
"seal", ObjectSeal
|
2009-09-15 11:51:40 +00:00
|
|
|
));
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
SetUpObject();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Boolean
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function BooleanToString() {
|
2008-07-03 15:10:15 +00:00
|
|
|
// NOTE: Both Boolean objects and values can enter here as
|
|
|
|
// 'this'. This is not as dictated by ECMA-262.
|
2011-01-05 13:52:00 +00:00
|
|
|
var b = this;
|
|
|
|
if (!IS_BOOLEAN(b)) {
|
|
|
|
if (!IS_BOOLEAN_WRAPPER(b)) {
|
|
|
|
throw new $TypeError('Boolean.prototype.toString is not generic');
|
|
|
|
}
|
|
|
|
b = %_ValueOf(b);
|
|
|
|
}
|
|
|
|
return b ? 'true' : 'false';
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function BooleanValueOf() {
|
2008-07-03 15:10:15 +00:00
|
|
|
// NOTE: Both Boolean objects and values can enter here as
|
|
|
|
// 'this'. This is not as dictated by ECMA-262.
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_BOOLEAN(this) && !IS_BOOLEAN_WRAPPER(this)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
throw new $TypeError('Boolean.prototype.valueOf is not generic');
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
return %_ValueOf(this);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
function SetUpBoolean () {
|
|
|
|
%CheckIsBootstrapping();
|
2008-10-03 18:00:28 +00:00
|
|
|
InstallFunctions($Boolean.prototype, DONT_ENUM, $Array(
|
|
|
|
"toString", BooleanToString,
|
2010-12-15 09:31:05 +00:00
|
|
|
"valueOf", BooleanValueOf
|
2008-10-03 18:00:28 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
SetUpBoolean();
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Number
|
|
|
|
|
|
|
|
// Set the Number function and constructor.
|
|
|
|
%SetCode($Number, function(x) {
|
|
|
|
var value = %_ArgumentsLength() == 0 ? 0 : ToNumber(x);
|
2009-06-29 08:14:06 +00:00
|
|
|
if (%_IsConstructCall()) {
|
2008-07-03 15:10:15 +00:00
|
|
|
%_SetValueOf(this, value);
|
|
|
|
} else {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
%FunctionSetPrototype($Number, new $Number(0));
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.4.2.
|
2008-10-03 18:00:28 +00:00
|
|
|
function NumberToString(radix) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// NOTE: Both Number objects and values can enter here as
|
|
|
|
// 'this'. This is not as dictated by ECMA-262.
|
|
|
|
var number = this;
|
|
|
|
if (!IS_NUMBER(this)) {
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_NUMBER_WRAPPER(this)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
throw new $TypeError('Number.prototype.toString is not generic');
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
// Get the value of this number in case it's an object.
|
|
|
|
number = %_ValueOf(this);
|
|
|
|
}
|
|
|
|
// Fast case: Convert number in radix 10.
|
|
|
|
if (IS_UNDEFINED(radix) || radix === 10) {
|
2011-01-05 13:52:00 +00:00
|
|
|
return %_NumberToString(number);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Convert the radix to an integer and check the range.
|
|
|
|
radix = TO_INTEGER(radix);
|
|
|
|
if (radix < 2 || radix > 36) {
|
|
|
|
throw new $RangeError('toString() radix argument must be between 2 and 36');
|
|
|
|
}
|
|
|
|
// Convert the number to a string in the given radix.
|
|
|
|
return %NumberToRadixString(number, radix);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.4.3
|
2008-10-03 18:00:28 +00:00
|
|
|
function NumberToLocaleString() {
|
2012-12-07 12:18:50 +00:00
|
|
|
return %_CallFunction(this, NumberToString);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.4.4
|
2008-10-03 18:00:28 +00:00
|
|
|
function NumberValueOf() {
|
2008-07-03 15:10:15 +00:00
|
|
|
// NOTE: Both Number objects and values can enter here as
|
|
|
|
// 'this'. This is not as dictated by ECMA-262.
|
2011-11-28 12:11:00 +00:00
|
|
|
if (!IS_NUMBER(this) && !IS_NUMBER_WRAPPER(this)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
throw new $TypeError('Number.prototype.valueOf is not generic');
|
2011-11-28 12:11:00 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
return %_ValueOf(this);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.4.5
|
2008-10-03 18:00:28 +00:00
|
|
|
function NumberToFixed(fractionDigits) {
|
2012-12-07 10:20:35 +00:00
|
|
|
var x = this;
|
|
|
|
if (!IS_NUMBER(this)) {
|
|
|
|
if (!IS_NUMBER_WRAPPER(this)) {
|
|
|
|
throw MakeTypeError("incompatible_method_receiver",
|
|
|
|
["Number.prototype.toFixed", this]);
|
|
|
|
}
|
|
|
|
// Get the value of this number in case it's an object.
|
|
|
|
x = %_ValueOf(this);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
var f = TO_INTEGER(fractionDigits);
|
2012-12-07 10:20:35 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
if (f < 0 || f > 20) {
|
|
|
|
throw new $RangeError("toFixed() digits argument must be between 0 and 20");
|
|
|
|
}
|
2012-12-07 10:20:35 +00:00
|
|
|
|
|
|
|
if (NUMBER_IS_NAN(x)) return "NaN";
|
|
|
|
if (x == 1/0) return "Infinity";
|
|
|
|
if (x == -1/0) return "-Infinity";
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
return %NumberToFixed(x, f);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.4.6
|
2008-10-03 18:00:28 +00:00
|
|
|
function NumberToExponential(fractionDigits) {
|
2012-12-07 10:20:35 +00:00
|
|
|
var x = this;
|
|
|
|
if (!IS_NUMBER(this)) {
|
|
|
|
if (!IS_NUMBER_WRAPPER(this)) {
|
|
|
|
throw MakeTypeError("incompatible_method_receiver",
|
|
|
|
["Number.prototype.toExponential", this]);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2012-12-07 10:20:35 +00:00
|
|
|
// Get the value of this number in case it's an object.
|
|
|
|
x = %_ValueOf(this);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2012-12-07 10:20:35 +00:00
|
|
|
var f = IS_UNDEFINED(fractionDigits) ? void 0 : TO_INTEGER(fractionDigits);
|
|
|
|
|
|
|
|
if (NUMBER_IS_NAN(x)) return "NaN";
|
|
|
|
if (x == 1/0) return "Infinity";
|
|
|
|
if (x == -1/0) return "-Infinity";
|
|
|
|
|
|
|
|
if (IS_UNDEFINED(f)) {
|
|
|
|
f = -1; // Signal for runtime function that f is not defined.
|
|
|
|
} else if (f < 0 || f > 20) {
|
|
|
|
throw new $RangeError("toExponential() argument must be between 0 and 20");
|
2011-05-05 05:21:30 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
return %NumberToExponential(x, f);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.4.7
|
2008-10-03 18:00:28 +00:00
|
|
|
function NumberToPrecision(precision) {
|
2012-12-07 10:20:35 +00:00
|
|
|
var x = this;
|
|
|
|
if (!IS_NUMBER(this)) {
|
|
|
|
if (!IS_NUMBER_WRAPPER(this)) {
|
|
|
|
throw MakeTypeError("incompatible_method_receiver",
|
|
|
|
["Number.prototype.toPrecision", this]);
|
|
|
|
}
|
|
|
|
// Get the value of this number in case it's an object.
|
|
|
|
x = %_ValueOf(this);
|
2011-05-05 05:21:30 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
if (IS_UNDEFINED(precision)) return ToString(%_ValueOf(this));
|
|
|
|
var p = TO_INTEGER(precision);
|
2012-12-07 10:20:35 +00:00
|
|
|
|
|
|
|
if (NUMBER_IS_NAN(x)) return "NaN";
|
|
|
|
if (x == 1/0) return "Infinity";
|
|
|
|
if (x == -1/0) return "-Infinity";
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
if (p < 1 || p > 21) {
|
|
|
|
throw new $RangeError("toPrecision() argument must be between 1 and 21");
|
|
|
|
}
|
|
|
|
return %NumberToPrecision(x, p);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-08 12:49:24 +00:00
|
|
|
// Harmony isFinite.
|
|
|
|
function NumberIsFinite(number) {
|
|
|
|
return IS_NUMBER(number) && NUMBER_IS_FINITE(number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Harmony isNaN.
|
|
|
|
function NumberIsNaN(number) {
|
|
|
|
return IS_NUMBER(number) && NUMBER_IS_NAN(number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
function SetUpNumber() {
|
|
|
|
%CheckIsBootstrapping();
|
2009-08-13 12:35:59 +00:00
|
|
|
%OptimizeObjectForAddingMultipleProperties($Number.prototype, 8);
|
2011-09-05 07:30:35 +00:00
|
|
|
// Set up the constructor property on the Number prototype object.
|
2008-10-03 18:00:28 +00:00
|
|
|
%SetProperty($Number.prototype, "constructor", $Number, DONT_ENUM);
|
|
|
|
|
2009-08-13 12:35:59 +00:00
|
|
|
%OptimizeObjectForAddingMultipleProperties($Number, 5);
|
2008-10-03 18:00:28 +00:00
|
|
|
// ECMA-262 section 15.7.3.1.
|
|
|
|
%SetProperty($Number,
|
|
|
|
"MAX_VALUE",
|
|
|
|
1.7976931348623157e+308,
|
|
|
|
DONT_ENUM | DONT_DELETE | READ_ONLY);
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.3.2.
|
2011-11-28 12:11:00 +00:00
|
|
|
%SetProperty($Number, "MIN_VALUE", 5e-324,
|
|
|
|
DONT_ENUM | DONT_DELETE | READ_ONLY);
|
2008-10-03 18:00:28 +00:00
|
|
|
|
|
|
|
// ECMA-262 section 15.7.3.3.
|
|
|
|
%SetProperty($Number, "NaN", $NaN, DONT_ENUM | DONT_DELETE | READ_ONLY);
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.3.4.
|
|
|
|
%SetProperty($Number,
|
|
|
|
"NEGATIVE_INFINITY",
|
|
|
|
-1/0,
|
|
|
|
DONT_ENUM | DONT_DELETE | READ_ONLY);
|
|
|
|
|
|
|
|
// ECMA-262 section 15.7.3.5.
|
|
|
|
%SetProperty($Number,
|
|
|
|
"POSITIVE_INFINITY",
|
|
|
|
1/0,
|
|
|
|
DONT_ENUM | DONT_DELETE | READ_ONLY);
|
2010-02-17 11:26:00 +00:00
|
|
|
%ToFastProperties($Number);
|
2008-10-03 18:00:28 +00:00
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
// Set up non-enumerable functions on the Number prototype object.
|
2008-10-03 18:00:28 +00:00
|
|
|
InstallFunctions($Number.prototype, DONT_ENUM, $Array(
|
|
|
|
"toString", NumberToString,
|
|
|
|
"toLocaleString", NumberToLocaleString,
|
|
|
|
"valueOf", NumberValueOf,
|
|
|
|
"toFixed", NumberToFixed,
|
|
|
|
"toExponential", NumberToExponential,
|
2010-12-15 09:31:05 +00:00
|
|
|
"toPrecision", NumberToPrecision
|
2008-10-03 18:00:28 +00:00
|
|
|
));
|
2012-03-08 12:49:24 +00:00
|
|
|
InstallFunctions($Number, DONT_ENUM, $Array(
|
|
|
|
"isFinite", NumberIsFinite,
|
|
|
|
"isNaN", NumberIsNaN
|
|
|
|
));
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
SetUpNumber();
|
2008-10-03 18:00:28 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Function
|
|
|
|
|
|
|
|
$Function.prototype.constructor = $Function;
|
|
|
|
|
|
|
|
function FunctionSourceString(func) {
|
2011-09-13 11:42:57 +00:00
|
|
|
while (%IsJSFunctionProxy(func)) {
|
|
|
|
func = %GetCallTrap(func);
|
|
|
|
}
|
|
|
|
|
2009-06-29 10:45:16 +00:00
|
|
|
if (!IS_FUNCTION(func)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
throw new $TypeError('Function.prototype.toString is not generic');
|
2009-06-29 10:45:16 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
var source = %FunctionGetSourceCode(func);
|
2009-09-17 11:11:01 +00:00
|
|
|
if (!IS_STRING(source) || %FunctionIsBuiltin(func)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
var name = %FunctionGetName(func);
|
|
|
|
if (name) {
|
|
|
|
// Mimic what KJS does.
|
|
|
|
return 'function ' + name + '() { [native code] }';
|
|
|
|
} else {
|
|
|
|
return 'function () { [native code] }';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-08 16:14:46 +00:00
|
|
|
var name = %FunctionNameShouldPrintAsAnonymous(func)
|
|
|
|
? 'anonymous'
|
|
|
|
: %FunctionGetName(func);
|
2008-07-03 15:10:15 +00:00
|
|
|
return 'function ' + name + source;
|
2008-10-03 07:14:31 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-03 18:00:28 +00:00
|
|
|
function FunctionToString() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return FunctionSourceString(this);
|
2008-10-03 18:00:28 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2010-07-23 10:08:55 +00:00
|
|
|
// ES5 15.3.4.5
|
|
|
|
function FunctionBind(this_arg) { // Length is 1.
|
2011-09-13 11:42:57 +00:00
|
|
|
if (!IS_SPEC_FUNCTION(this)) {
|
2011-10-17 12:44:16 +00:00
|
|
|
throw new $TypeError('Bind must be called on a function');
|
|
|
|
}
|
|
|
|
var boundFunction = function () {
|
2011-10-18 12:26:53 +00:00
|
|
|
// Poison .arguments and .caller, but is otherwise not detectable.
|
|
|
|
"use strict";
|
2011-10-17 12:44:16 +00:00
|
|
|
// This function must not use any object literals (Object, Array, RegExp),
|
|
|
|
// since the literals-array is being used to store the bound data.
|
|
|
|
if (%_IsConstructCall()) {
|
|
|
|
return %NewObjectFromBound(boundFunction);
|
2010-10-15 13:03:59 +00:00
|
|
|
}
|
2011-10-17 12:44:16 +00:00
|
|
|
var bindings = %BoundFunctionGetBindings(boundFunction);
|
2011-02-08 10:08:47 +00:00
|
|
|
|
2011-10-17 12:44:16 +00:00
|
|
|
var argc = %_ArgumentsLength();
|
|
|
|
if (argc == 0) {
|
|
|
|
return %Apply(bindings[0], bindings[1], bindings, 2, bindings.length - 2);
|
|
|
|
}
|
|
|
|
if (bindings.length === 2) {
|
|
|
|
return %Apply(bindings[0], bindings[1], arguments, 0, argc);
|
|
|
|
}
|
|
|
|
var bound_argc = bindings.length - 2;
|
|
|
|
var argv = new InternalArray(bound_argc + argc);
|
|
|
|
for (var i = 0; i < bound_argc; i++) {
|
|
|
|
argv[i] = bindings[i + 2];
|
|
|
|
}
|
|
|
|
for (var j = 0; j < argc; j++) {
|
|
|
|
argv[i++] = %_Arguments(j);
|
|
|
|
}
|
|
|
|
return %Apply(bindings[0], bindings[1], argv, 0, bound_argc + argc);
|
|
|
|
};
|
|
|
|
|
|
|
|
%FunctionRemovePrototype(boundFunction);
|
|
|
|
var new_length = 0;
|
|
|
|
if (%_ClassOf(this) == "Function") {
|
|
|
|
// Function or FunctionProxy.
|
|
|
|
var old_length = this.length;
|
|
|
|
// FunctionProxies might provide a non-UInt32 value. If so, ignore it.
|
|
|
|
if ((typeof old_length === "number") &&
|
|
|
|
((old_length >>> 0) === old_length)) {
|
2011-02-08 10:08:47 +00:00
|
|
|
var argc = %_ArgumentsLength();
|
2011-10-17 12:44:16 +00:00
|
|
|
if (argc > 0) argc--; // Don't count the thisArg as parameter.
|
|
|
|
new_length = old_length - argc;
|
|
|
|
if (new_length < 0) new_length = 0;
|
|
|
|
}
|
2010-07-23 10:08:55 +00:00
|
|
|
}
|
2011-10-17 12:44:16 +00:00
|
|
|
// This runtime function finds any remaining arguments on the stack,
|
|
|
|
// so we don't pass the arguments object.
|
2011-11-28 12:11:00 +00:00
|
|
|
var result = %FunctionBindArguments(boundFunction, this,
|
|
|
|
this_arg, new_length);
|
2010-07-23 10:08:55 +00:00
|
|
|
|
|
|
|
// We already have caller and arguments properties on functions,
|
|
|
|
// which are non-configurable. It therefore makes no sence to
|
|
|
|
// try to redefine these as defined by the spec. The spec says
|
|
|
|
// that bind should make these throw a TypeError if get or set
|
|
|
|
// is called and make them non-enumerable and non-configurable.
|
2010-10-15 13:03:59 +00:00
|
|
|
// To be consistent with our normal functions we leave this as it is.
|
2011-10-17 12:44:16 +00:00
|
|
|
// TODO(lrn): Do set these to be thrower.
|
2010-07-23 10:08:55 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
function NewFunction(arg1) { // length == 1
|
|
|
|
var n = %_ArgumentsLength();
|
|
|
|
var p = '';
|
|
|
|
if (n > 1) {
|
2011-03-03 11:49:03 +00:00
|
|
|
p = new InternalArray(n - 1);
|
2011-01-05 13:52:00 +00:00
|
|
|
for (var i = 0; i < n - 1; i++) p[i] = %_Arguments(i);
|
2011-01-07 13:21:34 +00:00
|
|
|
p = Join(p, n - 1, ',', NonStringToString);
|
2008-07-03 15:10:15 +00:00
|
|
|
// If the formal parameters string include ) - an illegal
|
|
|
|
// character - it may make the combined function expression
|
|
|
|
// compile. We avoid this problem by checking for this early on.
|
|
|
|
if (p.indexOf(')') != -1) throw MakeSyntaxError('unable_to_parse',[]);
|
|
|
|
}
|
|
|
|
var body = (n > 0) ? ToString(%_Arguments(n - 1)) : '';
|
2008-09-25 07:35:45 +00:00
|
|
|
var source = '(function(' + p + ') {\n' + body + '\n})';
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// The call to SetNewFunctionAttributes will ensure the prototype
|
|
|
|
// property of the resulting function is enumerable (ECMA262, 15.3.5.2).
|
2012-09-05 11:45:58 +00:00
|
|
|
var global_receiver = %GlobalReceiver(global);
|
|
|
|
var f = %_CallFunction(global_receiver, %CompileString(source));
|
|
|
|
|
2011-08-08 16:14:46 +00:00
|
|
|
%FunctionMarkNameShouldPrintAsAnonymous(f);
|
2008-09-25 07:35:45 +00:00
|
|
|
return %SetNewFunctionAttributes(f);
|
2008-10-03 07:14:31 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
%SetCode($Function, NewFunction);
|
2008-10-03 18:00:28 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
function SetUpFunction() {
|
|
|
|
%CheckIsBootstrapping();
|
2008-10-03 18:00:28 +00:00
|
|
|
InstallFunctions($Function.prototype, DONT_ENUM, $Array(
|
2010-07-23 10:08:55 +00:00
|
|
|
"bind", FunctionBind,
|
2008-10-03 18:00:28 +00:00
|
|
|
"toString", FunctionToString
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-09-05 07:30:35 +00:00
|
|
|
SetUpFunction();
|