Skip to content

Commit

Permalink
feat(swingset): comms: add importer tracking
Browse files Browse the repository at this point in the history
To support `dispatch.retireImport` (the "downstream" retirement that happens
when a Remotable is dropped, and downstream importers need to be informed of
the loss), the comms vat needs to know a list of the kernel and/or remotes
which have imported an object that the comms vat has exported.

The kernel currently implements this in a simple and non-performant way, by
reading every vat's c-list for a matching entry. This makes lookups (which
happen only when an upstream export is deleted) `O(numVats)`, but has no
space penalty. #3223 is about making this faster.

For now, the comms vat uses a differently-non-ideal approach: for each lref,
it maintains a key with a JSON-serialized list of remotes. Adding, removing,
and querying the importers are all is `O(numImporters)`, with a space penalty
of one key per object, and `O(numImporters)` space for the values.

A faster approach is implemented, but commented out, which uses distinct keys
for each (remoteID, lref) pair. This is `O(1)` for all operations, with a
space penalty of one key per import (and a constant-size value). However,
this approach would require a non-existent `syscall.vatstoreGetKeys()` API,
to iterate over vatstore keys. This wouldn't be hard to add, but we should
consider our storage model more carefully before deciding to commit to that
feature.
  • Loading branch information
warner committed Jun 19, 2021
1 parent 3a966ca commit 205ceb0
Show file tree
Hide file tree
Showing 3 changed files with 79 additions and 0 deletions.
10 changes: 10 additions & 0 deletions packages/SwingSet/src/vats/comms/remote.js
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,11 @@ export function makeRemote(state, store, remoteID) {
store.set(toKey, flipRemoteSlot(rref));
const mode = isImport ? 'clist-import' : 'clist-export';
state.incrementRefCount(lref, `{rref}|${remoteID}|clist`, mode);
if (type === 'object') {
if (isImport) {
state.addImporter(lref, remoteID);
}
}
}

// rref is what we would get from them
Expand All @@ -111,6 +116,11 @@ export function makeRemote(state, store, remoteID) {
store.delete(`${remoteID}.c.${rref}`);
store.delete(`${remoteID}.c.${lref}`);
state.decrementRefCount(lref, `{rref}|${remoteID}|clist`, mode);
if (type === 'object') {
if (isImport) {
state.removeImporter(lref, remoteID);
}
}
}

function deleteToRemoteMapping(lref) {
Expand Down
64 changes: 64 additions & 0 deletions packages/SwingSet/src/vats/comms/state.js
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,9 @@ export function makeState(syscall, identifierBase = 0) {
// c.$kfref = $lref // inbound kernel-facing c-list (o+NN/o-NN/p+NN/p-NN -> loNN/lpNN)
// c.$lref = $kfref // outbound kernel-facing c-list (loNN/lpNN -> o+NN/o-NN/p+NN/p-NN)
// cr.$lref = 1 | <missing> // isReachable flag
// //imps.$lref.$remoteID = 1 // one key per importer of $lref (FUTURE)
// imps.$lref = JSON([remoteIDs]) // importers of $lref
// imps.$lref.$remoteID = 1 // one key per importer of $lref
// meta.$kfref = true // flag that $kfref (o+NN/o-NN) is a directly addressable control object
//
// lo.nextID = $NN // local object identifier allocation counter (loNN)
Expand Down Expand Up @@ -164,6 +167,53 @@ export function makeState(syscall, identifierBase = 0) {
deleteLocalPromiseState(lpid);
}

/* we need syscall.vatstoreGetKeys to do it this way
function addImporter(lref, remoteID) {
assert(!lref.includes('.'), lref);
const key = `imps.${lref}.${remoteID}`;
store.set(key, '1');
}
function removeImporter(lref, remoteID) {
assert(!lref.includes('.'), lref);
const key = `imps.${lref}.${remoteID}`;
store.delete(key);
}
function getImporters(lref) {
const remoteIDs = [];
const prefix = `imps.${lref}`;
const startKey = `${prefix}.`;
const endKey = `${prefix}/`; // '.' and '/' are adjacent
for (const k of store.getKeys(startKey, endKey)) {
const remoteID = k.slice(0, prefix.length);
if (remoteID !== 'kernel') {
insistRemoteID(remoteID);
}
remoteIDs.push(remoteID);
}
return harden(remoteIDs);
}
*/

function addImporter(lref, remoteID) {
const key = `imps.${lref}`;
const value = JSON.parse(store.get(key) || '[]');
value.push(remoteID);
value.sort();
store.set(key, JSON.stringify(value));
}
function removeImporter(lref, remoteID) {
assert(!lref.includes('.'), lref);
const key = `imps.${lref}`;
let value = JSON.parse(store.get(key) || '[]');
value = value.filter(r => r !== remoteID);
store.set(key, JSON.stringify(value));
}
function getImporters(lref) {
const key = `imps.${lref}`;
const remoteIDs = JSON.parse(store.get(key) || '[]');
return harden(remoteIDs);
}

/* A mode of 'clist-import' means we increment recognizable, but not
* reachable, because the translation function will call setReachable in a
* moment, and the count should only be changed if it wasn't already
Expand Down Expand Up @@ -354,6 +404,11 @@ export function makeState(syscall, identifierBase = 0) {
store.set(`c.${lref}`, kfref);
const mode = isImport ? 'clist-import' : 'clist-export';
incrementRefCount(lref, `{kfref}|k|clist`, mode);
if (type === 'object') {
if (isImport) {
addImporter(lref, 'kernel');
}
}
}

// GC or delete-remote should just call deleteKernelMapping without any
Expand All @@ -369,6 +424,11 @@ export function makeState(syscall, identifierBase = 0) {
store.delete(`c.${kfref}`);
store.delete(`c.${lref}`);
decrementRefCount(lref, `{kfref}|k|clist`, mode);
if (type === 'object') {
if (isImport) {
removeImporter(lref, 'kernel');
}
}
}

function hasMetaObject(kfref) {
Expand Down Expand Up @@ -619,6 +679,10 @@ export function makeState(syscall, identifierBase = 0) {
getPromiseData,
allocatePromise,

addImporter,
removeImporter,
getImporters,

changeReachable,
lrefMightBeFree,
incrementRefCount,
Expand Down
5 changes: 5 additions & 0 deletions packages/SwingSet/test/test-comms.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,11 @@ test('provideRemoteForLocal', t => {
t.is(provideRemoteForLocal(remoteID, lo4), 'ro-20');
t.is(provideRemoteForLocal(remoteID, lo4), 'ro-20');
t.is(provideRemoteForLocal(remoteID, lo5), 'ro-21');

t.deepEqual(s.getImporters(lo4), [remoteID]);
const { remoteID: remoteID2 } = s.addRemote('remote2', 'o-2');
provideRemoteForLocal(remoteID2, lo4);
t.deepEqual(s.getImporters(lo4), [remoteID, remoteID2]);
});

function mockSyscall() {
Expand Down

0 comments on commit 205ceb0

Please sign in to comment.