| // Copyright 2011 the V8 project authors. All rights reserved. |
| // 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 |
| // |
| // in runtime.js: |
| // const $Object = global.Object; |
| // const $Boolean = global.Boolean; |
| // const $Number = global.Number; |
| // const $Function = global.Function; |
| // const $Array = global.Array; |
| // const $NaN = 0/0; |
| // |
| // in math.js: |
| // const $floor = MathFloor |
| |
| const $isNaN = GlobalIsNaN; |
| const $isFinite = GlobalIsFinite; |
| |
| |
| // ---------------------------------------------------------------------------- |
| |
| |
| // Helper function used to install functions on objects. |
| function InstallFunctions(object, attributes, functions) { |
| if (functions.length >= 8) { |
| %OptimizeObjectForAddingMultipleProperties(object, functions.length >> 1); |
| } |
| for (var i = 0; i < functions.length; i += 2) { |
| var key = functions[i]; |
| var f = functions[i + 1]; |
| %FunctionSetName(f, key); |
| %FunctionRemovePrototype(f); |
| %SetProperty(object, key, f, attributes); |
| %SetNativeFlag(f); |
| } |
| %ToFastProperties(object); |
| } |
| |
| // Emulates JSC by installing functions on a hidden prototype that |
| // lies above the current object/prototype. This lets you override |
| // functions on String.prototype etc. and then restore the old function |
| // with delete. See http://code.google.com/p/chromium/issues/detail?id=1717 |
| function InstallFunctionsOnHiddenPrototype(object, attributes, functions) { |
| var hidden_prototype = new $Object(); |
| %SetHiddenPrototype(object, hidden_prototype); |
| InstallFunctions(hidden_prototype, attributes, functions); |
| } |
| |
| |
| // ---------------------------------------------------------------------------- |
| |
| |
| // ECMA 262 - 15.1.4 |
| function GlobalIsNaN(number) { |
| var n = ToNumber(number); |
| return NUMBER_IS_NAN(n); |
| } |
| |
| |
| // ECMA 262 - 15.1.5 |
| function GlobalIsFinite(number) { |
| if (!IS_NUMBER(number)) number = NonNumberToNumber(number); |
| |
| // NaN - NaN == NaN, Infinity - Infinity == NaN, -Infinity - -Infinity == NaN. |
| return %_IsSmi(number) || number - number == 0; |
| } |
| |
| |
| // ECMA-262 - 15.1.2.2 |
| function GlobalParseInt(string, radix) { |
| if (IS_UNDEFINED(radix) || radix === 10 || radix === 0) { |
| // 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; |
| if (IS_NUMBER(string) && |
| ((0.01 < string && string < 1e9) || |
| (-1e9 < string && string < -0.01))) { |
| // Truncate number. |
| return string | 0; |
| } |
| string = TO_STRING_INLINE(string); |
| radix = radix | 0; |
| } else { |
| // The spec says ToString should be evaluated before ToInt32. |
| string = TO_STRING_INLINE(string); |
| radix = TO_INT32(radix); |
| if (!(radix == 0 || (2 <= radix && radix <= 36))) |
| return $NaN; |
| } |
| |
| if (%_HasCachedArrayIndex(string) && |
| (radix == 0 || radix == 10)) { |
| return %_GetCachedArrayIndex(string); |
| } |
| return %StringParseInt(string, radix); |
| } |
| |
| |
| // ECMA-262 - 15.1.2.3 |
| function GlobalParseFloat(string) { |
| string = TO_STRING_INLINE(string); |
| if (%_HasCachedArrayIndex(string)) return %_GetCachedArrayIndex(string); |
| return %StringParseFloat(string); |
| } |
| |
| |
| function GlobalEval(x) { |
| if (!IS_STRING(x)) return x; |
| |
| var receiver = this; |
| var global_receiver = %GlobalReceiver(global); |
| |
| if (receiver == null && !IS_UNDETECTABLE(receiver)) { |
| receiver = global_receiver; |
| } |
| |
| var this_is_global_receiver = (receiver === global_receiver); |
| var global_is_detached = (global === global_receiver); |
| |
| // 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. |
| if (!this_is_global_receiver || global_is_detached) { |
| throw new $EvalError('The "this" object passed to eval must ' + |
| 'be the global object from which eval originated'); |
| } |
| |
| var f = %CompileString(x); |
| if (!IS_FUNCTION(f)) return f; |
| |
| return %_CallFunction(receiver, f); |
| } |
| |
| |
| // ---------------------------------------------------------------------------- |
| |
| |
| function SetupGlobal() { |
| // ECMA 262 - 15.1.1.1. |
| %SetProperty(global, "NaN", $NaN, DONT_ENUM | DONT_DELETE); |
| |
| // ECMA-262 - 15.1.1.2. |
| %SetProperty(global, "Infinity", 1/0, DONT_ENUM | DONT_DELETE); |
| |
| // ECMA-262 - 15.1.1.3. |
| %SetProperty(global, "undefined", void 0, DONT_ENUM | DONT_DELETE); |
| |
| // Setup non-enumerable function on the global object. |
| InstallFunctions(global, DONT_ENUM, $Array( |
| "isNaN", GlobalIsNaN, |
| "isFinite", GlobalIsFinite, |
| "parseInt", GlobalParseInt, |
| "parseFloat", GlobalParseFloat, |
| "eval", GlobalEval |
| )); |
| } |
| |
| SetupGlobal(); |
| |
| |
| // ---------------------------------------------------------------------------- |
| // Boolean (first part of definition) |
| |
| |
| %SetCode($Boolean, function(x) { |
| if (%_IsConstructCall()) { |
| %_SetValueOf(this, ToBoolean(x)); |
| } else { |
| return ToBoolean(x); |
| } |
| }); |
| |
| %FunctionSetPrototype($Boolean, new $Boolean(false)); |
| |
| %SetProperty($Boolean.prototype, "constructor", $Boolean, DONT_ENUM); |
| |
| // ---------------------------------------------------------------------------- |
| // Object |
| |
| $Object.prototype.constructor = $Object; |
| |
| // ECMA-262 - 15.2.4.2 |
| function ObjectToString() { |
| if (IS_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| return '[object Undefined]'; |
| } |
| if (IS_NULL(this)) return '[object Null]'; |
| return "[object " + %_ClassOf(ToObject(this)) + "]"; |
| } |
| |
| |
| // ECMA-262 - 15.2.4.3 |
| function ObjectToLocaleString() { |
| if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| throw MakeTypeError("called_on_null_or_undefined", |
| ["Object.prototype.toLocaleString"]); |
| } |
| return this.toString(); |
| } |
| |
| |
| // ECMA-262 - 15.2.4.4 |
| function ObjectValueOf() { |
| return ToObject(this); |
| } |
| |
| |
| // ECMA-262 - 15.2.4.5 |
| function ObjectHasOwnProperty(V) { |
| if (%IsJSProxy(this)) { |
| var handler = %GetHandler(this); |
| return CallTrap1(handler, "hasOwn", DerivedHasOwnTrap, TO_STRING_INLINE(V)); |
| } |
| return %HasLocalProperty(TO_OBJECT_INLINE(this), TO_STRING_INLINE(V)); |
| } |
| |
| |
| // ECMA-262 - 15.2.4.6 |
| function ObjectIsPrototypeOf(V) { |
| if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| throw MakeTypeError("called_on_null_or_undefined", |
| ["Object.prototype.isPrototypeOf"]); |
| } |
| if (!IS_SPEC_OBJECT(V)) return false; |
| return %IsInPrototypeChain(this, V); |
| } |
| |
| |
| // ECMA-262 - 15.2.4.6 |
| function ObjectPropertyIsEnumerable(V) { |
| var P = ToString(V); |
| if (%IsJSProxy(this)) { |
| var desc = GetOwnProperty(this, P); |
| return IS_UNDEFINED(desc) ? false : desc.isEnumerable(); |
| } |
| return %IsPropertyEnumerable(ToObject(this), P); |
| } |
| |
| |
| // Extensions for providing property getters and setters. |
| function ObjectDefineGetter(name, fun) { |
| var receiver = this; |
| if (receiver == null && !IS_UNDETECTABLE(receiver)) { |
| receiver = %GlobalReceiver(global); |
| } |
| if (!IS_FUNCTION(fun)) { |
| throw new $TypeError('Object.prototype.__defineGetter__: Expecting function'); |
| } |
| var desc = new PropertyDescriptor(); |
| desc.setGet(fun); |
| desc.setEnumerable(true); |
| desc.setConfigurable(true); |
| DefineOwnProperty(ToObject(receiver), ToString(name), desc, false); |
| } |
| |
| |
| function ObjectLookupGetter(name) { |
| var receiver = this; |
| if (receiver == null && !IS_UNDETECTABLE(receiver)) { |
| receiver = %GlobalReceiver(global); |
| } |
| return %LookupAccessor(ToObject(receiver), ToString(name), GETTER); |
| } |
| |
| |
| function ObjectDefineSetter(name, fun) { |
| var receiver = this; |
| if (receiver == null && !IS_UNDETECTABLE(receiver)) { |
| receiver = %GlobalReceiver(global); |
| } |
| if (!IS_FUNCTION(fun)) { |
| throw new $TypeError( |
| 'Object.prototype.__defineSetter__: Expecting function'); |
| } |
| var desc = new PropertyDescriptor(); |
| desc.setSet(fun); |
| desc.setEnumerable(true); |
| desc.setConfigurable(true); |
| DefineOwnProperty(ToObject(receiver), ToString(name), desc, false); |
| } |
| |
| |
| function ObjectLookupSetter(name) { |
| var receiver = this; |
| if (receiver == null && !IS_UNDETECTABLE(receiver)) { |
| receiver = %GlobalReceiver(global); |
| } |
| return %LookupAccessor(ToObject(receiver), ToString(name), SETTER); |
| } |
| |
| |
| function ObjectKeys(obj) { |
| if (!IS_SPEC_OBJECT(obj)) |
| throw MakeTypeError("obj_ctor_property_non_object", ["keys"]); |
| if (%IsJSProxy(obj)) { |
| var handler = %GetHandler(obj); |
| var names = CallTrap0(handler, "keys", DerivedKeysTrap); |
| return ToStringArray(names); |
| } |
| return %LocalKeys(obj); |
| } |
| |
| |
| // ES5 8.10.1. |
| function IsAccessorDescriptor(desc) { |
| if (IS_UNDEFINED(desc)) return false; |
| return desc.hasGetter() || desc.hasSetter(); |
| } |
| |
| |
| // ES5 8.10.2. |
| function IsDataDescriptor(desc) { |
| if (IS_UNDEFINED(desc)) return false; |
| return desc.hasValue() || desc.hasWritable(); |
| } |
| |
| |
| // ES5 8.10.3. |
| function IsGenericDescriptor(desc) { |
| return !(IsAccessorDescriptor(desc) || IsDataDescriptor(desc)); |
| } |
| |
| |
| function IsInconsistentDescriptor(desc) { |
| return IsAccessorDescriptor(desc) && IsDataDescriptor(desc); |
| } |
| |
| |
| // ES5 8.10.4 |
| function FromPropertyDescriptor(desc) { |
| if (IS_UNDEFINED(desc)) return desc; |
| |
| if (IsDataDescriptor(desc)) { |
| return { value: desc.getValue(), |
| writable: desc.isWritable(), |
| enumerable: desc.isEnumerable(), |
| configurable: desc.isConfigurable() }; |
| } |
| // Must be an AccessorDescriptor then. We never return a generic descriptor. |
| return { get: desc.getGet(), |
| set: desc.getSet(), |
| enumerable: desc.isEnumerable(), |
| configurable: desc.isConfigurable() }; |
| } |
| |
| |
| // Harmony Proxies |
| function FromGenericPropertyDescriptor(desc) { |
| if (IS_UNDEFINED(desc)) return desc; |
| var obj = new $Object(); |
| |
| 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); |
| } |
| return obj; |
| } |
| |
| |
| // ES5 8.10.5. |
| function ToPropertyDescriptor(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| 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; |
| if (!IS_UNDEFINED(get) && !IS_FUNCTION(get)) { |
| throw MakeTypeError("getter_must_be_callable", [get]); |
| } |
| desc.setGet(get); |
| } |
| |
| if ("set" in obj) { |
| var set = obj.set; |
| if (!IS_UNDEFINED(set) && !IS_FUNCTION(set)) { |
| throw MakeTypeError("setter_must_be_callable", [set]); |
| } |
| desc.setSet(set); |
| } |
| |
| if (IsInconsistentDescriptor(desc)) { |
| throw MakeTypeError("value_and_accessor", [obj]); |
| } |
| return desc; |
| } |
| |
| |
| // For Harmony proxies. |
| function ToCompletePropertyDescriptor(obj) { |
| var desc = ToPropertyDescriptor(obj) |
| if (IsGenericDescriptor(desc) || IsDataDescriptor(desc)) { |
| if (!desc.hasValue()) desc.setValue(void 0); |
| if (!desc.hasWritable()) desc.setWritable(false); |
| } else { |
| // Is accessor descriptor. |
| if (!desc.hasGetter()) desc.setGet(void 0); |
| if (!desc.hasSetter()) desc.setSet(void 0); |
| } |
| if (!desc.hasEnumerable()) desc.setEnumerable(false); |
| if (!desc.hasConfigurable()) desc.setConfigurable(false); |
| return desc; |
| } |
| |
| |
| 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; |
| this.hasEnumerable_ = false; |
| this.configurable_ = false; |
| this.hasConfigurable_ = false; |
| this.get_ = void 0; |
| this.hasGetter_ = false; |
| this.set_ = void 0; |
| this.hasSetter_ = false; |
| } |
| |
| PropertyDescriptor.prototype.__proto__ = null; |
| |
| PropertyDescriptor.prototype.toString = function() { |
| return "[object PropertyDescriptor]"; |
| }; |
| |
| PropertyDescriptor.prototype.setValue = function(value) { |
| this.value_ = value; |
| this.hasValue_ = true; |
| } |
| |
| |
| PropertyDescriptor.prototype.getValue = function() { |
| return this.value_; |
| } |
| |
| |
| PropertyDescriptor.prototype.hasValue = function() { |
| return this.hasValue_; |
| } |
| |
| |
| PropertyDescriptor.prototype.setEnumerable = function(enumerable) { |
| this.enumerable_ = enumerable; |
| this.hasEnumerable_ = true; |
| } |
| |
| |
| PropertyDescriptor.prototype.isEnumerable = function () { |
| return this.enumerable_; |
| } |
| |
| |
| PropertyDescriptor.prototype.hasEnumerable = function() { |
| return this.hasEnumerable_; |
| } |
| |
| |
| PropertyDescriptor.prototype.setWritable = function(writable) { |
| this.writable_ = writable; |
| this.hasWritable_ = true; |
| } |
| |
| |
| PropertyDescriptor.prototype.isWritable = function() { |
| return this.writable_; |
| } |
| |
| |
| PropertyDescriptor.prototype.hasWritable = function() { |
| return this.hasWritable_; |
| } |
| |
| |
| PropertyDescriptor.prototype.setConfigurable = function(configurable) { |
| this.configurable_ = configurable; |
| this.hasConfigurable_ = true; |
| } |
| |
| |
| PropertyDescriptor.prototype.hasConfigurable = function() { |
| return this.hasConfigurable_; |
| } |
| |
| |
| PropertyDescriptor.prototype.isConfigurable = function() { |
| return this.configurable_; |
| } |
| |
| |
| PropertyDescriptor.prototype.setGet = function(get) { |
| this.get_ = get; |
| this.hasGetter_ = true; |
| } |
| |
| |
| PropertyDescriptor.prototype.getGet = function() { |
| return this.get_; |
| } |
| |
| |
| PropertyDescriptor.prototype.hasGetter = function() { |
| return this.hasGetter_; |
| } |
| |
| |
| PropertyDescriptor.prototype.setSet = function(set) { |
| this.set_ = set; |
| this.hasSetter_ = true; |
| } |
| |
| |
| PropertyDescriptor.prototype.getSet = function() { |
| return this.set_; |
| } |
| |
| |
| PropertyDescriptor.prototype.hasSetter = function() { |
| return this.hasSetter_; |
| } |
| |
| |
| // 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) { |
| if (desc_array === false) { |
| throw 'Internal error: invalid desc_array'; |
| } |
| |
| if (IS_UNDEFINED(desc_array)) { |
| return void 0; |
| } |
| |
| 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]); |
| } else { |
| desc.setValue(desc_array[VALUE_INDEX]); |
| desc.setWritable(desc_array[WRITABLE_INDEX]); |
| } |
| desc.setEnumerable(desc_array[ENUMERABLE_INDEX]); |
| desc.setConfigurable(desc_array[CONFIGURABLE_INDEX]); |
| |
| return desc; |
| } |
| |
| |
| // 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; |
| } else if (!IS_FUNCTION(trap)) { |
| 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)); |
| } |
| |
| |
| // ES5 section 8.12.1. |
| function GetOwnProperty(obj, v) { |
| var p = ToString(v); |
| if (%IsJSProxy(obj)) { |
| var handler = %GetHandler(obj); |
| var descriptor = CallTrap1(handler, "getOwnPropertyDescriptor", void 0, p); |
| 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; |
| } |
| |
| // GetOwnProperty returns an array indexed by the constants |
| // defined in macros.py. |
| // If p is not a property on obj undefined is returned. |
| var props = %GetOwnProperty(ToObject(obj), ToString(v)); |
| |
| // A false value here means that access checks failed. |
| if (props === false) return void 0; |
| |
| return ConvertDescriptorArrayToDescriptor(props); |
| } |
| |
| |
| // Harmony proxies. |
| function DefineProxyProperty(obj, p, attributes, should_throw) { |
| var handler = %GetHandler(obj); |
| var result = CallTrap2(handler, "defineProperty", void 0, p, attributes); |
| if (!ToBoolean(result)) { |
| if (should_throw) { |
| throw MakeTypeError("handler_returned_false", |
| [handler, "defineProperty"]); |
| } else { |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| |
| // ES5 8.12.9. |
| function DefineOwnProperty(obj, p, desc, should_throw) { |
| if (%IsJSProxy(obj)) { |
| var attributes = FromGenericPropertyDescriptor(desc); |
| return DefineProxyProperty(obj, p, attributes, should_throw); |
| } |
| |
| var current_or_access = %GetOwnProperty(ToObject(obj), ToString(p)); |
| // A false value here means that access checks failed. |
| if (current_or_access === false) return void 0; |
| |
| var current = ConvertDescriptorArrayToDescriptor(current_or_access); |
| var extensible = %IsExtensible(ToObject(obj)); |
| |
| // Error handling according to spec. |
| // Step 3 |
| if (IS_UNDEFINED(current) && !extensible) { |
| if (should_throw) { |
| throw MakeTypeError("define_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| |
| if (!IS_UNDEFINED(current)) { |
| // Step 5 and 6 |
| if ((IsGenericDescriptor(desc) || |
| IsDataDescriptor(desc) == IsDataDescriptor(current)) && |
| (!desc.hasEnumerable() || |
| SameValue(desc.isEnumerable(), current.isEnumerable())) && |
| (!desc.hasConfigurable() || |
| SameValue(desc.isConfigurable(), current.isConfigurable())) && |
| (!desc.hasWritable() || |
| 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; |
| } |
| if (!current.isConfigurable()) { |
| // Step 7 |
| if (desc.isConfigurable() || |
| (desc.hasEnumerable() && |
| desc.isEnumerable() != current.isEnumerable())) { |
| if (should_throw) { |
| throw MakeTypeError("redefine_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| // Step 8 |
| if (!IsGenericDescriptor(desc)) { |
| // Step 9a |
| if (IsDataDescriptor(current) != IsDataDescriptor(desc)) { |
| if (should_throw) { |
| throw MakeTypeError("redefine_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| // Step 10a |
| if (IsDataDescriptor(current) && IsDataDescriptor(desc)) { |
| if (!current.isWritable() && desc.isWritable()) { |
| if (should_throw) { |
| throw MakeTypeError("redefine_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| if (!current.isWritable() && desc.hasValue() && |
| !SameValue(desc.getValue(), current.getValue())) { |
| if (should_throw) { |
| throw MakeTypeError("redefine_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| } |
| // Step 11 |
| if (IsAccessorDescriptor(desc) && IsAccessorDescriptor(current)) { |
| if (desc.hasSetter() && !SameValue(desc.getSet(), current.getSet())) { |
| if (should_throw) { |
| throw MakeTypeError("redefine_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| if (desc.hasGetter() && !SameValue(desc.getGet(),current.getGet())) { |
| if (should_throw) { |
| throw MakeTypeError("redefine_disallowed", [p]); |
| } else { |
| return; |
| } |
| } |
| } |
| } |
| } |
| } |
| |
| // Send flags - enumerable and configurable are common - writable is |
| // 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; |
| } else { |
| 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; |
| |
| 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. |
| |
| if (desc.hasWritable()) { |
| flag |= desc.isWritable() ? 0 : READ_ONLY; |
| } else if (!IS_UNDEFINED(current)) { |
| flag |= current.isWritable() ? 0 : READ_ONLY; |
| } else { |
| flag |= READ_ONLY; |
| } |
| |
| var value = void 0; // Default value is undefined. |
| if (desc.hasValue()) { |
| value = desc.getValue(); |
| } else if (!IS_UNDEFINED(current) && IsDataDescriptor(current)) { |
| value = current.getValue(); |
| } |
| |
| %DefineOrRedefineDataProperty(obj, p, value, flag); |
| } else if (IsGenericDescriptor(desc)) { |
| // Step 12 - updating an existing accessor property with generic |
| // descriptor. Changing flags only. |
| %DefineOrRedefineAccessorProperty(obj, p, GETTER, current.getGet(), flag); |
| } else { |
| // 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. |
| if (desc.hasGetter()) { |
| %DefineOrRedefineAccessorProperty(obj, p, GETTER, desc.getGet(), flag); |
| } |
| if (desc.hasSetter()) { |
| %DefineOrRedefineAccessorProperty(obj, p, SETTER, desc.getSet(), flag); |
| } |
| } |
| return true; |
| } |
| |
| |
| // ES5 section 15.2.3.2. |
| function ObjectGetPrototypeOf(obj) { |
| if (!IS_SPEC_OBJECT(obj)) |
| throw MakeTypeError("obj_ctor_property_non_object", ["getPrototypeOf"]); |
| return %GetPrototype(obj); |
| } |
| |
| |
| // ES5 section 15.2.3.3 |
| function ObjectGetOwnPropertyDescriptor(obj, p) { |
| if (!IS_SPEC_OBJECT(obj)) |
| throw MakeTypeError("obj_ctor_property_non_object", |
| ["getOwnPropertyDescriptor"]); |
| var desc = GetOwnProperty(obj, p); |
| return FromPropertyDescriptor(desc); |
| } |
| |
| |
| // 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); |
| var names = {} |
| for (var index = 0; index < n; index++) { |
| var s = ToString(obj[index]); |
| if (s in names) { |
| throw MakeTypeError("proxy_repeated_prop_name", [obj, trap, s]) |
| } |
| array[index] = s; |
| names.s = 0; |
| } |
| return array; |
| } |
| |
| |
| // ES5 section 15.2.3.4. |
| function ObjectGetOwnPropertyNames(obj) { |
| if (!IS_SPEC_OBJECT(obj)) |
| throw MakeTypeError("obj_ctor_property_non_object", ["getOwnPropertyNames"]); |
| |
| // Special handling for proxies. |
| if (%IsJSProxy(obj)) { |
| var handler = %GetHandler(obj); |
| var names = CallTrap0(handler, "getOwnPropertyNames", void 0); |
| return ToStringArray(names, "getOwnPropertyNames"); |
| } |
| |
| // Find all the indexed properties. |
| |
| // Get the local element names. |
| var propertyNames = %GetLocalElementNames(obj); |
| |
| // Get names for indexed interceptor properties. |
| if (%GetInterceptorInfo(obj) & 1) { |
| var indexedInterceptorNames = |
| %GetIndexedInterceptorElementNames(obj); |
| if (indexedInterceptorNames) |
| propertyNames = propertyNames.concat(indexedInterceptorNames); |
| } |
| |
| // Find all the named properties. |
| |
| // Get the local property names. |
| propertyNames = propertyNames.concat(%GetLocalPropertyNames(obj)); |
| |
| // Get names for named interceptor properties if any. |
| |
| if (%GetInterceptorInfo(obj) & 2) { |
| var namedInterceptorNames = |
| %GetNamedInterceptorPropertyNames(obj); |
| if (namedInterceptorNames) { |
| propertyNames = propertyNames.concat(namedInterceptorNames); |
| } |
| } |
| |
| // Property names are expected to be unique strings. |
| var propertySet = {}; |
| 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; |
| } |
| propertyNames.length = j; |
| |
| return propertyNames; |
| } |
| |
| |
| // ES5 section 15.2.3.5. |
| function ObjectCreate(proto, properties) { |
| if (!IS_SPEC_OBJECT(proto) && proto !== null) { |
| throw MakeTypeError("proto_object_or_null", [proto]); |
| } |
| var obj = new $Object(); |
| obj.__proto__ = proto; |
| if (!IS_UNDEFINED(properties)) ObjectDefineProperties(obj, properties); |
| return obj; |
| } |
| |
| |
| // ES5 section 15.2.3.6. |
| function ObjectDefineProperty(obj, p, attributes) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["defineProperty"]); |
| } |
| var name = ToString(p); |
| 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?). |
| var attributesClone = {} |
| 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); |
| } |
| return obj; |
| } |
| |
| |
| function GetOwnEnumerablePropertyNames(properties) { |
| var names = new InternalArray(); |
| for (var key in properties) { |
| if (%HasLocalProperty(properties, key)) { |
| names.push(key); |
| } |
| } |
| return names; |
| } |
| |
| |
| // ES5 section 15.2.3.7. |
| function ObjectDefineProperties(obj, properties) { |
| if (!IS_SPEC_OBJECT(obj)) |
| throw MakeTypeError("obj_ctor_property_non_object", ["defineProperties"]); |
| var props = ToObject(properties); |
| var names = GetOwnEnumerablePropertyNames(props); |
| for (var i = 0; i < names.length; i++) { |
| var name = names[i]; |
| var desc = ToPropertyDescriptor(props[name]); |
| DefineOwnProperty(obj, name, desc, true); |
| } |
| return obj; |
| } |
| |
| |
| // Harmony proxies. |
| function ProxyFix(obj) { |
| var handler = %GetHandler(obj); |
| var props = CallTrap0(handler, "fix", void 0); |
| if (IS_UNDEFINED(props)) { |
| throw MakeTypeError("handler_returned_undefined", [handler, "fix"]); |
| } |
| %Fix(obj); |
| ObjectDefineProperties(obj, props); |
| } |
| |
| |
| // ES5 section 15.2.3.8. |
| function ObjectSeal(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["seal"]); |
| } |
| if (%IsJSProxy(obj)) { |
| ProxyFix(obj); |
| } |
| var names = ObjectGetOwnPropertyNames(obj); |
| for (var i = 0; i < names.length; i++) { |
| var name = names[i]; |
| var desc = GetOwnProperty(obj, name); |
| if (desc.isConfigurable()) { |
| desc.setConfigurable(false); |
| DefineOwnProperty(obj, name, desc, true); |
| } |
| } |
| %PreventExtensions(obj); |
| return obj; |
| } |
| |
| |
| // ES5 section 15.2.3.9. |
| function ObjectFreeze(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["freeze"]); |
| } |
| if (%IsJSProxy(obj)) { |
| ProxyFix(obj); |
| } |
| var names = ObjectGetOwnPropertyNames(obj); |
| for (var i = 0; i < names.length; i++) { |
| var name = names[i]; |
| var desc = GetOwnProperty(obj, name); |
| if (desc.isWritable() || desc.isConfigurable()) { |
| if (IsDataDescriptor(desc)) desc.setWritable(false); |
| desc.setConfigurable(false); |
| DefineOwnProperty(obj, name, desc, true); |
| } |
| } |
| %PreventExtensions(obj); |
| return obj; |
| } |
| |
| |
| // ES5 section 15.2.3.10 |
| function ObjectPreventExtension(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["preventExtension"]); |
| } |
| if (%IsJSProxy(obj)) { |
| ProxyFix(obj); |
| } |
| %PreventExtensions(obj); |
| return obj; |
| } |
| |
| |
| // ES5 section 15.2.3.11 |
| function ObjectIsSealed(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["isSealed"]); |
| } |
| if (%IsJSProxy(obj)) { |
| return false; |
| } |
| var names = ObjectGetOwnPropertyNames(obj); |
| for (var i = 0; i < names.length; i++) { |
| var name = names[i]; |
| var desc = GetOwnProperty(obj, name); |
| if (desc.isConfigurable()) return false; |
| } |
| if (!ObjectIsExtensible(obj)) { |
| return true; |
| } |
| return false; |
| } |
| |
| |
| // ES5 section 15.2.3.12 |
| function ObjectIsFrozen(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["isFrozen"]); |
| } |
| if (%IsJSProxy(obj)) { |
| return false; |
| } |
| var names = ObjectGetOwnPropertyNames(obj); |
| for (var i = 0; i < names.length; i++) { |
| var name = names[i]; |
| var desc = GetOwnProperty(obj, name); |
| if (IsDataDescriptor(desc) && desc.isWritable()) return false; |
| if (desc.isConfigurable()) return false; |
| } |
| if (!ObjectIsExtensible(obj)) { |
| return true; |
| } |
| return false; |
| } |
| |
| |
| // ES5 section 15.2.3.13 |
| function ObjectIsExtensible(obj) { |
| if (!IS_SPEC_OBJECT(obj)) { |
| throw MakeTypeError("obj_ctor_property_non_object", ["isExtensible"]); |
| } |
| if (%IsJSProxy(obj)) { |
| return true; |
| } |
| return %IsExtensible(obj); |
| } |
| |
| |
| %SetCode($Object, function(x) { |
| if (%_IsConstructCall()) { |
| if (x == null) return this; |
| return ToObject(x); |
| } else { |
| if (x == null) return { }; |
| return ToObject(x); |
| } |
| }); |
| |
| %SetExpectedNumberOfProperties($Object, 4); |
| |
| // ---------------------------------------------------------------------------- |
| |
| |
| function SetupObject() { |
| // Setup non-enumerable functions on the Object.prototype object. |
| 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 |
| )); |
| InstallFunctions($Object, DONT_ENUM, $Array( |
| "keys", ObjectKeys, |
| "create", ObjectCreate, |
| "defineProperty", ObjectDefineProperty, |
| "defineProperties", ObjectDefineProperties, |
| "freeze", ObjectFreeze, |
| "getPrototypeOf", ObjectGetPrototypeOf, |
| "getOwnPropertyDescriptor", ObjectGetOwnPropertyDescriptor, |
| "getOwnPropertyNames", ObjectGetOwnPropertyNames, |
| "isExtensible", ObjectIsExtensible, |
| "isFrozen", ObjectIsFrozen, |
| "isSealed", ObjectIsSealed, |
| "preventExtensions", ObjectPreventExtension, |
| "seal", ObjectSeal |
| )); |
| } |
| |
| SetupObject(); |
| |
| |
| // ---------------------------------------------------------------------------- |
| // Boolean |
| |
| function BooleanToString() { |
| // NOTE: Both Boolean objects and values can enter here as |
| // 'this'. This is not as dictated by ECMA-262. |
| 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'; |
| } |
| |
| |
| function BooleanValueOf() { |
| // NOTE: Both Boolean objects and values can enter here as |
| // 'this'. This is not as dictated by ECMA-262. |
| if (!IS_BOOLEAN(this) && !IS_BOOLEAN_WRAPPER(this)) |
| throw new $TypeError('Boolean.prototype.valueOf is not generic'); |
| return %_ValueOf(this); |
| } |
| |
| |
| // ---------------------------------------------------------------------------- |
| |
| |
| function SetupBoolean() { |
| InstallFunctions($Boolean.prototype, DONT_ENUM, $Array( |
| "toString", BooleanToString, |
| "valueOf", BooleanValueOf |
| )); |
| } |
| |
| SetupBoolean(); |
| |
| // ---------------------------------------------------------------------------- |
| // Number |
| |
| // Set the Number function and constructor. |
| %SetCode($Number, function(x) { |
| var value = %_ArgumentsLength() == 0 ? 0 : ToNumber(x); |
| if (%_IsConstructCall()) { |
| %_SetValueOf(this, value); |
| } else { |
| return value; |
| } |
| }); |
| |
| %FunctionSetPrototype($Number, new $Number(0)); |
| |
| // ECMA-262 section 15.7.4.2. |
| function NumberToString(radix) { |
| // 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)) { |
| if (!IS_NUMBER_WRAPPER(this)) |
| throw new $TypeError('Number.prototype.toString is not generic'); |
| // 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) { |
| return %_NumberToString(number); |
| } |
| |
| // 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); |
| } |
| |
| |
| // ECMA-262 section 15.7.4.3 |
| function NumberToLocaleString() { |
| if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| throw MakeTypeError("called_on_null_or_undefined", |
| ["Number.prototype.toLocaleString"]); |
| } |
| return this.toString(); |
| } |
| |
| |
| // ECMA-262 section 15.7.4.4 |
| function NumberValueOf() { |
| // NOTE: Both Number objects and values can enter here as |
| // 'this'. This is not as dictated by ECMA-262. |
| if (!IS_NUMBER(this) && !IS_NUMBER_WRAPPER(this)) |
| throw new $TypeError('Number.prototype.valueOf is not generic'); |
| return %_ValueOf(this); |
| } |
| |
| |
| // ECMA-262 section 15.7.4.5 |
| function NumberToFixed(fractionDigits) { |
| var f = TO_INTEGER(fractionDigits); |
| if (f < 0 || f > 20) { |
| throw new $RangeError("toFixed() digits argument must be between 0 and 20"); |
| } |
| if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| throw MakeTypeError("called_on_null_or_undefined", |
| ["Number.prototype.toFixed"]); |
| } |
| var x = ToNumber(this); |
| return %NumberToFixed(x, f); |
| } |
| |
| |
| // ECMA-262 section 15.7.4.6 |
| function NumberToExponential(fractionDigits) { |
| var f = -1; |
| if (!IS_UNDEFINED(fractionDigits)) { |
| f = TO_INTEGER(fractionDigits); |
| if (f < 0 || f > 20) { |
| throw new $RangeError("toExponential() argument must be between 0 and 20"); |
| } |
| } |
| if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| throw MakeTypeError("called_on_null_or_undefined", |
| ["Number.prototype.toExponential"]); |
| } |
| var x = ToNumber(this); |
| return %NumberToExponential(x, f); |
| } |
| |
| |
| // ECMA-262 section 15.7.4.7 |
| function NumberToPrecision(precision) { |
| if (IS_NULL_OR_UNDEFINED(this) && !IS_UNDETECTABLE(this)) { |
| throw MakeTypeError("called_on_null_or_undefined", |
| ["Number.prototype.toPrecision"]); |
| } |
| if (IS_UNDEFINED(precision)) return ToString(%_ValueOf(this)); |
| var p = TO_INTEGER(precision); |
| if (p < 1 || p > 21) { |
| throw new $RangeError("toPrecision() argument must be between 1 and 21"); |
| } |
| var x = ToNumber(this); |
| return %NumberToPrecision(x, p); |
| } |
| |
| |
| // ---------------------------------------------------------------------------- |
| |
| function SetupNumber() { |
| %OptimizeObjectForAddingMultipleProperties($Number.prototype, 8); |
| // Setup the constructor property on the Number prototype object. |
| %SetProperty($Number.prototype, "constructor", $Number, DONT_ENUM); |
| |
| %OptimizeObjectForAddingMultipleProperties($Number, 5); |
| // 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. |
| %SetProperty($Number, "MIN_VALUE", 5e-324, DONT_ENUM | DONT_DELETE | READ_ONLY); |
| |
| // 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); |
| %ToFastProperties($Number); |
| |
| // Setup non-enumerable functions on the Number prototype object. |
| InstallFunctions($Number.prototype, DONT_ENUM, $Array( |
| "toString", NumberToString, |
| "toLocaleString", NumberToLocaleString, |
| "valueOf", NumberValueOf, |
| "toFixed", NumberToFixed, |
| "toExponential", NumberToExponential, |
| "toPrecision", NumberToPrecision |
| )); |
| } |
| |
| SetupNumber(); |
| |
| |
| // ---------------------------------------------------------------------------- |
| // Function |
| |
| $Function.prototype.constructor = $Function; |
| |
| function FunctionSourceString(func) { |
| if (!IS_FUNCTION(func)) { |
| throw new $TypeError('Function.prototype.toString is not generic'); |
| } |
| |
| var source = %FunctionGetSourceCode(func); |
| if (!IS_STRING(source) || %FunctionIsBuiltin(func)) { |
| var name = %FunctionGetName(func); |
| if (name) { |
| // Mimic what KJS does. |
| return 'function ' + name + '() { [native code] }'; |
| } else { |
| return 'function () { [native code] }'; |
| } |
| } |
| |
| var name = %FunctionNameShouldPrintAsAnonymous(func) |
| ? 'anonymous' |
| : %FunctionGetName(func); |
| return 'function ' + name + source; |
| } |
| |
| |
| function FunctionToString() { |
| return FunctionSourceString(this); |
| } |
| |
| |
| // ES5 15.3.4.5 |
| function FunctionBind(this_arg) { // Length is 1. |
| if (!IS_FUNCTION(this)) { |
| throw new $TypeError('Bind must be called on a function'); |
| } |
| // this_arg is not an argument that should be bound. |
| var argc_bound = (%_ArgumentsLength() || 1) - 1; |
| var fn = this; |
| if (argc_bound == 0) { |
| var result = function() { |
| if (%_IsConstructCall()) { |
| // %NewObjectFromBound implicitly uses arguments passed to this |
| // function. We do not pass the arguments object explicitly to avoid |
| // materializing it and guarantee that this function will be optimized. |
| return %NewObjectFromBound(fn, null); |
| } |
| |
| return fn.apply(this_arg, arguments); |
| }; |
| } else { |
| var bound_args = new InternalArray(argc_bound); |
| for(var i = 0; i < argc_bound; i++) { |
| bound_args[i] = %_Arguments(i+1); |
| } |
| |
| var result = function() { |
| // If this is a construct call we use a special runtime method |
| // to generate the actual object using the bound function. |
| if (%_IsConstructCall()) { |
| // %NewObjectFromBound implicitly uses arguments passed to this |
| // function. We do not pass the arguments object explicitly to avoid |
| // materializing it and guarantee that this function will be optimized. |
| return %NewObjectFromBound(fn, bound_args); |
| } |
| |
| // Combine the args we got from the bind call with the args |
| // given as argument to the invocation. |
| var argc = %_ArgumentsLength(); |
| var args = new InternalArray(argc + argc_bound); |
| // Add bound arguments. |
| for (var i = 0; i < argc_bound; i++) { |
| args[i] = bound_args[i]; |
| } |
| // Add arguments from call. |
| for (var i = 0; i < argc; i++) { |
| args[argc_bound + i] = %_Arguments(i); |
| } |
| return fn.apply(this_arg, args); |
| }; |
| } |
| |
| // 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. |
| // To be consistent with our normal functions we leave this as it is. |
| |
| // Set the correct length. |
| var length = (this.length - argc_bound) > 0 ? this.length - argc_bound : 0; |
| %FunctionRemovePrototype(result); |
| %FunctionSetBound(result); |
| %BoundFunctionSetLength(result, length); |
| return result; |
| } |
| |
| |
| function NewFunction(arg1) { // length == 1 |
| var n = %_ArgumentsLength(); |
| var p = ''; |
| if (n > 1) { |
| p = new InternalArray(n - 1); |
| for (var i = 0; i < n - 1; i++) p[i] = %_Arguments(i); |
| p = Join(p, n - 1, ',', NonStringToString); |
| // 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)) : ''; |
| var source = '(function(' + p + ') {\n' + body + '\n})'; |
| |
| // The call to SetNewFunctionAttributes will ensure the prototype |
| // property of the resulting function is enumerable (ECMA262, 15.3.5.2). |
| var f = %CompileString(source)(); |
| %FunctionMarkNameShouldPrintAsAnonymous(f); |
| return %SetNewFunctionAttributes(f); |
| } |
| |
| %SetCode($Function, NewFunction); |
| |
| // ---------------------------------------------------------------------------- |
| |
| function SetupFunction() { |
| InstallFunctions($Function.prototype, DONT_ENUM, $Array( |
| "bind", FunctionBind, |
| "toString", FunctionToString |
| )); |
| } |
| |
| SetupFunction(); |