2019-09-04 18:51:41 +02:00
|
|
|
// Copyright (C) 2019 Ecma International. All rights reserved.
|
|
|
|
// This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
/*---
|
|
|
|
description: >
|
|
|
|
Collection of functions used to capture references cleanup from garbage collectors
|
|
|
|
features: [Symbol, async-functions]
|
|
|
|
flags: [non-deterministic]
|
|
|
|
features: [FinalizationGroup]
|
2019-09-25 02:22:26 +02:00
|
|
|
defines: [asyncGC, asyncGCDeref, resolveAsyncGC]
|
2019-09-04 18:51:41 +02:00
|
|
|
---*/
|
|
|
|
|
|
|
|
function asyncGC(...targets) {
|
|
|
|
var fg = new FinalizationGroup(() => {});
|
|
|
|
var length = targets.length;
|
|
|
|
|
|
|
|
for (let target of targets) {
|
|
|
|
fg.register(target, 'target');
|
|
|
|
target = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
targets = null;
|
|
|
|
|
|
|
|
return Promise.resolve('tick').then(() => asyncGCDeref()).then(() => {
|
|
|
|
var names;
|
|
|
|
|
|
|
|
// consume iterator to capture names
|
|
|
|
fg.cleanupSome(iter => { names = [...iter]; });
|
|
|
|
|
|
|
|
if (!names || names.length != length) {
|
|
|
|
throw asyncGC.notCollected;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
asyncGC.notCollected = Symbol('Object was not collected');
|
|
|
|
|
|
|
|
async function asyncGCDeref() {
|
|
|
|
var trigger;
|
|
|
|
|
|
|
|
// TODO: Remove this when $262.clearKeptObject becomes documented and required
|
|
|
|
if ($262.clearKeptObjects) {
|
|
|
|
trigger = $262.clearKeptObjects();
|
|
|
|
}
|
|
|
|
|
|
|
|
await $262.gc();
|
|
|
|
|
|
|
|
return Promise.resolve(trigger);
|
|
|
|
}
|
|
|
|
|
|
|
|
function resolveAsyncGC(err) {
|
2019-09-23 18:49:32 +02:00
|
|
|
if (err === asyncGC.notCollected) {
|
|
|
|
// Do not fail as GC can't provide necessary resources.
|
|
|
|
$DONE();
|
|
|
|
}
|
2019-09-04 18:51:41 +02:00
|
|
|
|
|
|
|
$DONE(err);
|
|
|
|
}
|