937b5011b8
This way, each lazy function needs to handle only the data relevant to itself. This reduced data handling overheads. Other changes: 1) Don't deserialize the data; once it's on the heap, it can stay there. Lazy function compilation is only done in the main thread. 2) Separate ProducedPreParsedScopeData and ConsumedPreParsedScopeData. It's clearer, because: - The data looks fundamentally different when we're producing it and when we're consuming it. - Cleanly separates the operations we can do in the "producing phase" and in the "consuming phase". Bug: v8:5516 Change-Id: I6985a6621f71b348a55155724765624b5d5f7c33 Reviewed-on: https://chromium-review.googlesource.com/528094 Commit-Queue: Marja Hölttä <marja@chromium.org> Reviewed-by: Michael Starzinger <mstarzinger@chromium.org> Reviewed-by: Adam Klein <adamk@chromium.org> Cr-Commit-Position: refs/heads/master@{#46347}
206 lines
5.3 KiB
JavaScript
206 lines
5.3 KiB
JavaScript
// Copyright 2017 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.
|
|
|
|
// Flags: --experimental-preparser-scope-analysis
|
|
|
|
(function TestBasicSkipping() {
|
|
var result = 0;
|
|
|
|
function lazy(ctxt_alloc_param) {
|
|
var ctxt_alloc_var = 10;
|
|
function skip_me() {
|
|
result = ctxt_alloc_param + ctxt_alloc_var;
|
|
}
|
|
return skip_me;
|
|
}
|
|
// Test that parameters and variables of the outer function get context
|
|
// allocated even if we skip the inner function.
|
|
lazy(9)();
|
|
assertEquals(19, result);
|
|
})();
|
|
|
|
(function TestSkippingFunctionWithEval() {
|
|
var result = 0;
|
|
|
|
function lazy(ctxt_alloc_param) {
|
|
var ctxt_alloc_var = 10;
|
|
function skip_me() {
|
|
eval('result = ctxt_alloc_param + ctxt_alloc_var');
|
|
}
|
|
return skip_me;
|
|
}
|
|
// Test that parameters and variables of the outer function get context
|
|
// allocated even if we skip the inner function.
|
|
lazy(9)();
|
|
assertEquals(19, result);
|
|
})();
|
|
|
|
(function TestCtxtAllocatingNonSimpleParams1() {
|
|
var result = 0;
|
|
|
|
function lazy([other_param1, ctxt_alloc_param, other_param2]) {
|
|
function skip_me() {
|
|
result = ctxt_alloc_param;
|
|
}
|
|
return skip_me;
|
|
}
|
|
// Test that parameters and variables of the outer function get context
|
|
// allocated even if we skip the inner function.
|
|
lazy([30, 29, 28])();
|
|
assertEquals(29, result);
|
|
})();
|
|
|
|
(function TestCtxtAllocatingNonSimpleParams2() {
|
|
var result = 0;
|
|
|
|
function lazy({a: other_param1, b: ctxt_alloc_param, c: other_param2}) {
|
|
function skip_me() {
|
|
result = ctxt_alloc_param;
|
|
}
|
|
return skip_me;
|
|
}
|
|
// Test that parameters and variables of the outer function get context
|
|
// allocated even if we skip the inner function.
|
|
lazy({a: 31, b: 32, c: 33})();
|
|
assertEquals(32, result);
|
|
})();
|
|
|
|
(function TestCtxtAllocatingNonSimpleParams3() {
|
|
var result = 0;
|
|
|
|
function lazy(...ctxt_alloc_param) {
|
|
function skip_me() {
|
|
result = ctxt_alloc_param;
|
|
}
|
|
return skip_me;
|
|
}
|
|
// Test that parameters and variables of the outer function get context
|
|
// allocated even if we skip the inner function.
|
|
lazy(34, 35)();
|
|
assertEquals([34, 35], result);
|
|
})();
|
|
|
|
// Skippable top level functions.
|
|
let result = 0;
|
|
function lazy_top_level(ctxt_alloc_param) {
|
|
let ctxt_alloc_var = 24;
|
|
function skip_me() {
|
|
result = ctxt_alloc_param + ctxt_alloc_var;
|
|
}
|
|
skip_me();
|
|
}
|
|
|
|
lazy_top_level(10);
|
|
assertEquals(34, result);
|
|
|
|
// Tests for using a function name in an inner function.
|
|
var TestUsingNamedExpressionName1 = function this_is_the_name() {
|
|
function inner() {
|
|
this_is_the_name;
|
|
}
|
|
inner();
|
|
}
|
|
TestUsingNamedExpressionName1();
|
|
|
|
function TestUsingNamedExpressionName2() {
|
|
let f = function this_is_the_name() {
|
|
function inner() {
|
|
this_is_the_name;
|
|
}
|
|
inner();
|
|
}
|
|
f();
|
|
}
|
|
TestUsingNamedExpressionName2();
|
|
|
|
function TestSkippedFunctionInsideLoopInitializer() {
|
|
let saved_func;
|
|
for (let i = 0, f = function() { return i }; i < 1; ++i) {
|
|
saved_func = f;
|
|
}
|
|
assertEquals(0, saved_func());
|
|
}
|
|
TestSkippedFunctionInsideLoopInitializer();
|
|
|
|
(function TestSkippedFunctionWithParameters() {
|
|
var result = 0;
|
|
|
|
function lazy(ctxt_alloc_param) {
|
|
var ctxt_alloc_var = 10;
|
|
function skip_me(param1, param2) {
|
|
result = ctxt_alloc_param + ctxt_alloc_var + param1 + param2;
|
|
}
|
|
return skip_me;
|
|
}
|
|
lazy(9)(8, 7);
|
|
assertEquals(34, result);
|
|
})();
|
|
|
|
function TestSkippingDeeperLazyFunctions() {
|
|
let result = 0;
|
|
function inner_lazy(ctxt_alloc_param) {
|
|
let ctxt_alloc_var = 13;
|
|
function skip_me() {
|
|
result = ctxt_alloc_param + ctxt_alloc_var;
|
|
}
|
|
return skip_me;
|
|
}
|
|
let f = inner_lazy(12);
|
|
f();
|
|
assertEquals(25, result);
|
|
}
|
|
|
|
TestSkippingDeeperLazyFunctions();
|
|
|
|
function TestEagerFunctionsBetweenLazyFunctions() {
|
|
let result = 0;
|
|
// We produce one data set for TestEagerFunctionsBetweenLazyFunctions and
|
|
// another one for inner. The variable data for eager belongs to the former
|
|
// data set.
|
|
let ctxt_allocated1 = 3;
|
|
(function eager() {
|
|
let ctxt_allocated2 = 4;
|
|
function inner() {
|
|
result = ctxt_allocated1 + ctxt_allocated2;
|
|
}
|
|
return inner;
|
|
})()();
|
|
assertEquals(7, result);
|
|
}
|
|
|
|
TestEagerFunctionsBetweenLazyFunctions();
|
|
|
|
function TestEagerNotIifeFunctionsBetweenLazyFunctions() {
|
|
let result = 0;
|
|
// We produce one data set for TestEagerFunctionsBetweenLazyFunctions and
|
|
// another one for inner. The variable data for eager belongs to the former
|
|
// data set.
|
|
let ctxt_allocated1 = 3;
|
|
(function eager_not_iife() {
|
|
let ctxt_allocated2 = 4;
|
|
function inner() {
|
|
result = ctxt_allocated1 + ctxt_allocated2;
|
|
}
|
|
return inner;
|
|
}); // Function not called; not an iife.
|
|
// This is just a regression test. We cannot test that the context allocation
|
|
// was done correctly (since there's no way to call eager_not_iife), but code
|
|
// like this used to trigger some DCHECKs.
|
|
}
|
|
|
|
TestEagerNotIifeFunctionsBetweenLazyFunctions();
|
|
|
|
// Regression test for functions inside a lazy arrow function. (Only top-level
|
|
// arrow functions are lazy, so this cannot be wrapped in a function.)
|
|
result = 0;
|
|
let f1 = (ctxt_alloc_param) => {
|
|
let ctxt_alloc_var = 10;
|
|
function inner() {
|
|
result = ctxt_alloc_param + ctxt_alloc_var;
|
|
}
|
|
return inner;
|
|
}
|
|
f1(9)();
|
|
assertEquals(19, result);
|