#![expect(unsafe_code)]
use crate::run::CommonState;
use crate::run::RunnerBuilder;
use anyhow::Context as _;
use futures::executor::block_on;
use guestmem::GuestMemory;
use hvdef::Vtl;
use std::future::Future;
use std::future::poll_fn;
use std::pin::pin;
use std::sync::Arc;
use std::task::Context;
use std::task::Waker;
use virt::BindProcessor;
use virt::Hypervisor;
use virt::Partition;
use virt::PartitionConfig;
use virt::PartitionMemoryMapper;
use virt::ProtoPartition;
use virt::ProtoPartitionConfig;
use virt::VpIndex;
impl CommonState {
pub async fn run_host_vmm<H: Hypervisor>(&mut self, mut hv: H) -> anyhow::Result<()>
where
H::Partition: Partition + PartitionMemoryMapper,
{
let proto = hv
.new_partition(ProtoPartitionConfig {
processor_topology: &self.processor_topology,
hv_config: None,
vmtime: &self.vmtime_source,
user_mode_apic: false,
isolation: virt::IsolationType::None,
})
.context("failed to create proto partition")?;
let guest_memory = GuestMemory::allocate(self.memory_layout.end_of_ram() as usize);
let (partition, vps) = proto
.build(PartitionConfig {
mem_layout: &self.memory_layout,
guest_memory: &guest_memory,
cpuid: &[],
vtl0_alias_map: None,
})
.context("failed to build partition")?;
let partition = Arc::new(partition);
for r in self.memory_layout.ram() {
let range = r.range;
unsafe {
partition
.memory_mapper(Vtl::Vtl0)
.map_range(
guest_memory.inner_buf().unwrap()
[range.start() as usize..range.end() as usize]
.as_ptr()
.cast_mut()
.cast(),
range.len() as usize,
range.start(),
true,
true,
)
.context("failed to map memory")
}?;
}
self.run(&guest_memory, partition.caps(), async |_this, runner| {
let [vp] = vps.try_into().ok().unwrap();
start_vp(partition.clone(), vp, runner).await?;
Ok(())
})
.await
}
}
trait RequestYield: Send + Sync {
fn request_yield(&self, vp_index: VpIndex);
}
impl<T: Partition> RequestYield for T {
fn request_yield(&self, vp_index: VpIndex) {
self.request_yield(vp_index)
}
}
struct VpWaker {
partition: Arc<dyn RequestYield>,
vp: VpIndex,
inner: Waker,
}
impl VpWaker {
fn new(partition: Arc<dyn RequestYield>, vp: VpIndex, waker: Waker) -> Self {
Self {
partition,
vp,
inner: waker,
}
}
}
impl std::task::Wake for VpWaker {
fn wake_by_ref(self: &Arc<Self>) {
self.partition.request_yield(self.vp);
self.inner.wake_by_ref();
}
fn wake(self: Arc<Self>) {
self.wake_by_ref()
}
}
async fn start_vp(
partition: Arc<dyn RequestYield>,
mut vp: impl 'static + BindProcessor + Send,
mut runner: RunnerBuilder,
) -> anyhow::Result<()> {
let (bind_result_send, bind_result_recv) = mesh::oneshot();
let _vp_thread = std::thread::spawn(move || {
let vp_index = VpIndex::BSP;
let r = vp
.bind()
.context("failed to bind vp")
.and_then(|vp| runner.build(vp));
let (vp, r) = match r {
Ok(vp) => (Some(vp), Ok(())),
Err(err) => (None, Err(err)),
};
bind_result_send.send(r);
let Some(mut vp) = vp else { return };
block_on(async {
let mut run = pin!(vp.run_vp());
poll_fn(|cx| {
let waker = Waker::from(Arc::new(VpWaker::new(
partition.clone(),
vp_index,
cx.waker().clone(),
)));
run.as_mut().poll(&mut Context::from_waker(&waker))
})
.await
})
});
bind_result_recv.await.unwrap()
}