// Copyright © 2021 Intel Corporation // // SPDX-License-Identifier: Apache-2.0 use crate::async_io::{AsyncIo, AsyncIoResult, DiskFile, DiskFileError, DiskFileResult}; use crate::vhdx::{Result as VhdxResult, Vhdx}; use crate::AsyncAdaptor; use std::collections::VecDeque; use std::fs::File; use std::sync::{Arc, Mutex, MutexGuard}; 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().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: VecDeque<(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: VecDeque::new(), }) } } impl AsyncAdaptor for Arc> { fn file(&mut self) -> MutexGuard { self.lock().unwrap() } } impl AsyncIo for VhdxSync { fn notifier(&self) -> &EventFd { &self.eventfd } fn read_vectored( &mut self, offset: libc::off_t, iovecs: &[libc::iovec], user_data: u64, ) -> AsyncIoResult<()> { self.vhdx_file.read_vectored_sync( offset, iovecs, user_data, &self.eventfd, &mut self.completion_list, ) } fn write_vectored( &mut self, offset: libc::off_t, iovecs: &[libc::iovec], user_data: u64, ) -> AsyncIoResult<()> { self.vhdx_file.write_vectored_sync( offset, iovecs, user_data, &self.eventfd, &mut self.completion_list, ) } fn fsync(&mut self, user_data: Option) -> AsyncIoResult<()> { self.vhdx_file .fsync_sync(user_data, &self.eventfd, &mut self.completion_list) } fn next_completed_request(&mut self) -> Option<(u64, i32)> { self.completion_list.pop_front() } }