#![expect(missing_docs)]
#![forbid(unsafe_code)]
use guid::Guid;
use inspect::Inspect;
use mesh::MeshPayload;
use serde::Serialize;
mod errors;
pub mod schema;
const IDE_NUM_CHANNELS: u8 = 2;
const IDE_MAX_DRIVES_PER_CHANNEL: u8 = 2;
const SCSI_CONTROLLER_NUM: usize = 4;
pub const SCSI_LUN_NUM: usize = 64;
#[derive(Debug, Copy, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub enum DeviceType {
NVMe,
VScsi,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct PhysicalDevice {
pub device_type: DeviceType,
pub vmbus_instance_id: Guid,
pub sub_device_path: u32,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
#[inspect(external_tag)]
pub enum PhysicalDevices {
EmptyDrive,
Single {
device: PhysicalDevice,
},
Striped {
#[inspect(iter_by_index)]
devices: Vec<PhysicalDevice>,
chunk_size_in_kb: u32,
},
}
impl PhysicalDevices {
pub fn is_striping(&self) -> bool {
matches!(self, PhysicalDevices::Striped { .. })
}
pub fn is_empty(&self) -> bool {
matches!(self, PhysicalDevices::EmptyDrive)
}
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub enum GuestMediaType {
Hdd,
DvdLoaded,
DvdUnloaded,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct DiskParameters {
pub device_id: String,
pub vendor_id: String,
pub product_id: String,
pub product_revision_level: String,
pub serial_number: String,
pub model_number: String,
pub medium_rotation_rate: u16,
pub physical_sector_size: Option<u32>,
pub fua: Option<bool>,
pub write_cache: Option<bool>,
pub scsi_disk_size_in_bytes: Option<u64>,
pub odx: Option<bool>,
pub unmap: Option<bool>,
pub max_transfer_length: Option<usize>,
}
#[derive(Debug)]
pub enum StorageDisk {
Ide(IdeDisk),
Scsi(ScsiDisk),
}
impl StorageDisk {
pub fn physical_devices(&self) -> &PhysicalDevices {
match self {
StorageDisk::Ide(ide_disk) => &ide_disk.physical_devices,
StorageDisk::Scsi(scsi_disk) => &scsi_disk.physical_devices,
}
}
pub fn is_dvd(&self) -> bool {
match self {
StorageDisk::Ide(ide_disk) => ide_disk.is_dvd,
StorageDisk::Scsi(scsi_disk) => scsi_disk.is_dvd,
}
}
pub fn ntfs_guid(&self) -> Option<Guid> {
match self {
StorageDisk::Ide(ide_disk) => ide_disk.ntfs_guid,
StorageDisk::Scsi(scsi_disk) => scsi_disk.ntfs_guid,
}
}
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct IdeDisk {
pub channel: u8,
pub location: u8,
pub disk_params: DiskParameters,
pub physical_devices: PhysicalDevices,
pub ntfs_guid: Option<Guid>,
pub is_dvd: bool,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct IdeController {
pub instance_id: Guid,
#[inspect(iter_by_index)]
pub disks: Vec<IdeDisk>,
pub io_queue_depth: Option<u32>,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct ScsiDisk {
pub location: u8,
pub disk_params: DiskParameters,
pub physical_devices: PhysicalDevices,
pub ntfs_guid: Option<Guid>,
pub is_dvd: bool,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct ScsiController {
pub instance_id: Guid,
#[inspect(iter_by_index)]
pub disks: Vec<ScsiDisk>,
pub io_queue_depth: Option<u32>,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct NvmeNamespace {
pub nsid: u32,
pub disk_params: DiskParameters,
pub physical_devices: PhysicalDevices,
}
#[derive(Debug, Clone, Eq, PartialEq, MeshPayload, Inspect)]
pub struct NvmeController {
pub instance_id: Guid,
#[inspect(iter_by_index)]
pub namespaces: Vec<NvmeNamespace>,
}
#[derive(Debug, Clone, MeshPayload, Inspect)]
pub struct NicDevice {
pub instance_id: Guid,
pub subordinate_instance_id: Option<Guid>,
pub max_sub_channels: Option<u16>,
}
#[derive(Debug, Clone, MeshPayload, Inspect)]
pub struct Vtl2SettingsFixed {
pub scsi_sub_channels: u16,
pub io_ring_size: u32,
pub max_bounce_buffer_pages: Option<u32>,
}
#[derive(Debug, Clone, MeshPayload, Inspect)]
pub struct Vtl2SettingsDynamic {
pub ide_controller: Option<IdeController>,
#[inspect(iter_by_index)]
pub scsi_controllers: Vec<ScsiController>,
#[inspect(iter_by_index)]
pub nic_devices: Vec<NicDevice>,
#[inspect(iter_by_index)]
pub nvme_controllers: Vec<NvmeController>,
}
#[derive(Debug, Default, Clone, MeshPayload, Inspect)]
pub struct Vtl2Settings {
pub fixed: Vtl2SettingsFixed,
pub dynamic: Vtl2SettingsDynamic,
}
enum Component {
Underhill,
Storage,
Network,
}
enum EscalationCategory {
Underhill,
VmCreation,
Configuration,
}
macro_rules! error_codes {
{
$(#[$enum_attr:meta])*
$vis:vis enum $enum_name:ident {
$(
$(#[$attr:meta])*
$name:ident => ($component:tt, $category:tt),
)*
}
} => {
$(#[$enum_attr])*
$vis enum $enum_name {
$(
$(#[$attr])*
$name,
)*
}
impl $enum_name {
fn name(&self) -> &'static str {
match self {
$(
$enum_name::$name => {
let _ = Component::$component;
let _ = EscalationCategory::$category;
concat!(stringify!($category), ".", stringify!($name))
}
)*
}
}
}
};
}
error_codes! {
#[derive(Clone, Copy, Debug, PartialEq)]
pub enum Vtl2SettingsErrorCode {
InternalFailure => (Underhill, Underhill),
JsonFormatError => (Underhill, Configuration),
NoVmbusServer => (Underhill, VmCreation),
UnsupportedSchemaVersion => (Underhill, Configuration),
InvalidInstanceId => (Underhill, Configuration),
ProtobufFormatError => (Underhill, Configuration),
UnsupportedSchemaNamespace => (Underhill, Configuration),
EmptyNamespaceChunk => (Underhill, Configuration),
StorageCannotAddRemoveControllerAtRuntime => (Storage, Configuration),
StorageLunLocationExceedsMaxLimits => (Storage, Configuration),
StorageLunLocationDuplicated => (Storage, Configuration),
StorageUnsupportedDeviceType => (Storage, Configuration),
StorageCannotFindVtl2Device => (Storage, Configuration),
EmptyDriveNotAllowed => (Storage, Configuration),
StorageCannotOpenVtl2Device => (Storage, Underhill),
StorageScsiControllerNotFound => (Storage, Underhill),
StorageAttachDiskFailed => (Storage, Underhill),
StorageRmDiskFailed => (Storage, Underhill),
StorageControllerGuidAlreadyExists => (Storage, Configuration),
StorageScsiControllerExceedsMaxLimits => (Storage, Configuration),
StorageInvalidVendorId => (Storage, Configuration),
StorageInvalidProductId => (Storage, Configuration),
StorageInvalidProductRevisionLevel => (Storage, Configuration),
StorageIdeChannelNotProvided => (Storage, Configuration),
StorageIdeChannelExceedsMaxLimits => (Storage, Configuration),
StorageIdeLocationExceedsMaxLimits => (Storage, Configuration),
StorageIdeChannelInvalidConfiguration => (Storage, Configuration),
StripedStorageCannotChangeControllerAtRuntime => (Storage, Configuration),
StorageInvalidPhysicalDiskCount => (Storage, Configuration),
StorageCannotModifyIdeAtRuntime => (Storage, Configuration),
StorageInvalidControllerType => (Storage, Configuration),
StorageInvalidDeviceId => (Storage, Configuration),
StorageChangeMediaFailed => (Storage, Underhill),
StorageInvalidNtfsFormatGuid => (Storage, Configuration),
NetworkingModifyNicFailed => (Network, Configuration),
NetworkingAddNicFailed => (Network, Configuration),
NetworkingRemoveNicFailed => (Network, Configuration),
}
}
impl Serialize for Vtl2SettingsErrorCode {
fn serialize<S: serde::Serializer>(&self, ser: S) -> Result<S::Ok, S::Error> {
ser.serialize_str(self.name())
}
}
#[derive(Debug, Serialize)]
pub struct Vtl2SettingsErrorInfo {
error_id: Vtl2SettingsErrorCode,
message: String,
file_name: &'static str,
line: u32,
}
impl Vtl2SettingsErrorInfo {
#[track_caller]
pub fn new(code: Vtl2SettingsErrorCode, message: String) -> Self {
let caller = std::panic::Location::caller();
Vtl2SettingsErrorInfo {
error_id: code,
message,
file_name: caller.file(),
line: caller.line(),
}
}
pub fn code(&self) -> Vtl2SettingsErrorCode {
self.error_id
}
}
impl std::fmt::Display for Vtl2SettingsErrorInfo {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let json = serde_json::to_string(self).map_err(|_| std::fmt::Error)?;
write!(f, "{}", json)
}
}
impl std::error::Error for Vtl2SettingsErrorInfo {}
#[derive(Debug)]
pub struct Vtl2SettingsErrorInfoVec {
pub errors: Vec<Vtl2SettingsErrorInfo>,
}
impl std::fmt::Display for Vtl2SettingsErrorInfoVec {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
for e in &self.errors {
writeln!(f, "{}", e)?;
}
Ok(())
}
}
impl std::error::Error for Vtl2SettingsErrorInfoVec {}