#![expect(missing_docs)]
#![cfg(windows)]
#![expect(unsafe_code)]
#![expect(clippy::undocumented_unsafe_blocks)]
use disk_backend::DiskError;
use disk_backend::DiskIo;
use disk_backend::resolve::ResolveDiskParameters;
use disk_backend::resolve::ResolvedDisk;
use disk_file::FileDisk;
use guid::Guid;
use inspect::Inspect;
use mesh::MeshPayload;
use scsi_buffers::RequestBuffers;
use std::fs;
use std::os::windows::prelude::*;
use std::path::Path;
use thiserror::Error;
use vm_resource::ResolveResource;
use vm_resource::ResourceId;
use vm_resource::declare_static_resolver;
use vm_resource::kind::DiskHandleKind;
mod virtdisk {
#![allow(
non_snake_case,
dead_code,
non_camel_case_types,
clippy::upper_case_acronyms
)]
use std::os::windows::prelude::*;
use winapi::shared::guiddef::GUID;
use winapi::shared::minwindef::BOOL;
use winapi::um::minwinbase::OVERLAPPED;
use winapi::um::winnt::SECURITY_DESCRIPTOR;
#[repr(C)]
#[derive(Copy, Clone)]
pub struct VIRTUAL_STORAGE_TYPE {
pub DeviceId: u32,
pub VendorId: GUID,
}
pub const OPEN_VIRTUAL_DISK_FLAG_NO_PARENTS: u32 = 0x0000_0001;
pub const OPEN_VIRTUAL_DISK_FLAG_BLANK_FILE: u32 = 0x0000_0002;
pub const OPEN_VIRTUAL_DISK_FLAG_BOOT_DRIVE: u32 = 0x0000_0004;
pub const OPEN_VIRTUAL_DISK_FLAG_CACHED_IO: u32 = 0x0000_0008;
pub const OPEN_VIRTUAL_DISK_FLAG_CUSTOM_DIFF_CHAIN: u32 = 0x0000_0010;
pub const OPEN_VIRTUAL_DISK_FLAG_PARENT_CACHED_IO: u32 = 0x0000_0020;
pub const OPEN_VIRTUAL_DISK_FLAG_VHDSET_FILE_ONLY: u32 = 0x0000_0040;
pub const OPEN_VIRTUAL_DISK_FLAG_IGNORE_RELATIVE_PARENT_LOCATOR: u32 = 0x0000_0080;
pub const OPEN_VIRTUAL_DISK_FLAG_NO_WRITE_HARDENING: u32 = 0x0000_0100;
#[repr(C)]
pub struct OPEN_VIRTUAL_DISK_PARAMETERS {
pub Version: u32,
pub u: OPEN_VIRTUAL_DISK_PARAMETERS_u,
}
#[repr(C)]
pub union OPEN_VIRTUAL_DISK_PARAMETERS_u {
pub Version1: OPEN_VIRTUAL_DISK_PARAMETERS_1,
pub Version2: OPEN_VIRTUAL_DISK_PARAMETERS_2,
pub Version3: OPEN_VIRTUAL_DISK_PARAMETERS_3,
}
#[repr(C)]
#[derive(Copy, Clone)]
pub struct OPEN_VIRTUAL_DISK_PARAMETERS_1 {
pub RWDepth: u32,
}
#[repr(C)]
#[derive(Copy, Clone)]
pub struct OPEN_VIRTUAL_DISK_PARAMETERS_2 {
pub GetInfoOnly: BOOL,
pub ReadOnly: BOOL,
pub ResiliencyGuid: GUID,
}
#[repr(C)]
#[derive(Copy, Clone)]
pub struct OPEN_VIRTUAL_DISK_PARAMETERS_3 {
pub GetInfoOnly: BOOL,
pub ReadOnly: BOOL,
pub ResiliencyGuid: GUID,
pub SnapshotId: GUID,
}
pub const VIRTUAL_DISK_ACCESS_ATTACH_RO: u32 = 0x00010000;
pub const VIRTUAL_DISK_ACCESS_ATTACH_RW: u32 = 0x00020000;
pub const VIRTUAL_DISK_ACCESS_DETACH: u32 = 0x00040000;
pub const VIRTUAL_DISK_ACCESS_GET_INFO: u32 = 0x00080000;
pub const VIRTUAL_DISK_ACCESS_CREATE: u32 = 0x00100000;
pub const VIRTUAL_DISK_ACCESS_METAOPS: u32 = 0x00200000;
pub const VIRTUAL_DISK_ACCESS_READ: u32 = 0x000d0000;
pub const ATTACH_VIRTUAL_DISK_FLAG_READ_ONLY: u32 = 0x0000_0001;
pub const ATTACH_VIRTUAL_DISK_FLAG_NO_DRIVE_LETTER: u32 = 0x0000_0002;
pub const ATTACH_VIRTUAL_DISK_FLAG_PERMANENT_LIFETIME: u32 = 0x0000_0004;
pub const ATTACH_VIRTUAL_DISK_FLAG_NO_LOCAL_HOST: u32 = 0x0000_0008;
pub const ATTACH_VIRTUAL_DISK_FLAG_NO_SECURITY_DESCRIPTOR: u32 = 0x0000_0010;
pub const ATTACH_VIRTUAL_DISK_FLAG_BYPASS_DEFAULT_ENCRYPTION_POLICY: u32 = 0x0000_0020;
pub const GET_VIRTUAL_DISK_INFO_UNSPECIFIED: u32 = 0;
pub const GET_VIRTUAL_DISK_INFO_SIZE: u32 = 1;
pub const GET_VIRTUAL_DISK_INFO_IDENTIFIER: u32 = 2;
pub const GET_VIRTUAL_DISK_INFO_PARENT_LOCATION: u32 = 3;
pub const GET_VIRTUAL_DISK_INFO_PARENT_IDENTIFIER: u32 = 4;
pub const GET_VIRTUAL_DISK_INFO_PARENT_TIMESTAMP: u32 = 5;
pub const GET_VIRTUAL_DISK_INFO_VIRTUAL_STORAGE_TYPE: u32 = 6;
pub const GET_VIRTUAL_DISK_INFO_PROVIDER_SUBTYPE: u32 = 7;
pub const GET_VIRTUAL_DISK_INFO_IS_4K_ALIGNED: u32 = 8;
pub const GET_VIRTUAL_DISK_INFO_PHYSICAL_DISK: u32 = 9;
pub const GET_VIRTUAL_DISK_INFO_VHD_PHYSICAL_SECTOR_SIZE: u32 = 10;
pub const GET_VIRTUAL_DISK_INFO_SMALLEST_SAFE_VIRTUAL_SIZE: u32 = 11;
pub const GET_VIRTUAL_DISK_INFO_FRAGMENTATION: u32 = 12;
pub const GET_VIRTUAL_DISK_INFO_IS_LOADED: u32 = 13;
pub const GET_VIRTUAL_DISK_INFO_VIRTUAL_DISK_ID: u32 = 14;
pub const GET_VIRTUAL_DISK_INFO_CHANGE_TRACKING_STATE: u32 = 15;
#[repr(C)]
#[derive(Copy, Clone)]
pub struct GET_VIRTUAL_DISK_INFO {
pub Version: u32,
pub u: GET_VIRTUAL_DISK_INFO_u,
}
#[repr(C)]
#[derive(Copy, Clone)]
pub union GET_VIRTUAL_DISK_INFO_u {
pub Size: GET_VIRTUAL_DISK_INFO_Size,
pub Identifier: GUID,
pub ParentIdentifier: GUID,
pub ParentTimestamp: u32,
pub VirtualStorageType: VIRTUAL_STORAGE_TYPE,
pub ProviderSubtype: u32,
pub Is4kAligned: BOOL,
pub IsLoaded: BOOL,
pub VhdPhysicalSectorSize: u32,
pub SmallestSafeVirtualSize: u64,
pub FragmentationPercentage: u32,
pub VirtualDiskId: GUID,
}
#[repr(C)]
#[derive(Copy, Clone)]
pub struct GET_VIRTUAL_DISK_INFO_Size {
pub VirtualSize: u64,
pub PhysicalSize: u64,
pub BlockSize: u32,
pub SectorSize: u32,
}
#[link(name = "virtdisk")]
unsafe extern "system" {
pub fn OpenVirtualDisk(
virtual_storage_type: &mut VIRTUAL_STORAGE_TYPE,
path: *const u16,
virtual_disk_access_mask: u32,
flags: u32,
parameters: Option<&mut OPEN_VIRTUAL_DISK_PARAMETERS>,
handle: &mut RawHandle,
) -> u32;
pub fn AttachVirtualDisk(
virtual_disk_handle: RawHandle,
security_descriptor: Option<&mut SECURITY_DESCRIPTOR>,
flags: u32,
provider_specific_flags: u32,
parameters: usize,
overlapped: Option<&mut OVERLAPPED>,
) -> u32;
pub fn GetVirtualDiskInformation(
virtual_disk_handle: RawHandle,
virtual_disk_info_size: &mut u32,
virtual_disk_info: Option<&mut GET_VIRTUAL_DISK_INFO>,
size_use: Option<&mut u32>,
) -> u32;
}
}
#[derive(Debug, MeshPayload)]
pub struct Vhd(fs::File);
fn chk_win32(err: u32) -> std::io::Result<()> {
if err == 0 {
Ok(())
} else {
Err(std::io::Error::from_raw_os_error(err as i32))
}
}
impl Vhd {
fn open(path: &Path, read_only: bool) -> std::io::Result<Self> {
let file = unsafe {
let mut storage_type = std::mem::zeroed();
let resiliency_guid = Guid::new_random();
let mut parameters = virtdisk::OPEN_VIRTUAL_DISK_PARAMETERS {
Version: 2,
u: virtdisk::OPEN_VIRTUAL_DISK_PARAMETERS_u {
Version2: virtdisk::OPEN_VIRTUAL_DISK_PARAMETERS_2 {
ReadOnly: read_only.into(),
ResiliencyGuid: resiliency_guid.into(),
..std::mem::zeroed()
},
},
};
let mut path16: Vec<_> = path.as_os_str().encode_wide().collect();
path16.push(0);
let mut handle = std::mem::zeroed();
chk_win32(virtdisk::OpenVirtualDisk(
&mut storage_type,
path16.as_ptr(),
0,
0,
Some(&mut parameters),
&mut handle,
))?;
fs::File::from_raw_handle(handle)
};
Ok(Self(file))
}
fn attach_for_raw_access(&self, read_only: bool) -> std::io::Result<()> {
unsafe {
let mut flags = virtdisk::ATTACH_VIRTUAL_DISK_FLAG_NO_LOCAL_HOST;
if read_only {
flags |= virtdisk::ATTACH_VIRTUAL_DISK_FLAG_READ_ONLY;
}
chk_win32(virtdisk::AttachVirtualDisk(
self.0.as_raw_handle(),
None,
flags,
0,
0,
None,
))?;
}
Ok(())
}
fn info_static(&self, info_type: u32) -> std::io::Result<virtdisk::GET_VIRTUAL_DISK_INFO> {
unsafe {
let mut info = virtdisk::GET_VIRTUAL_DISK_INFO {
Version: info_type,
..std::mem::zeroed()
};
let mut size = size_of_val(&info) as u32;
chk_win32(virtdisk::GetVirtualDiskInformation(
self.0.as_raw_handle(),
&mut size,
Some(&mut info),
None,
))?;
Ok(info)
}
}
fn get_size(&self) -> std::io::Result<virtdisk::GET_VIRTUAL_DISK_INFO_Size> {
unsafe {
Ok(self
.info_static(virtdisk::GET_VIRTUAL_DISK_INFO_SIZE)?
.u
.Size)
}
}
fn get_physical_sector_size(&self) -> std::io::Result<u32> {
unsafe {
Ok(self
.info_static(virtdisk::GET_VIRTUAL_DISK_INFO_VHD_PHYSICAL_SECTOR_SIZE)?
.u
.VhdPhysicalSectorSize)
}
}
fn get_disk_id(&self) -> std::io::Result<Guid> {
unsafe {
Ok(self
.info_static(virtdisk::GET_VIRTUAL_DISK_INFO_VIRTUAL_DISK_ID)?
.u
.VirtualDiskId
.into())
}
}
}
#[derive(MeshPayload)]
pub struct OpenVhdmpDiskConfig(pub Vhd);
impl ResourceId<DiskHandleKind> for OpenVhdmpDiskConfig {
const ID: &'static str = "vhdmp";
}
pub struct VhdmpDiskResolver;
declare_static_resolver!(VhdmpDiskResolver, (DiskHandleKind, OpenVhdmpDiskConfig));
#[derive(Debug, Error)]
pub enum ResolveVhdmpDiskError {
#[error("failed to open VHD")]
Vhdmp(#[source] Error),
#[error("invalid disk")]
InvalidDisk(#[source] disk_backend::InvalidDisk),
}
impl ResolveResource<DiskHandleKind, OpenVhdmpDiskConfig> for VhdmpDiskResolver {
type Output = ResolvedDisk;
type Error = ResolveVhdmpDiskError;
fn resolve(
&self,
rsrc: OpenVhdmpDiskConfig,
input: ResolveDiskParameters<'_>,
) -> Result<Self::Output, Self::Error> {
ResolvedDisk::new(
VhdmpDisk::new(rsrc.0, input.read_only).map_err(ResolveVhdmpDiskError::Vhdmp)?,
)
.map_err(ResolveVhdmpDiskError::InvalidDisk)
}
}
#[derive(Debug, Inspect)]
pub struct VhdmpDisk {
#[inspect(flatten)]
vhd: FileDisk,
#[inspect(skip)]
io_lock: futures::lock::Mutex<()>,
disk_id: Guid,
}
#[derive(Debug, Error)]
pub enum Error {
#[error("failed to open VHD")]
Open(#[source] std::io::Error),
#[error("failed to attach VHD")]
Attach(#[source] std::io::Error),
#[error("failed to query VHD metadata")]
Query(#[source] std::io::Error),
}
impl VhdmpDisk {
pub fn open_vhd(path: &Path, read_only: bool) -> Result<Vhd, Error> {
let vhd = Vhd::open(path, read_only).map_err(Error::Open)?;
vhd.attach_for_raw_access(read_only)
.map_err(Error::Attach)?;
Ok(vhd)
}
pub fn new(vhd: Vhd, read_only: bool) -> Result<Self, Error> {
let size = vhd.get_size().map_err(Error::Query)?;
let disk_id = vhd.get_disk_id().map_err(Error::Query)?;
let metadata = disk_file::Metadata {
disk_size: size.VirtualSize,
sector_size: size.SectorSize,
physical_sector_size: vhd.get_physical_sector_size().map_err(Error::Query)?,
read_only,
};
let vhd = FileDisk::with_metadata(vhd.0, metadata);
Ok(Self {
vhd,
io_lock: Default::default(),
disk_id,
})
}
}
impl DiskIo for VhdmpDisk {
fn disk_type(&self) -> &str {
"vhdmp"
}
fn sector_count(&self) -> u64 {
self.vhd.sector_count()
}
fn sector_size(&self) -> u32 {
self.vhd.sector_size()
}
fn is_read_only(&self) -> bool {
self.vhd.is_read_only()
}
fn disk_id(&self) -> Option<[u8; 16]> {
Some(self.disk_id.into())
}
fn physical_sector_size(&self) -> u32 {
self.vhd.physical_sector_size()
}
fn is_fua_respected(&self) -> bool {
self.vhd.is_fua_respected()
}
async fn read_vectored(
&self,
buffers: &RequestBuffers<'_>,
sector: u64,
) -> Result<(), DiskError> {
let _locked = self.io_lock.lock().await;
self.vhd.read(buffers, sector).await
}
async fn write_vectored(
&self,
buffers: &RequestBuffers<'_>,
sector: u64,
fua: bool,
) -> Result<(), DiskError> {
let _locked = self.io_lock.lock().await;
self.vhd.write(buffers, sector, fua).await
}
async fn sync_cache(&self) -> Result<(), DiskError> {
let _locked = self.io_lock.lock().await;
self.vhd.flush().await
}
async fn unmap(
&self,
_sector: u64,
_count: u64,
_block_level_only: bool,
) -> Result<(), DiskError> {
Ok(())
}
fn unmap_behavior(&self) -> disk_backend::UnmapBehavior {
disk_backend::UnmapBehavior::Ignored
}
}
#[cfg(test)]
mod tests {
use super::VhdmpDisk;
use disk_backend::DiskError;
use disk_backend::DiskIo;
use disk_vhd1::Vhd1Disk;
use guestmem::GuestMemory;
use pal_async::async_test;
use scsi_buffers::OwnedRequestBuffers;
use std::io::Write;
use tempfile::TempPath;
fn make_test_vhd() -> TempPath {
let mut f = tempfile::Builder::new().suffix(".vhd").tempfile().unwrap();
let size = 0x300000;
f.write_all(&vec![0u8; size]).unwrap();
Vhd1Disk::make_fixed(f.as_file()).unwrap();
f.into_temp_path()
}
#[test]
fn open_readonly() {
let path = make_test_vhd();
let _vhd = VhdmpDisk::open_vhd(path.as_ref(), true).unwrap();
let _vhd = VhdmpDisk::open_vhd(path.as_ref(), true).unwrap();
let _vhd = VhdmpDisk::open_vhd(path.as_ref(), false).unwrap_err();
}
#[async_test]
async fn test_invalid_lba() {
let path = make_test_vhd();
let vhd = VhdmpDisk::open_vhd(path.as_ref(), true).unwrap();
let disk = VhdmpDisk::new(vhd, true).unwrap();
let gm = GuestMemory::allocate(512);
match disk
.read_vectored(
&OwnedRequestBuffers::linear(0, 512, true).buffer(&gm),
0x10000000,
)
.await
{
Err(DiskError::IllegalBlock) => {}
r => panic!("unexpected result: {:?}", r),
}
}
}