vmgs_broker/lib.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
//! A task + RPC client for interacting with a shared VMGS instance.
mod broker;
mod client;
pub mod non_volatile_store;
pub mod resolver;
pub use client::VmgsClient;
pub use client::VmgsClientError;
use crate::broker::VmgsBrokerTask;
use pal_async::task::Spawn;
use pal_async::task::Task;
/// Given a fully-initialized VMGS instance, return a VMGS broker task +
/// clonable VmgsClient
pub fn spawn_vmgs_broker(spawner: impl Spawn, vmgs: vmgs::Vmgs) -> (VmgsClient, Task<()>) {
let (control_send, control_recv) = mesh_channel::mpsc_channel();
let process_loop_handle = spawner.spawn("vmgs-broker", async move {
VmgsBrokerTask::new(vmgs).run(control_recv).await
});
(
VmgsClient {
control: control_send,
},
process_loop_handle,
)
}
/// A wrapper around [`VmgsClient`] that restricts its API down to operations
/// that perform no storage IO.
///
/// This types is useful for keeping performance-sensitive code "honest" by
/// making it harder for future refactors to accidentally introduce VMGS IO into
/// performance hotpaths.
#[derive(inspect::Inspect)]
#[inspect(transparent)]
pub struct VmgsThinClient(VmgsClient);
impl VmgsThinClient {
/// Restrict an existing [`VmgsClient`] to only non-IO operations.
pub fn new(vmgs_client: VmgsClient) -> Self {
Self(vmgs_client)
}
/// See [`VmgsClient::save`]
pub async fn save(&self) -> Result<vmgs::save_restore::state::SavedVmgsState, VmgsClientError> {
self.0.save().await
}
}