Commit d1b4d4fe authored by Clemens Hammacher's avatar Clemens Hammacher Committed by Commit Bot

[wasm] [interpreter] Fix GC issue

Make sure that we call the destructors on all embedded object by
replacing the WasmInterpreterInternals::Delete method by an actual
destructor. This way, the compiler automatically calls destructors on
all embedded objects, in particular the IdentityMap in the CodeMap.

This change also requires to release managed objects *before*
tearing down the heap, because the wasm interpreter, referenced via
Managed<>, contains global handles. When those are destroyed, the
isolate still needs to be intact.

Drive-by: Fix include guard in managed.h.

R=ahaas@chromium.org, ulan@chromium.org, mvstanton@chromium.org
BUG=v8:5822

Change-Id: I9a067f037e013c84e4d697a1e913b27c683bb529
Reviewed-on: https://chromium-review.googlesource.com/466187Reviewed-by: 's avatarUlan Degenbaev <ulan@chromium.org>
Reviewed-by: 's avatarMichael Stanton <mvstanton@chromium.org>
Reviewed-by: 's avatarAndreas Haas <ahaas@chromium.org>
Commit-Queue: Clemens Hammacher <clemensh@chromium.org>
Cr-Commit-Position: refs/heads/master@{#44368}
parent ce06d1f2
...@@ -2001,12 +2001,15 @@ Isolate::ThreadDataTable::~ThreadDataTable() { ...@@ -2001,12 +2001,15 @@ Isolate::ThreadDataTable::~ThreadDataTable() {
void Isolate::ReleaseManagedObjects() { void Isolate::ReleaseManagedObjects() {
Isolate::ManagedObjectFinalizer* current = Isolate::ManagedObjectFinalizer* current =
managed_object_finalizers_list_.next_; managed_object_finalizers_list_.next_;
managed_object_finalizers_list_.next_ = nullptr;
while (current != nullptr) { while (current != nullptr) {
Isolate::ManagedObjectFinalizer* next = current->next_; Isolate::ManagedObjectFinalizer* next = current->next_;
current->Dispose(); current->Dispose();
delete current; delete current;
current = next; current = next;
} }
// No new managed objects should pop up during finalization.
DCHECK_NULL(managed_object_finalizers_list_.next_);
} }
Isolate::ManagedObjectFinalizer* Isolate::RegisterForReleaseAtTeardown( Isolate::ManagedObjectFinalizer* Isolate::RegisterForReleaseAtTeardown(
...@@ -2345,6 +2348,9 @@ void Isolate::Deinit() { ...@@ -2345,6 +2348,9 @@ void Isolate::Deinit() {
debug()->Unload(); debug()->Unload();
FreeThreadResources(); FreeThreadResources();
// Release managed objects before shutting down the heap. The finalizer might
// need to access heap objects.
ReleaseManagedObjects();
if (concurrent_recompilation_enabled()) { if (concurrent_recompilation_enabled()) {
optimizing_compile_dispatcher_->Stop(); optimizing_compile_dispatcher_->Stop();
...@@ -2408,7 +2414,6 @@ void Isolate::Deinit() { ...@@ -2408,7 +2414,6 @@ void Isolate::Deinit() {
root_index_map_ = NULL; root_index_map_ = NULL;
ClearSerializerData(); ClearSerializerData();
ReleaseManagedObjects();
} }
......
...@@ -2,8 +2,8 @@ ...@@ -2,8 +2,8 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
#ifndef V8_WASM_MANAGED_H_ #ifndef V8_MANAGED_H_
#define V8_WASM_MANAGED_H_ #define V8_MANAGED_H_
#include "src/factory.h" #include "src/factory.h"
#include "src/global-handles.h" #include "src/global-handles.h"
...@@ -78,4 +78,4 @@ class Managed : public Foreign { ...@@ -78,4 +78,4 @@ class Managed : public Foreign {
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
#endif // V8_WASM_MANAGED_H_ #endif // V8_MANAGED_H_
...@@ -926,10 +926,12 @@ class CodeMap { ...@@ -926,10 +926,12 @@ class CodeMap {
} }
~CodeMap() { ~CodeMap() {
// Destroy the global handles (passing nullptr is ok). // Destroy the global handles.
GlobalHandles::Destroy(Handle<Object>::cast(instance_).location()); // Cast the location, not the handle, because the handle cast might access
// the object behind the handle.
GlobalHandles::Destroy(reinterpret_cast<Object**>(instance_.location()));
GlobalHandles::Destroy( GlobalHandles::Destroy(
Handle<Object>::cast(imported_functions_).location()); reinterpret_cast<Object**>(imported_functions_.location()));
} }
const WasmModule* module() const { return module_; } const WasmModule* module() const { return module_; }
...@@ -2353,8 +2355,6 @@ class WasmInterpreterInternals : public ZoneObject { ...@@ -2353,8 +2355,6 @@ class WasmInterpreterInternals : public ZoneObject {
threads_(zone) { threads_(zone) {
threads_.emplace_back(zone, &codemap_, env.module_env.instance); threads_.emplace_back(zone, &codemap_, env.module_env.instance);
} }
void Delete() { threads_.clear(); }
}; };
//============================================================================ //============================================================================
...@@ -2364,7 +2364,7 @@ WasmInterpreter::WasmInterpreter(Isolate* isolate, const ModuleBytesEnv& env) ...@@ -2364,7 +2364,7 @@ WasmInterpreter::WasmInterpreter(Isolate* isolate, const ModuleBytesEnv& env)
: zone_(isolate->allocator(), ZONE_NAME), : zone_(isolate->allocator(), ZONE_NAME),
internals_(new (&zone_) WasmInterpreterInternals(isolate, &zone_, env)) {} internals_(new (&zone_) WasmInterpreterInternals(isolate, &zone_, env)) {}
WasmInterpreter::~WasmInterpreter() { internals_->Delete(); } WasmInterpreter::~WasmInterpreter() { internals_->~WasmInterpreterInternals(); }
void WasmInterpreter::Run() { internals_->threads_[0].Run(); } void WasmInterpreter::Run() { internals_->threads_[0].Run(); }
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
// Flags: --wasm-interpret-all --allow-natives-syntax // Flags: --wasm-interpret-all --allow-natives-syntax --expose-gc
load('test/mjsunit/wasm/wasm-constants.js'); load('test/mjsunit/wasm/wasm-constants.js');
load('test/mjsunit/wasm/wasm-module-builder.js'); load('test/mjsunit/wasm/wasm-module-builder.js');
...@@ -365,3 +365,36 @@ function checkStack(stack, expected_lines) { ...@@ -365,3 +365,36 @@ function checkStack(stack, expected_lines) {
assertEquals(2 * (11 + 2), instance.exports.main()); assertEquals(2 * (11 + 2), instance.exports.main());
assertEquals(interpreted_before + 2, % WasmNumInterpretedCalls(instance)); assertEquals(interpreted_before + 2, % WasmNumInterpretedCalls(instance));
})(); })();
(function testInterpreterGC() {
function run(f) {
// wrap the creation in a closure so that the only thing returned is
// the module (i.e. the underlying array buffer of WASM wire bytes dies).
var module = (() => {
var builder = new WasmModuleBuilder();
var imp = builder.addImport('mod', 'the_name_of_my_import', kSig_i_i);
builder.addFunction('main', kSig_i_i)
.addBody([kExprGetLocal, 0, kExprCallFunction, imp])
.exportAs('main');
print('module');
return new WebAssembly.Module(builder.toBuffer());
})();
gc();
for (var i = 0; i < 10; i++) {
print(' instance ' + i);
var instance =
new WebAssembly.Instance(module, {'mod': {the_name_of_my_import: f}});
var g = instance.exports.main;
assertEquals('function', typeof g);
for (var j = 0; j < 10; j++) {
assertEquals(f(j), g(j));
}
}
}
for (var i = 0; i < 3; i++) {
run(x => (x + 19));
run(x => (x - 18));
}
})();
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment