[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
// Copyright 2016 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: --expose-wasm --expose-gc --allow-natives-syntax
|
|
|
|
|
2021-06-01 12:46:36 +00:00
|
|
|
d8.file.execute("test/mjsunit/wasm/wasm-module-builder.js");
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
|
2016-11-03 15:53:20 +00:00
|
|
|
// Use global variables for all values where the test wants to maintain strict
|
|
|
|
// control over value lifetime. Using local variables would not give sufficient
|
|
|
|
// guarantees of the value lifetime.
|
|
|
|
var module;
|
|
|
|
var instance1;
|
|
|
|
var instance2;
|
|
|
|
var instance3;
|
|
|
|
var instance4;
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
|
2016-11-03 15:53:20 +00:00
|
|
|
(function CompiledModuleInstancesInitialize1to3() {
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
var builder = new WasmModuleBuilder();
|
|
|
|
|
|
|
|
builder.addMemory(1,1, true);
|
2016-12-20 15:32:56 +00:00
|
|
|
builder.addImport("", "getValue", kSig_i_v);
|
2016-10-06 15:43:10 +00:00
|
|
|
builder.addFunction("f", kSig_i_v)
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
.addBody([
|
2016-09-27 20:46:10 +00:00
|
|
|
kExprCallFunction, 0
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
]).exportFunc();
|
|
|
|
|
2016-11-03 15:53:20 +00:00
|
|
|
module = new WebAssembly.Module(builder.toBuffer());
|
2018-04-06 10:18:18 +00:00
|
|
|
|
|
|
|
print("Initial instances=0");
|
2018-06-27 12:50:53 +00:00
|
|
|
assertEquals(0, %WasmGetNumberOfInstances(module));
|
2016-12-20 15:32:56 +00:00
|
|
|
instance1 = new WebAssembly.Instance(module, {"": {getValue: () => 1}});
|
2018-04-06 10:18:18 +00:00
|
|
|
|
|
|
|
print("Initial instances=1");
|
2018-06-27 12:50:53 +00:00
|
|
|
assertEquals(1, %WasmGetNumberOfInstances(module));
|
2016-12-20 15:32:56 +00:00
|
|
|
instance2 = new WebAssembly.Instance(module, {"": {getValue: () => 2}});
|
2018-04-06 10:18:18 +00:00
|
|
|
|
|
|
|
print("Initial instances=2");
|
2018-06-27 12:50:53 +00:00
|
|
|
assertEquals(2, %WasmGetNumberOfInstances(module));
|
2016-12-20 15:32:56 +00:00
|
|
|
instance3 = new WebAssembly.Instance(module, {"": {getValue: () => 3}});
|
2018-04-06 10:18:18 +00:00
|
|
|
|
|
|
|
print("Initial instances=3");
|
2018-06-27 12:50:53 +00:00
|
|
|
assertEquals(3, %WasmGetNumberOfInstances(module));
|
2016-11-03 15:53:20 +00:00
|
|
|
})();
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
|
2016-11-03 15:53:20 +00:00
|
|
|
(function CompiledModuleInstancesClear1() {
|
|
|
|
assertEquals(1, instance1.exports.f());
|
|
|
|
instance1 = null;
|
|
|
|
})();
|
|
|
|
|
2018-06-27 12:50:53 +00:00
|
|
|
// Note that two GC's are required because weak slots clearing is deferred.
|
|
|
|
gc();
|
2016-11-03 15:53:20 +00:00
|
|
|
gc();
|
2018-04-06 10:18:18 +00:00
|
|
|
print("After gc instances=2");
|
2018-06-27 12:50:53 +00:00
|
|
|
assertEquals(2, %WasmGetNumberOfInstances(module));
|
2016-11-03 15:53:20 +00:00
|
|
|
|
|
|
|
(function CompiledModuleInstancesClear3() {
|
|
|
|
assertEquals(3, instance3.exports.f());
|
|
|
|
instance3 = null;
|
|
|
|
})();
|
|
|
|
|
2018-06-27 12:50:53 +00:00
|
|
|
// Note that two GC's are required because weak slots clearing is deferred.
|
|
|
|
gc();
|
2016-11-03 15:53:20 +00:00
|
|
|
gc();
|
2018-04-06 10:18:18 +00:00
|
|
|
print("After gc instances=1");
|
2018-06-27 12:50:53 +00:00
|
|
|
assertEquals(1, %WasmGetNumberOfInstances(module));
|
2016-11-03 15:53:20 +00:00
|
|
|
|
|
|
|
(function CompiledModuleInstancesClear2() {
|
|
|
|
assertEquals(2, instance2.exports.f());
|
|
|
|
instance2 = null;
|
|
|
|
})();
|
|
|
|
|
2018-06-27 12:50:53 +00:00
|
|
|
// Note that two GC's are required because weak slots clearing is deferred.
|
2018-04-06 10:18:18 +00:00
|
|
|
gc();
|
2016-11-03 15:53:20 +00:00
|
|
|
gc();
|
2018-06-27 12:50:53 +00:00
|
|
|
print("After gc instances=0");
|
|
|
|
assertEquals(0, %WasmGetNumberOfInstances(module));
|
2016-11-03 15:53:20 +00:00
|
|
|
|
|
|
|
(function CompiledModuleInstancesInitialize4AndClearModule() {
|
2016-12-20 15:32:56 +00:00
|
|
|
instance4 = new WebAssembly.Instance(module, {"": {getValue: () => 4}});
|
2016-11-03 15:53:20 +00:00
|
|
|
assertEquals(4, instance4.exports.f());
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
module = null;
|
2020-04-09 13:53:20 +00:00
|
|
|
instance4 = null;
|
[wasm] reuse the first compiled module.
This change avoids needing to keep around an unused compiled
module. Instead, the result of compiling the wasm bytes is
given to the first instance. The module object and that instance object
point to the same compiled module. Instances are, then, cloned from
the compiled module the module object points to. When an instance is
collected, we make sure that the module object still has a clone
available, and, if the last instance is GC-ed, we also reset the compiled
module so that it does not reference its heap, so that it (==heap) may
be collected.
This is achieved by linking the clones in a double-linked list and
registering a finalizer for each. When we create an instance, we tie it
in the front of the list, making the module object point to it (O(1)). When
the finalizer is called, we relink the list over the dying object (O(1)). The
costliest operation is finalizing the last instance, since we need to visit
all wasm functions and reset heap references.
BUG=v8:5316
Committed: https://crrev.com/01f5af515728aebe6c5246f4f7dd6c573e8748af
Review-Url: https://codereview.chromium.org/2305903002
Cr-Original-Commit-Position: refs/heads/master@{#39153}
Cr-Commit-Position: refs/heads/master@{#39361}
2016-09-12 23:12:25 +00:00
|
|
|
})();
|
2016-11-03 15:53:20 +00:00
|
|
|
|
2018-06-27 12:50:53 +00:00
|
|
|
// Note that two GC's are required because weak slots clearing is deferred.
|
2017-08-19 16:35:05 +00:00
|
|
|
gc();
|
2016-11-03 15:53:20 +00:00
|
|
|
gc();
|