lower_vtl_permissions_guard/
lib.rs#![cfg(target_os = "linux")]
mod device_dma;
pub use device_dma::LowerVtlDmaBuffer;
use anyhow::Context;
use anyhow::Result;
use inspect::Inspect;
use std::sync::Arc;
use user_driver::DmaClient;
use user_driver::memory::MemoryBlock;
use virt::VtlMemoryProtection;
#[derive(Inspect)]
struct PagesAccessibleToLowerVtl {
#[inspect(skip)]
vtl_protect: Arc<dyn VtlMemoryProtection + Send + Sync>,
#[inspect(with = "|x| inspect::iter_by_index(x).map_value(inspect::AsHex)")]
pages: Vec<u64>,
}
impl PagesAccessibleToLowerVtl {
fn new_from_pages(
vtl_protect: Arc<dyn VtlMemoryProtection + Send + Sync>,
pages: &[u64],
) -> Result<Self> {
for pfn in pages {
vtl_protect
.modify_vtl_page_setting(*pfn, hvdef::HV_MAP_GPA_PERMISSIONS_ALL)
.context("failed to update VTL protections on page")?;
}
Ok(Self {
vtl_protect,
pages: pages.to_vec(),
})
}
}
impl Drop for PagesAccessibleToLowerVtl {
fn drop(&mut self) {
if let Err(err) = self
.pages
.iter()
.map(|pfn| {
self.vtl_protect
.modify_vtl_page_setting(*pfn, hvdef::HV_MAP_GPA_PERMISSIONS_NONE)
.context("failed to update VTL protections on page")
})
.collect::<Result<Vec<_>>>()
{
panic!(
"failed to reset page protections {}",
err.as_ref() as &dyn std::error::Error
);
}
}
}
#[derive(Inspect)]
pub struct LowerVtlMemorySpawner<T: DmaClient> {
#[inspect(skip)]
spawner: T,
#[inspect(skip)]
vtl_protect: Arc<dyn VtlMemoryProtection + Send + Sync>,
}
impl<T: DmaClient> LowerVtlMemorySpawner<T> {
pub fn new(spawner: T, vtl_protect: Arc<dyn VtlMemoryProtection + Send + Sync>) -> Self {
Self {
spawner,
vtl_protect,
}
}
}
impl<T: DmaClient> DmaClient for LowerVtlMemorySpawner<T> {
fn allocate_dma_buffer(&self, len: usize) -> Result<MemoryBlock> {
let mem = self.spawner.allocate_dma_buffer(len)?;
let vtl_guard =
PagesAccessibleToLowerVtl::new_from_pages(self.vtl_protect.clone(), mem.pfns())
.context("failed to lower VTL permissions on memory block")?;
Ok(MemoryBlock::new(LowerVtlDmaBuffer {
block: mem,
_vtl_guard: vtl_guard,
}))
}
fn attach_dma_buffer(&self, _len: usize, _base_pfn: u64) -> Result<MemoryBlock> {
anyhow::bail!("restore is not supported for LowerVtlMemorySpawner")
}
}