From 66ac657d51d3d1be61ee4a6e9a621a664086ee57 Mon Sep 17 00:00:00 2001 From: Richard Gibson Date: Thu, 16 Mar 2023 13:24:45 -0400 Subject: [PATCH] fix: Move upgrade-time abandonExports responsibility into the kernel Fixes #6696 --- packages/SwingSet/src/kernel/kernel.js | 31 +++++++++++------- packages/SwingSet/src/kernel/kernelSyscall.js | 2 ++ .../SwingSet/src/kernel/state/kernelKeeper.js | 32 +++++++++++++++++++ .../src/kernel/state/storageHelper.js | 17 ++++++++-- packages/SwingSet/src/kernel/vatTranslator.js | 2 ++ packages/SwingSet/test/test-state.js | 15 +++++++++ packages/swingset-liveslots/src/stop-vat.js | 16 +++++++--- 7 files changed, 96 insertions(+), 19 deletions(-) diff --git a/packages/SwingSet/src/kernel/kernel.js b/packages/SwingSet/src/kernel/kernel.js index e2a5e5ef22a..b3799c5b86f 100644 --- a/packages/SwingSet/src/kernel/kernel.js +++ b/packages/SwingSet/src/kernel/kernel.js @@ -885,25 +885,32 @@ export default function buildKernel( return results; } - // stopVat succeeded. finish cleanup on behalf of the worker. + // stopVat succeeded, now we finish cleanup on behalf of the worker - // TODO: send BOYD to the vat, to give it one last chance to clean - // up, drop imports, and delete durable data. If we ever have a - // vat that is so broken it can't do BOYD, we can make that - // optional. #7001 + // TODO: send BOYD so the terminating vat has one last chance to clean + // up, drop imports, and delete durable data. + // If a vat is so broken it can't do BOYD, we can make that optional. + // https://github.com/Agoric/agoric-sdk/issues/7001 - // walk c-list for all decided promises, reject them all + // reject all promises for which the vat was decider for (const kpid of kernelKeeper.enumeratePromisesByDecider(vatID)) { resolveToError(kpid, disconnectionCapData, vatID); } - // TODO: getNonDurableObjectExports, synthesize abandonVSO, - // execute it as if it were a syscall. (maybe distinguish between - // reachable/recognizable exports, abandon the reachable, retire - // the recognizable) #6696 + // simulate an abandonExports syscall from the vat, + // without making an *actual* syscall that could pollute logs + const abandonedObjects = [ + ...kernelKeeper.enumerateNonDurableObjectExports(vatID), + ]; + for (const { kref, vref } of abandonedObjects) { + /** @see translateAbandonExports in {@link ./vatTranslator.js} */ + vatKeeper.deleteCListEntry(kref, vref); + /** @see abandonExports in {@link ./kernelSyscall.js} */ + kernelKeeper.orphanKernelObject(kref, vatID); + } - // cleanup done, now we stop the worker, delete the transcript and - // any snapshot + // cleanup done, now we reset the worker to a clean state with no + // transcript or snapshot and prime everything for the next incarnation. await vatWarehouse.resetWorker(vatID); const source = { bundleID }; diff --git a/packages/SwingSet/src/kernel/kernelSyscall.js b/packages/SwingSet/src/kernel/kernelSyscall.js index 807526d9233..191c39b5c54 100644 --- a/packages/SwingSet/src/kernel/kernelSyscall.js +++ b/packages/SwingSet/src/kernel/kernelSyscall.js @@ -196,6 +196,8 @@ export function makeKernelSyscallHandler(tools) { function abandonExports(vatID, koids) { Array.isArray(koids) || Fail`abandonExports given non-Array ${koids}`; for (const koid of koids) { + // note that this is effectful and also performed outside of a syscall + // by processUpgradeVat in {@link ./kernel.js} kernelKeeper.orphanKernelObject(koid, vatID); } return OKNULL; diff --git a/packages/SwingSet/src/kernel/state/kernelKeeper.js b/packages/SwingSet/src/kernel/state/kernelKeeper.js index e3afc63d10d..791be925754 100644 --- a/packages/SwingSet/src/kernel/state/kernelKeeper.js +++ b/packages/SwingSet/src/kernel/state/kernelKeeper.js @@ -541,6 +541,37 @@ export default function makeKernelKeeper(kernelStorage, kernelSlog) { kvStore.set(`${kernelSlot}.refCount`, `${reachable},${recognizable}`); } + /** + * Iterate over non-durable objects exported by a vat. + * + * @param {string} vatID + * @yields {{kref: string, vref: string}} + */ + function* enumerateNonDurableObjectExports(vatID) { + insistVatID(vatID); + // vrefs for exported objects start with o+NN (ephemeral), + // o+vNN/MM (merely-virtual), or o+dNN/MM (durable). + // We iterate through all ephemeral and virtual entries so the kernel + // can ensure that they are abandoned by a vat being upgraded. + const prefix = `${vatID}.c.`; + const ephStart = `${prefix}o+`; + const durStart = `${prefix}o+d`; + const virStart = `${prefix}o+v`; + /** @type {[string, string?][]} */ + const ranges = [[ephStart, durStart], [virStart]]; + for (const range of ranges) { + for (const k of enumeratePrefixedKeys(kvStore, ...range)) { + const vref = k.slice(prefix.length); + // exclude the root object, which is replaced by upgrade + if (vref !== 'o+0') { + const kref = kvStore.get(k); + assert.typeof(kref, 'string'); + yield { kref, vref }; + } + } + } + } + /** * Allocate a new koid. * @@ -1574,6 +1605,7 @@ export default function makeKernelKeeper(kernelStorage, kernelSlog) { incrementRefCount, decrementRefCount, getObjectRefCount, + enumerateNonDurableObjectExports, addToRunQueue, isRunQueueEmpty, diff --git a/packages/SwingSet/src/kernel/state/storageHelper.js b/packages/SwingSet/src/kernel/state/storageHelper.js index 70334ca2210..432260da99d 100644 --- a/packages/SwingSet/src/kernel/state/storageHelper.js +++ b/packages/SwingSet/src/kernel/state/storageHelper.js @@ -2,15 +2,26 @@ import { assert } from '@agoric/assert'; -export function* enumeratePrefixedKeys(kvStore, prefix) { - // return an iterator of all existing keys that start with - // ${prefix}, in lexicographic order, excluding ${prefix} itself +/** + * Iterate over keys with a given prefix, in lexicographic order, + * excluding an exact match of the prefix. + * + * @param {KVStore} kvStore + * @param {string} prefix + * @param {string} [exclusiveEnd] + * @yields {string} the next key with the prefix that is not >= exclusiveEnd + */ +export function* enumeratePrefixedKeys(kvStore, prefix, exclusiveEnd) { + /** @type {string | undefined} */ let key = prefix; for (;;) { key = kvStore.getNextKey(key); if (!key || !key.startsWith(prefix)) { break; } + if (exclusiveEnd && key >= exclusiveEnd) { + break; + } yield key; } } diff --git a/packages/SwingSet/src/kernel/vatTranslator.js b/packages/SwingSet/src/kernel/vatTranslator.js index c1ec8dcacd6..e8b37c9b4f1 100644 --- a/packages/SwingSet/src/kernel/vatTranslator.js +++ b/packages/SwingSet/src/kernel/vatTranslator.js @@ -479,6 +479,8 @@ function makeTranslateVatSyscallToKernelSyscall(vatID, kernelKeeper) { assert.equal(allocatedByVat, true); // abandon *exports*, not imports // kref must already be in the clist const kref = mapVatSlotToKernelSlot(vref, gcSyscallMapOpts); + // note that this is effectful and also performed outside of a syscall + // by processUpgradeVat in {@link ./kernel.js} vatKeeper.deleteCListEntry(kref, vref); return kref; }); diff --git a/packages/SwingSet/test/test-state.js b/packages/SwingSet/test/test-state.js index 72932c659aa..ae67607fa63 100644 --- a/packages/SwingSet/test/test-state.js +++ b/packages/SwingSet/test/test-state.js @@ -125,6 +125,21 @@ test('storage helpers', t => { 'bar.3', 'bar.5', ]); + + t.deepEqual(Array.from(enumeratePrefixedKeys(kv, 'bar', 'bar.1')), []); + t.deepEqual(Array.from(enumeratePrefixedKeys(kv, 'bar', 'bar.4')), [ + 'bar.1', + 'bar.3', + ]); + t.deepEqual(Array.from(enumeratePrefixedKeys(kv, 'bar', 'bar.5')), [ + 'bar.1', + 'bar.3', + ]); + t.deepEqual(Array.from(enumeratePrefixedKeys(kv, 'bar', 'bar.6')), [ + 'bar.1', + 'bar.3', + 'bar.5', + ]); }); function buildKeeperStorageInMemory() { diff --git a/packages/swingset-liveslots/src/stop-vat.js b/packages/swingset-liveslots/src/stop-vat.js index 4a5ab226ff5..2784a0e81da 100644 --- a/packages/swingset-liveslots/src/stop-vat.js +++ b/packages/swingset-liveslots/src/stop-vat.js @@ -33,6 +33,7 @@ import { enumerateKeysWithPrefix } from './vatstore-iterators.js'; const rootSlot = makeVatSlot('object', true, 0n); +// eslint-disable-next-line no-unused-vars function identifyExportedRemotables( vrefSet, { exportedRemotables, valToSlot }, @@ -59,6 +60,7 @@ function identifyExportedRemotables( } } +// eslint-disable-next-line no-unused-vars function identifyExportedFacets(vrefSet, { syscall, vrm }) { // Find all exported (non-durable) virtual object facets, which are // doomed because merely-virtual objects don't survive upgrade. We @@ -96,6 +98,7 @@ function identifyExportedFacets(vrefSet, { syscall, vrm }) { } } +// eslint-disable-next-line no-unused-vars function abandonExports(vrefSet, tools) { // Pretend the kernel dropped everything in the set. The Remotables // will be removed from exportedRemotables. If the export was the @@ -285,10 +288,15 @@ export async function releaseOldState(tools) { // refcount decrements which may drop some virtuals from the DB. It // might also drop some objects from RAM. - const abandonedVrefSet = new Set(); - identifyExportedRemotables(abandonedVrefSet, tools); - identifyExportedFacets(abandonedVrefSet, tools); - abandonExports(abandonedVrefSet, tools); + // TODO: Decide how much (if any) cleanup to do here in the vat. + // The kernel simulates abandonExports as part of vat upgrade, + // but does not decrement vat-side refcounts which could allow us to + // drop durable objects that were only being kept alive by references from + // non-durable objects. + // const abandonedVrefSet = new Set(); + // identifyExportedRemotables(abandonedVrefSet, tools); + // identifyExportedFacets(abandonedVrefSet, tools); + // abandonExports(abandonedVrefSet, tools); // bringOutYourDead remains to ensure that the LRU cache is flushed, // but the rest of this function has been disabled to improve stop-vat