virtiofs/virtio_util.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
use guestmem::GuestMemory;
use std::cmp;
use std::io;
use std::io::Read;
use std::io::Write;
use virtio::VirtioQueueCallbackWork;
use virtio::queue::VirtioQueuePayload;
/// An implementation of `Read` that allows reading data from a virtio payload that may use
/// multiple buffers.
pub struct VirtioPayloadReader<'payload, 'mem> {
guest_memory: &'mem GuestMemory,
payload: std::slice::Iter<'payload, VirtioQueuePayload>,
current: Option<&'payload VirtioQueuePayload>,
offset: usize,
position: usize,
len: usize,
}
impl<'payload, 'mem> VirtioPayloadReader<'payload, 'mem> {
/// Create a new reader for the specified payload.
pub fn new(guest_memory: &'mem GuestMemory, work: &'payload VirtioQueueCallbackWork) -> Self {
let mut reader = Self {
guest_memory,
payload: work.payload.iter(),
current: None,
offset: 0,
position: 0,
len: work.get_payload_length(false) as usize,
};
reader.next_payload();
reader
}
/// Skip ahead to the next readable payload.
fn next_payload(&mut self) {
// It would be nice to use filter instead when assigning to self.payload in the ctor, but
// I can't figure out how to store the result of that (the type it returns is generic over
// the callback type, and I don't want to use Box<dyn Iterator> if I can avoid it).
self.current = self.payload.find(|p| !p.writeable);
}
/// Gets the current payload, or None if the end was reached.
///
/// This function takes care of moving to the next payload buffer if the current offset matches
/// the end.
fn get_payload(&mut self) -> Option<&'payload VirtioQueuePayload> {
let payload = self.current?;
if self.offset == payload.length as usize {
self.next_payload();
self.offset = 0;
}
self.current
}
/// Gets the remaining length of the current payload buffer only.
fn get_current_remaining_len(&mut self) -> usize {
if let Some(payload) = self.get_payload() {
payload.length as usize - self.offset
} else {
0
}
}
}
impl Read for VirtioPayloadReader<'_, '_> {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
if let Some(payload) = self.get_payload() {
// Determine how much space is left in the current buffer, and read at most that much.
// A single call to read won't cross payload buffers, so to read more you must call
// it repeatedly or e.g. use read_exact.
let remaining = payload.length as usize - self.offset;
let size = cmp::min(remaining, buf.len());
self.guest_memory
.read_at(payload.address + self.offset as u64, &mut buf[..size])
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
self.offset += size;
self.position += size;
Ok(size)
} else {
Ok(0)
}
}
}
impl fuse::RequestReader for VirtioPayloadReader<'_, '_> {
fn read_until(&mut self, byte: u8) -> lx::Result<Vec<u8>> {
// Unlike with a simple slice, we can't just scan ahead for the desired byte. Instead,
// repeatedly read some data and see if it contains the byte. It is expected that in most
// cases, the string will be contained in the remainder of the current payload buffer,
// in which case this loop only has one iteration.
let mut buffer: Vec<u8> = Vec::new();
let mut buffer_offset = 0;
loop {
// Read the rest of the current payload buffer.
let len = self.get_current_remaining_len();
if len == 0 {
return Err(lx::Error::EINVAL);
}
buffer.resize_with(buffer.len() + len, Default::default);
let start_offset = self.offset;
assert!(self.read(&mut buffer[buffer_offset..])? == len);
// Search for a matching byte in the portion of the buffer we just read.
if let Some(length) = buffer[buffer_offset..].iter().position(|&c| c == byte) {
// Return up to the matching byte.
buffer.truncate(buffer_offset + length);
// Rewind the offset to be just after the matching byte.
self.offset = start_offset + length + 1;
return Ok(buffer);
} else {
buffer_offset += len;
}
}
}
fn remaining_len(&self) -> usize {
self.len - self.position
}
}
/// An implementation of `Write` that allows writing data to virtio payload that may use multiple
/// buffers.
pub struct VirtioPayloadWriter<'payload, 'mem> {
guest_memory: &'mem GuestMemory,
payload: std::slice::Iter<'payload, VirtioQueuePayload>,
current: Option<&'payload VirtioQueuePayload>,
offset: usize,
}
impl<'payload, 'mem> VirtioPayloadWriter<'payload, 'mem> {
/// Create a new writer for the specified work.
pub fn new(guest_memory: &'mem GuestMemory, work: &'payload VirtioQueueCallbackWork) -> Self {
let mut writer = Self {
guest_memory,
payload: work.payload.iter(),
current: None,
offset: 0,
};
writer.next_payload();
writer
}
/// Skip ahead to the next writable payload buffer.
fn next_payload(&mut self) {
self.current = self.payload.find(|p| p.writeable);
}
/// Gets the current payload, or None if the end was reached.
///
/// This function takes care of moving to the next payload buffer if the current offset matches
/// the end.
fn get_payload(&mut self) -> Option<&'payload VirtioQueuePayload> {
let payload = self.current?;
if self.offset == payload.length as usize {
self.next_payload();
self.offset = 0;
}
self.current
}
}
impl Write for VirtioPayloadWriter<'_, '_> {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
if let Some(payload) = self.get_payload() {
// Find the remaining size of the current payload buffer, and write at most that much.
// This method never writes data spanning multiple buffers, so to write more you must
// call it repeatedly or use write_all.
let remaining = payload.length as usize - self.offset;
let size = cmp::min(remaining, buf.len());
self.guest_memory
.write_at(payload.address + self.offset as u64, &buf[..size])
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
self.offset += size;
Ok(size)
} else {
Ok(0)
}
}
fn flush(&mut self) -> io::Result<()> {
Ok(())
}
}