2015-05-21 06:15:33 +00:00
|
|
|
// Copyright 2015 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2015-09-16 21:00:45 +00:00
|
|
|
(function(global, utils, extrasUtils) {
|
2015-05-21 06:15:33 +00:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
%CheckIsBootstrapping();
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
// Utils
|
|
|
|
|
|
|
|
var imports = UNDEFINED;
|
2015-05-26 07:24:13 +00:00
|
|
|
var imports_from_experimental = UNDEFINED;
|
2015-08-28 10:22:31 +00:00
|
|
|
var exports_container = %ExportFromRuntime({});
|
2015-10-22 14:40:43 +00:00
|
|
|
var typed_array_setup = UNDEFINED;
|
|
|
|
|
|
|
|
// Register context value to be initialized with a typed array in
|
|
|
|
// Genesis::InitializeBuiltinTypedArrays.
|
|
|
|
function SetupTypedArray(f) {
|
|
|
|
f.next = typed_array_setup;
|
|
|
|
typed_array_setup = f;
|
|
|
|
}
|
2015-05-21 06:15:33 +00:00
|
|
|
|
2015-05-26 07:24:13 +00:00
|
|
|
// Export to other scripts.
|
|
|
|
// In normal natives, this exports functions to other normal natives.
|
|
|
|
// In experimental natives, this exports to other experimental natives and
|
|
|
|
// to normal natives that import using utils.ImportFromExperimental.
|
|
|
|
function Export(f) {
|
2015-08-19 08:34:21 +00:00
|
|
|
f(exports_container);
|
2015-08-18 13:08:05 +00:00
|
|
|
}
|
2015-05-21 06:15:33 +00:00
|
|
|
|
|
|
|
|
2015-08-19 08:34:21 +00:00
|
|
|
// Import from other scripts. The actual importing happens in PostNatives and
|
|
|
|
// PostExperimental so that we can import from scripts executed later. However,
|
|
|
|
// that means that the import is not available until the very end. If the
|
|
|
|
// import needs to be available immediate, use ImportNow.
|
2015-05-26 07:24:13 +00:00
|
|
|
// In normal natives, this imports from other normal natives.
|
|
|
|
// In experimental natives, this imports from other experimental natives and
|
|
|
|
// whitelisted exports from normal natives.
|
|
|
|
function Import(f) {
|
2015-05-21 06:15:33 +00:00
|
|
|
f.next = imports;
|
|
|
|
imports = f;
|
2015-08-18 13:08:05 +00:00
|
|
|
}
|
2015-05-21 06:15:33 +00:00
|
|
|
|
2015-08-28 10:22:31 +00:00
|
|
|
|
2015-08-19 08:34:21 +00:00
|
|
|
// Import immediately from exports of previous scripts. We need this for
|
|
|
|
// functions called during bootstrapping. Hooking up imports in PostNatives
|
|
|
|
// would be too late.
|
2015-08-28 10:22:31 +00:00
|
|
|
function ImportNow(name) {
|
|
|
|
return exports_container[name];
|
2015-08-19 08:34:21 +00:00
|
|
|
}
|
|
|
|
|
2015-05-26 07:24:13 +00:00
|
|
|
|
|
|
|
// In normal natives, import from experimental natives.
|
|
|
|
// Not callable from experimental natives.
|
|
|
|
function ImportFromExperimental(f) {
|
|
|
|
f.next = imports_from_experimental;
|
|
|
|
imports_from_experimental = f;
|
2015-08-18 13:08:05 +00:00
|
|
|
}
|
2015-05-26 07:24:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
function SetFunctionName(f, name, prefix) {
|
|
|
|
if (IS_SYMBOL(name)) {
|
|
|
|
name = "[" + %SymbolDescription(name) + "]";
|
|
|
|
}
|
|
|
|
if (IS_UNDEFINED(prefix)) {
|
|
|
|
%FunctionSetName(f, name);
|
|
|
|
} else {
|
|
|
|
%FunctionSetName(f, prefix + " " + name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function InstallConstants(object, constants) {
|
|
|
|
%CheckIsBootstrapping();
|
|
|
|
%OptimizeObjectForAddingMultipleProperties(object, constants.length >> 1);
|
|
|
|
var attributes = DONT_ENUM | DONT_DELETE | READ_ONLY;
|
|
|
|
for (var i = 0; i < constants.length; i += 2) {
|
|
|
|
var name = constants[i];
|
|
|
|
var k = constants[i + 1];
|
|
|
|
%AddNamedProperty(object, name, k, attributes);
|
|
|
|
}
|
|
|
|
%ToFastProperties(object);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function InstallFunctions(object, attributes, functions) {
|
2015-08-18 13:08:05 +00:00
|
|
|
%CheckIsBootstrapping();
|
|
|
|
%OptimizeObjectForAddingMultipleProperties(object, functions.length >> 1);
|
|
|
|
for (var i = 0; i < functions.length; i += 2) {
|
|
|
|
var key = functions[i];
|
|
|
|
var f = functions[i + 1];
|
|
|
|
SetFunctionName(f, key);
|
|
|
|
%FunctionRemovePrototype(f);
|
|
|
|
%AddNamedProperty(object, key, f, attributes);
|
|
|
|
%SetNativeFlag(f);
|
|
|
|
}
|
|
|
|
%ToFastProperties(object);
|
2015-05-26 07:24:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Helper function to install a getter-only accessor property.
|
2015-11-10 06:58:04 +00:00
|
|
|
function InstallGetter(object, name, getter, attributes, prefix) {
|
2015-05-26 07:24:13 +00:00
|
|
|
%CheckIsBootstrapping();
|
2015-11-10 06:58:04 +00:00
|
|
|
if (IS_UNDEFINED(attributes)) attributes = DONT_ENUM;
|
|
|
|
SetFunctionName(getter, name, IS_UNDEFINED(prefix) ? "get" : prefix);
|
2015-05-26 07:24:13 +00:00
|
|
|
%FunctionRemovePrototype(getter);
|
2015-11-10 06:58:04 +00:00
|
|
|
%DefineGetterPropertyUnchecked(object, name, getter, attributes);
|
2015-05-26 07:24:13 +00:00
|
|
|
%SetNativeFlag(getter);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Helper function to install a getter/setter accessor property.
|
2015-11-10 06:58:04 +00:00
|
|
|
function InstallGetterSetter(object, name, getter, setter, attributes) {
|
2015-05-26 07:24:13 +00:00
|
|
|
%CheckIsBootstrapping();
|
2015-11-10 06:58:04 +00:00
|
|
|
if (IS_UNDEFINED(attributes)) attributes = DONT_ENUM;
|
2015-05-26 07:24:13 +00:00
|
|
|
SetFunctionName(getter, name, "get");
|
|
|
|
SetFunctionName(setter, name, "set");
|
|
|
|
%FunctionRemovePrototype(getter);
|
|
|
|
%FunctionRemovePrototype(setter);
|
|
|
|
%DefineAccessorPropertyUnchecked(object, name, getter, setter, DONT_ENUM);
|
|
|
|
%SetNativeFlag(getter);
|
|
|
|
%SetNativeFlag(setter);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Prevents changes to the prototype of a built-in function.
|
|
|
|
// 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++) {
|
|
|
|
%AddNamedProperty(prototype, fields[i],
|
|
|
|
UNDEFINED, DONT_ENUM | DONT_DELETE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (var i = 0; i < methods.length; i += 2) {
|
|
|
|
var key = methods[i];
|
|
|
|
var f = methods[i + 1];
|
|
|
|
%AddNamedProperty(prototype, key, f, DONT_ENUM | DONT_DELETE | READ_ONLY);
|
|
|
|
%SetNativeFlag(f);
|
|
|
|
}
|
|
|
|
%InternalSetPrototype(prototype, null);
|
|
|
|
%ToFastProperties(prototype);
|
|
|
|
}
|
|
|
|
|
2015-08-21 06:44:17 +00:00
|
|
|
|
2015-05-21 06:15:33 +00:00
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
// To be called by bootstrapper
|
|
|
|
|
2015-05-26 07:24:13 +00:00
|
|
|
function PostNatives(utils) {
|
2015-05-21 06:15:33 +00:00
|
|
|
%CheckIsBootstrapping();
|
|
|
|
|
2015-08-19 08:34:21 +00:00
|
|
|
for ( ; !IS_UNDEFINED(imports); imports = imports.next) {
|
|
|
|
imports(exports_container);
|
|
|
|
}
|
2015-05-21 06:15:33 +00:00
|
|
|
|
2015-08-19 08:34:21 +00:00
|
|
|
// Whitelist of exports from normal natives to experimental natives and debug.
|
|
|
|
var expose_list = [
|
2015-06-02 18:58:12 +00:00
|
|
|
"ArrayToString",
|
2015-10-15 10:22:48 +00:00
|
|
|
"ErrorToString",
|
2015-05-26 07:24:13 +00:00
|
|
|
"GetIterator",
|
|
|
|
"GetMethod",
|
|
|
|
"InnerArrayEvery",
|
|
|
|
"InnerArrayFilter",
|
|
|
|
"InnerArrayForEach",
|
|
|
|
"InnerArrayIndexOf",
|
2015-06-02 18:58:12 +00:00
|
|
|
"InnerArrayJoin",
|
2015-05-26 07:24:13 +00:00
|
|
|
"InnerArrayLastIndexOf",
|
|
|
|
"InnerArrayMap",
|
2015-06-04 16:36:37 +00:00
|
|
|
"InnerArrayReduce",
|
|
|
|
"InnerArrayReduceRight",
|
2015-05-26 07:24:13 +00:00
|
|
|
"InnerArrayReverse",
|
|
|
|
"InnerArraySome",
|
|
|
|
"InnerArraySort",
|
2015-06-02 18:58:12 +00:00
|
|
|
"InnerArrayToLocaleString",
|
2015-05-26 07:24:13 +00:00
|
|
|
"IsNaN",
|
2015-10-22 11:29:40 +00:00
|
|
|
"MakeError",
|
|
|
|
"MakeTypeError",
|
2015-10-15 10:22:48 +00:00
|
|
|
"MapEntries",
|
2015-10-16 11:27:14 +00:00
|
|
|
"MapIterator",
|
2015-10-15 10:22:48 +00:00
|
|
|
"MapIteratorNext",
|
2015-05-21 06:15:33 +00:00
|
|
|
"MathMax",
|
|
|
|
"MathMin",
|
2015-10-20 13:43:44 +00:00
|
|
|
"MaxSimple",
|
|
|
|
"MinSimple",
|
2015-05-26 07:24:13 +00:00
|
|
|
"ObjectIsFrozen",
|
2015-06-11 20:40:54 +00:00
|
|
|
"ObjectDefineProperty",
|
2015-10-15 08:32:34 +00:00
|
|
|
"ObserveArrayMethods",
|
|
|
|
"ObserveObjectMethods",
|
2015-12-10 23:57:39 +00:00
|
|
|
"PromiseChain",
|
|
|
|
"PromiseDeferred",
|
|
|
|
"PromiseResolved",
|
2015-10-16 11:27:14 +00:00
|
|
|
"SameValueZero",
|
|
|
|
"SetIterator",
|
2015-10-15 10:22:48 +00:00
|
|
|
"SetIteratorNext",
|
|
|
|
"SetValues",
|
2015-08-28 10:22:31 +00:00
|
|
|
"SymbolToString",
|
2015-10-15 10:22:48 +00:00
|
|
|
"ToPositiveInteger",
|
2015-08-28 10:22:31 +00:00
|
|
|
// From runtime:
|
|
|
|
"is_concat_spreadable_symbol",
|
|
|
|
"iterator_symbol",
|
|
|
|
"promise_status_symbol",
|
|
|
|
"promise_value_symbol",
|
|
|
|
"reflect_apply",
|
|
|
|
"reflect_construct",
|
2015-11-05 13:08:39 +00:00
|
|
|
"regexp_flags_symbol",
|
2015-08-28 10:22:31 +00:00
|
|
|
"to_string_tag_symbol",
|
2015-12-11 04:26:34 +00:00
|
|
|
"object_to_string",
|
2015-05-21 06:15:33 +00:00
|
|
|
];
|
2015-08-19 08:34:21 +00:00
|
|
|
|
|
|
|
var filtered_exports = {};
|
2015-05-21 06:15:33 +00:00
|
|
|
%OptimizeObjectForAddingMultipleProperties(
|
2015-08-19 08:34:21 +00:00
|
|
|
filtered_exports, expose_list.length);
|
|
|
|
for (var key of expose_list) {
|
|
|
|
filtered_exports[key] = exports_container[key];
|
2015-05-26 07:24:13 +00:00
|
|
|
}
|
2015-08-19 08:34:21 +00:00
|
|
|
%ToFastProperties(filtered_exports);
|
|
|
|
exports_container = filtered_exports;
|
2015-05-21 06:15:33 +00:00
|
|
|
|
2015-05-22 11:21:32 +00:00
|
|
|
utils.PostNatives = UNDEFINED;
|
2015-05-26 07:24:13 +00:00
|
|
|
utils.ImportFromExperimental = UNDEFINED;
|
2015-08-18 13:08:05 +00:00
|
|
|
}
|
2015-05-26 07:24:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
function PostExperimentals(utils) {
|
|
|
|
%CheckIsBootstrapping();
|
2015-08-28 10:22:31 +00:00
|
|
|
%ExportExperimentalFromRuntime(exports_container);
|
2015-05-26 07:24:13 +00:00
|
|
|
for ( ; !IS_UNDEFINED(imports); imports = imports.next) {
|
2015-08-19 08:34:21 +00:00
|
|
|
imports(exports_container);
|
2015-05-26 07:24:13 +00:00
|
|
|
}
|
|
|
|
for ( ; !IS_UNDEFINED(imports_from_experimental);
|
|
|
|
imports_from_experimental = imports_from_experimental.next) {
|
2015-08-19 08:34:21 +00:00
|
|
|
imports_from_experimental(exports_container);
|
2015-05-26 07:24:13 +00:00
|
|
|
}
|
|
|
|
|
2015-11-03 08:24:27 +00:00
|
|
|
utils.CreateDoubleResultArray();
|
|
|
|
utils.CreateDoubleResultArray = UNDEFINED;
|
2015-11-02 13:44:52 +00:00
|
|
|
|
2015-08-12 14:22:07 +00:00
|
|
|
utils.Export = UNDEFINED;
|
2015-10-16 11:27:14 +00:00
|
|
|
utils.PostDebug = UNDEFINED;
|
|
|
|
utils.PostExperimentals = UNDEFINED;
|
2015-10-22 14:40:43 +00:00
|
|
|
typed_array_setup = UNDEFINED;
|
2015-08-18 13:08:05 +00:00
|
|
|
}
|
2015-08-12 14:22:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
function PostDebug(utils) {
|
|
|
|
for ( ; !IS_UNDEFINED(imports); imports = imports.next) {
|
2015-08-19 08:34:21 +00:00
|
|
|
imports(exports_container);
|
2015-08-12 14:22:07 +00:00
|
|
|
}
|
|
|
|
|
2015-11-03 08:24:27 +00:00
|
|
|
utils.CreateDoubleResultArray();
|
|
|
|
utils.CreateDoubleResultArray = UNDEFINED;
|
2015-11-02 13:44:52 +00:00
|
|
|
|
2015-08-19 08:34:21 +00:00
|
|
|
exports_container = UNDEFINED;
|
|
|
|
|
2015-10-16 11:27:14 +00:00
|
|
|
utils.Export = UNDEFINED;
|
|
|
|
utils.Import = UNDEFINED;
|
|
|
|
utils.ImportNow = UNDEFINED;
|
2015-08-12 14:22:07 +00:00
|
|
|
utils.PostDebug = UNDEFINED;
|
2015-05-26 07:24:13 +00:00
|
|
|
utils.PostExperimentals = UNDEFINED;
|
2015-10-22 14:40:43 +00:00
|
|
|
typed_array_setup = UNDEFINED;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-11-02 12:21:13 +00:00
|
|
|
function InitializeBuiltinTypedArrays(utils, rng_state, rempio2result) {
|
2015-10-22 14:40:43 +00:00
|
|
|
var setup_list = typed_array_setup;
|
|
|
|
|
|
|
|
for ( ; !IS_UNDEFINED(setup_list); setup_list = setup_list.next) {
|
2015-11-02 12:21:13 +00:00
|
|
|
setup_list(rng_state, rempio2result);
|
2015-10-22 14:40:43 +00:00
|
|
|
}
|
2015-08-18 13:08:05 +00:00
|
|
|
}
|
2015-05-21 06:15:33 +00:00
|
|
|
|
2015-10-16 11:27:14 +00:00
|
|
|
|
2015-05-26 07:24:13 +00:00
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
|
2015-11-24 13:18:12 +00:00
|
|
|
%OptimizeObjectForAddingMultipleProperties(utils, 14);
|
2015-08-18 13:08:05 +00:00
|
|
|
|
|
|
|
utils.Import = Import;
|
2015-08-19 08:34:21 +00:00
|
|
|
utils.ImportNow = ImportNow;
|
2015-08-18 13:08:05 +00:00
|
|
|
utils.Export = Export;
|
|
|
|
utils.ImportFromExperimental = ImportFromExperimental;
|
|
|
|
utils.SetFunctionName = SetFunctionName;
|
|
|
|
utils.InstallConstants = InstallConstants;
|
|
|
|
utils.InstallFunctions = InstallFunctions;
|
|
|
|
utils.InstallGetter = InstallGetter;
|
|
|
|
utils.InstallGetterSetter = InstallGetterSetter;
|
|
|
|
utils.SetUpLockedPrototype = SetUpLockedPrototype;
|
|
|
|
utils.PostNatives = PostNatives;
|
|
|
|
utils.PostExperimentals = PostExperimentals;
|
|
|
|
utils.PostDebug = PostDebug;
|
|
|
|
|
|
|
|
%ToFastProperties(utils);
|
2015-05-26 07:24:13 +00:00
|
|
|
|
2015-09-16 21:00:45 +00:00
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
|
|
|
|
%OptimizeObjectForAddingMultipleProperties(extrasUtils, 5);
|
|
|
|
|
|
|
|
extrasUtils.logStackTrace = function logStackTrace() {
|
|
|
|
%DebugTrace();
|
|
|
|
};
|
|
|
|
|
|
|
|
extrasUtils.log = function log() {
|
|
|
|
let message = '';
|
|
|
|
for (const arg of arguments) {
|
|
|
|
message += arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
%GlobalPrint(message);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Extras need the ability to store private state on their objects without
|
|
|
|
// exposing it to the outside world.
|
|
|
|
|
|
|
|
extrasUtils.createPrivateSymbol = function createPrivateSymbol(name) {
|
|
|
|
return %CreatePrivateSymbol(name);
|
|
|
|
};
|
|
|
|
|
|
|
|
// These functions are key for safe meta-programming:
|
|
|
|
// http://wiki.ecmascript.org/doku.php?id=conventions:safe_meta_programming
|
|
|
|
//
|
|
|
|
// Technically they could all be derived from combinations of
|
|
|
|
// Function.prototype.{bind,call,apply} but that introduces lots of layers of
|
|
|
|
// indirection and slowness given how un-optimized bind is.
|
|
|
|
|
|
|
|
extrasUtils.simpleBind = function simpleBind(func, thisArg) {
|
|
|
|
return function() {
|
|
|
|
return %Apply(func, thisArg, arguments, 0, arguments.length);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
extrasUtils.uncurryThis = function uncurryThis(func) {
|
|
|
|
return function(thisArg) {
|
|
|
|
return %Apply(func, thisArg, arguments, 1, arguments.length - 1);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
%ToFastProperties(extrasUtils);
|
|
|
|
|
2015-05-21 06:15:33 +00:00
|
|
|
})
|