// Copyright © 2021 Intel Corporation // // SPDX-License-Identifier: Apache-2.0 use crate::async_io::{AsyncIo, AsyncIoResult, DiskFile, DiskFileError, DiskFileResult}; use crate::{fsync_sync, read_vectored_sync, write_vectored_sync, ReadWriteSeekFile}; use std::fs::File; use std::sync::{Arc, Mutex}; use vhdx::vhdx::{Result as VhdxResult, Vhdx}; use vmm_sys_util::eventfd::EventFd; pub struct VhdxDiskSync { vhdx_file: Arc>, } impl VhdxDiskSync { pub fn new(f: File) -> VhdxResult { Ok(VhdxDiskSync { vhdx_file: Arc::new(Mutex::new(Vhdx::new(f)?)), }) } } impl DiskFile for VhdxDiskSync { fn size(&mut self) -> DiskFileResult { Ok(self .vhdx_file .lock() .unwrap() .as_any() .downcast_ref::() .unwrap() .virtual_disk_size()) } fn new_async_io(&self, _ring_depth: u32) -> DiskFileResult> { Ok( Box::new(VhdxSync::new(self.vhdx_file.clone()).map_err(DiskFileError::NewAsyncIo)?) as Box, ) } } pub struct VhdxSync { vhdx_file: Arc>, eventfd: EventFd, completion_list: Vec<(u64, i32)>, } impl VhdxSync { pub fn new( vhdx_file: Arc>, ) -> std::io::Result { Ok(VhdxSync { vhdx_file, eventfd: EventFd::new(libc::EFD_NONBLOCK)?, completion_list: Vec::new(), }) } } impl AsyncIo for VhdxSync { fn notifier(&self) -> &EventFd { &self.eventfd } fn read_vectored( &mut self, offset: libc::off_t, iovecs: Vec, user_data: u64, ) -> AsyncIoResult<()> { read_vectored_sync( offset, iovecs, user_data, &mut self.vhdx_file, &self.eventfd, &mut self.completion_list, ) } fn write_vectored( &mut self, offset: libc::off_t, iovecs: Vec, user_data: u64, ) -> AsyncIoResult<()> { write_vectored_sync( offset, iovecs, user_data, &mut self.vhdx_file, &self.eventfd, &mut self.completion_list, ) } fn fsync(&mut self, user_data: Option) -> AsyncIoResult<()> { fsync_sync( user_data, &mut self.vhdx_file, &self.eventfd, &mut self.completion_list, ) } fn complete(&mut self) -> Vec<(u64, i32)> { self.completion_list.drain(..).collect() } }