2021-01-20 09:55:39 +00:00
|
|
|
// Copyright © 2021 Intel Corporation
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 AND BSD-3-Clause
|
|
|
|
|
|
|
|
use crate::async_io::{
|
|
|
|
AsyncIo, AsyncIoError, AsyncIoResult, DiskFile, DiskFileError, DiskFileResult,
|
|
|
|
};
|
2021-02-28 17:06:01 +00:00
|
|
|
use io_uring::{opcode, squeue, types, IoUring};
|
2021-01-20 09:55:39 +00:00
|
|
|
use std::fs::File;
|
|
|
|
use std::io::{Seek, SeekFrom};
|
|
|
|
use std::os::unix::io::{AsRawFd, RawFd};
|
|
|
|
use vmm_sys_util::eventfd::EventFd;
|
|
|
|
|
|
|
|
pub struct RawFileDisk {
|
|
|
|
file: File,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl RawFileDisk {
|
|
|
|
pub fn new(file: File) -> Self {
|
|
|
|
RawFileDisk { file }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DiskFile for RawFileDisk {
|
|
|
|
fn size(&mut self) -> DiskFileResult<u64> {
|
|
|
|
Ok(self
|
|
|
|
.file
|
|
|
|
.seek(SeekFrom::End(0))
|
|
|
|
.map_err(DiskFileError::Size)? as u64)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn new_async_io(&self, ring_depth: u32) -> DiskFileResult<Box<dyn AsyncIo>> {
|
|
|
|
Ok(Box::new(
|
|
|
|
RawFileAsync::new(self.file.as_raw_fd(), ring_depth)
|
|
|
|
.map_err(DiskFileError::NewAsyncIo)?,
|
|
|
|
) as Box<dyn AsyncIo>)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct RawFileAsync {
|
|
|
|
fd: RawFd,
|
|
|
|
io_uring: IoUring,
|
|
|
|
eventfd: EventFd,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl RawFileAsync {
|
|
|
|
pub fn new(fd: RawFd, ring_depth: u32) -> std::io::Result<Self> {
|
|
|
|
let io_uring = IoUring::new(ring_depth)?;
|
|
|
|
let eventfd = EventFd::new(libc::EFD_NONBLOCK)?;
|
|
|
|
|
|
|
|
// Register the io_uring eventfd that will notify when something in
|
|
|
|
// the completion queue is ready.
|
|
|
|
io_uring.submitter().register_eventfd(eventfd.as_raw_fd())?;
|
|
|
|
|
|
|
|
Ok(RawFileAsync {
|
|
|
|
fd,
|
|
|
|
io_uring,
|
|
|
|
eventfd,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsyncIo for RawFileAsync {
|
|
|
|
fn notifier(&self) -> &EventFd {
|
|
|
|
&self.eventfd
|
|
|
|
}
|
|
|
|
|
|
|
|
fn read_vectored(
|
|
|
|
&mut self,
|
|
|
|
offset: libc::off_t,
|
|
|
|
iovecs: Vec<libc::iovec>,
|
|
|
|
user_data: u64,
|
|
|
|
) -> AsyncIoResult<()> {
|
2021-02-28 17:06:01 +00:00
|
|
|
let (submitter, mut sq, _) = self.io_uring.split();
|
2021-01-20 09:55:39 +00:00
|
|
|
|
|
|
|
// Safe because we know the file descriptor is valid and we
|
|
|
|
// relied on vm-memory to provide the buffer address.
|
|
|
|
let _ = unsafe {
|
2021-02-28 17:06:01 +00:00
|
|
|
sq.push(
|
|
|
|
&opcode::Readv::new(types::Fd(self.fd), iovecs.as_ptr(), iovecs.len() as u32)
|
|
|
|
.offset(offset)
|
|
|
|
.build()
|
|
|
|
.flags(squeue::Flags::ASYNC)
|
|
|
|
.user_data(user_data),
|
2021-01-20 09:55:39 +00:00
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
// Update the submission queue and submit new operations to the
|
|
|
|
// io_uring instance.
|
2021-02-28 17:06:01 +00:00
|
|
|
sq.sync();
|
2021-01-20 09:55:39 +00:00
|
|
|
submitter.submit().map_err(AsyncIoError::ReadVectored)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn write_vectored(
|
|
|
|
&mut self,
|
|
|
|
offset: libc::off_t,
|
|
|
|
iovecs: Vec<libc::iovec>,
|
|
|
|
user_data: u64,
|
|
|
|
) -> AsyncIoResult<()> {
|
2021-02-28 17:06:01 +00:00
|
|
|
let (submitter, mut sq, _) = self.io_uring.split();
|
2021-01-20 09:55:39 +00:00
|
|
|
|
|
|
|
// Safe because we know the file descriptor is valid and we
|
|
|
|
// relied on vm-memory to provide the buffer address.
|
|
|
|
let _ = unsafe {
|
2021-02-28 17:06:01 +00:00
|
|
|
sq.push(
|
|
|
|
&opcode::Writev::new(types::Fd(self.fd), iovecs.as_ptr(), iovecs.len() as u32)
|
|
|
|
.offset(offset)
|
|
|
|
.build()
|
|
|
|
.flags(squeue::Flags::ASYNC)
|
|
|
|
.user_data(user_data),
|
2021-01-20 09:55:39 +00:00
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
// Update the submission queue and submit new operations to the
|
|
|
|
// io_uring instance.
|
2021-02-28 17:06:01 +00:00
|
|
|
sq.sync();
|
2021-01-20 09:55:39 +00:00
|
|
|
submitter.submit().map_err(AsyncIoError::WriteVectored)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fsync(&mut self, user_data: Option<u64>) -> AsyncIoResult<()> {
|
|
|
|
if let Some(user_data) = user_data {
|
2021-02-28 17:06:01 +00:00
|
|
|
let (submitter, mut sq, _) = self.io_uring.split();
|
2021-01-20 09:55:39 +00:00
|
|
|
|
|
|
|
// Safe because we know the file descriptor is valid.
|
|
|
|
let _ = unsafe {
|
2021-02-28 17:06:01 +00:00
|
|
|
sq.push(
|
|
|
|
&opcode::Fsync::new(types::Fd(self.fd))
|
2021-01-20 09:55:39 +00:00
|
|
|
.build()
|
|
|
|
.flags(squeue::Flags::ASYNC)
|
|
|
|
.user_data(user_data),
|
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
// Update the submission queue and submit new operations to the
|
|
|
|
// io_uring instance.
|
2021-02-28 17:06:01 +00:00
|
|
|
sq.sync();
|
2021-01-20 09:55:39 +00:00
|
|
|
submitter.submit().map_err(AsyncIoError::Fsync)?;
|
|
|
|
} else {
|
|
|
|
unsafe { libc::fsync(self.fd) };
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn complete(&mut self) -> Vec<(u64, i32)> {
|
|
|
|
let mut completion_list = Vec::new();
|
|
|
|
|
|
|
|
let cq = self.io_uring.completion();
|
2021-02-28 17:06:01 +00:00
|
|
|
for cq_entry in cq {
|
2021-01-20 09:55:39 +00:00
|
|
|
completion_list.push((cq_entry.user_data(), cq_entry.result()));
|
|
|
|
}
|
|
|
|
|
|
|
|
completion_list
|
|
|
|
}
|
|
|
|
}
|