mod construct;
mod modify;
mod runtime;
mod start;
pub use runtime::PetriVmOpenVmm;
use crate::Firmware;
use crate::PetriLogFile;
use crate::PetriLogSource;
use crate::PetriVm;
use crate::PetriVmConfig;
use crate::disk_image::AgentImage;
use crate::linux_direct_serial_agent::LinuxDirectSerialAgent;
use crate::openhcl_diag::OpenHclDiagHandler;
use async_trait::async_trait;
use framebuffer::FramebufferAccess;
use get_resources::ged::FirmwareEvent;
use guid::Guid;
use hvlite_defs::config::Config;
use hyperv_ic_resources::shutdown::ShutdownRpc;
use mesh::Receiver;
use mesh::Sender;
use pal_async::DefaultDriver;
use pal_async::socket::PolledSocket;
use pal_async::task::Task;
use petri_artifacts_common::tags::MachineArch;
use petri_artifacts_common::tags::OsFlavor;
use petri_artifacts_core::ArtifactResolver;
use petri_artifacts_core::ResolvedArtifact;
use pipette_client::PipetteClient;
use std::path::PathBuf;
use tempfile::TempPath;
use unix_socket::UnixListener;
use vtl2_settings_proto::Vtl2Settings;
pub(crate) const SCSI_INSTANCE: Guid = guid::guid!("27b553e8-8b39-411b-a55f-839971a7884f");
pub(crate) const BOOT_NVME_INSTANCE: Guid = guid::guid!("92bc8346-718b-449a-8751-edbf3dcd27e4");
const MANA_INSTANCE: Guid = guid::guid!("f9641cf4-d915-4743-a7d8-efa75db7b85a");
pub(crate) const BOOT_NVME_NSID: u32 = 37;
pub(crate) const BOOT_NVME_LUN: u32 = 1;
pub struct PetriVmArtifactsOpenVmm {
firmware: Firmware,
arch: MachineArch,
agent_image: AgentImage,
openhcl_agent_image: Option<AgentImage>,
openvmm_path: ResolvedArtifact,
}
impl PetriVmArtifactsOpenVmm {
pub fn new(resolver: &ArtifactResolver<'_>, firmware: Firmware, arch: MachineArch) -> Self {
let agent_image = AgentImage::new(resolver, arch, firmware.os_flavor());
let openhcl_agent_image = if firmware.is_openhcl() {
Some(AgentImage::new(resolver, arch, OsFlavor::Linux))
} else {
None
};
Self {
firmware,
arch,
agent_image,
openhcl_agent_image,
openvmm_path: resolver
.require(petri_artifacts_vmm_test::artifacts::OPENVMM_NATIVE)
.erase(),
}
}
}
pub struct PetriVmConfigOpenVmm {
firmware: Firmware,
arch: MachineArch,
config: Config,
resources: PetriVmResourcesOpenVmm,
openvmm_log_file: PetriLogFile,
ged: Option<get_resources::ged::GuestEmulationDeviceHandle>,
vtl2_settings: Option<Vtl2Settings>,
framebuffer_access: Option<FramebufferAccess>,
}
#[async_trait]
impl PetriVmConfig for PetriVmConfigOpenVmm {
async fn run_without_agent(self: Box<Self>) -> anyhow::Result<Box<dyn PetriVm>> {
Ok(Box::new(Self::run_without_agent(*self).await?))
}
async fn run_with_lazy_pipette(mut self: Box<Self>) -> anyhow::Result<Box<dyn PetriVm>> {
Ok(Box::new(Self::run_with_lazy_pipette(*self).await?))
}
async fn run(self: Box<Self>) -> anyhow::Result<(Box<dyn PetriVm>, PipetteClient)> {
let (vm, client) = Self::run(*self).await?;
Ok((Box::new(vm), client))
}
}
struct PetriVmResourcesOpenVmm {
log_stream_tasks: Vec<Task<anyhow::Result<()>>>,
firmware_event_recv: Receiver<FirmwareEvent>,
shutdown_ic_send: Sender<ShutdownRpc>,
expected_boot_event: Option<FirmwareEvent>,
ged_send: Option<Sender<get_resources::ged::GuestEmulationRequest>>,
pipette_listener: PolledSocket<UnixListener>,
vtl2_pipette_listener: Option<PolledSocket<UnixListener>>,
openhcl_diag_handler: Option<OpenHclDiagHandler>,
linux_direct_serial_agent: Option<LinuxDirectSerialAgent>,
driver: DefaultDriver,
agent_image: AgentImage,
openhcl_agent_image: Option<AgentImage>,
openvmm_path: ResolvedArtifact,
output_dir: PathBuf,
log_source: PetriLogSource,
vtl2_vsock_path: Option<TempPath>,
_vmbus_vsock_path: TempPath,
}
impl PetriVmConfigOpenVmm {
pub fn os_flavor(&self) -> OsFlavor {
self.firmware.os_flavor()
}
}