mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-05 12:25:19 +00:00
vm-virtio: Simplify virtio-console input processing
The existing code was a bit too complex and it was introducing a bug when trying to paste long lines directly to the console. By simplifying the code, and by doing proper usage of the drain() function, the bug is fixed by this commit. Here is the similar output one could have gotten from time to time, when pasting important amounts of bytes: ERROR:vm-virtio/src/console.rs:104 -- Failed to write slice: InvalidGuestAddress(GuestAddress(1040617472)) Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
This commit is contained in:
parent
ad223f2b50
commit
7a3e6caca4
@ -79,37 +79,27 @@ impl ConsoleEpollHandler {
|
|||||||
*/
|
*/
|
||||||
fn process_input_queue(&mut self) -> bool {
|
fn process_input_queue(&mut self) -> bool {
|
||||||
let mut in_buffer = self.in_buffer.lock().unwrap();
|
let mut in_buffer = self.in_buffer.lock().unwrap();
|
||||||
let count = in_buffer.len();
|
|
||||||
let recv_queue = &mut self.queues[0]; //receiveq
|
let recv_queue = &mut self.queues[0]; //receiveq
|
||||||
let mut used_desc_heads = [(0, 0); QUEUE_SIZE as usize];
|
let mut used_desc_heads = [(0, 0); QUEUE_SIZE as usize];
|
||||||
let mut used_count = 0;
|
let mut used_count = 0;
|
||||||
let mut write_count = 0;
|
|
||||||
|
if in_buffer.is_empty() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
let mem = self.mem.load();
|
let mem = self.mem.load();
|
||||||
for avail_desc in recv_queue.iter(&mem) {
|
for avail_desc in recv_queue.iter(&mem) {
|
||||||
let len;
|
let len = cmp::min(avail_desc.len as u32, in_buffer.len() as u32);
|
||||||
|
let source_slice = in_buffer.drain(..len as usize).collect::<Vec<u8>>();
|
||||||
let limit = cmp::min(write_count + avail_desc.len as u32, count as u32);
|
if let Err(e) = mem.write_slice(&source_slice[..], avail_desc.addr) {
|
||||||
let source_slice = in_buffer
|
error!("Failed to write slice: {:?}", e);
|
||||||
.drain(write_count as usize..limit as usize)
|
break;
|
||||||
.collect::<Vec<u8>>();
|
|
||||||
let write_result = mem.write_slice(&source_slice[..], avail_desc.addr);
|
|
||||||
|
|
||||||
match write_result {
|
|
||||||
Ok(_) => {
|
|
||||||
len = limit - write_count; //avail_desc.len;
|
|
||||||
write_count = limit;
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
error!("Failed to write slice: {:?}", e);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
used_desc_heads[used_count] = (avail_desc.index, len);
|
used_desc_heads[used_count] = (avail_desc.index, len);
|
||||||
used_count += 1;
|
used_count += 1;
|
||||||
|
|
||||||
if write_count >= count as u32 {
|
if in_buffer.is_empty() {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -117,6 +107,7 @@ impl ConsoleEpollHandler {
|
|||||||
for &(desc_index, len) in &used_desc_heads[..used_count] {
|
for &(desc_index, len) in &used_desc_heads[..used_count] {
|
||||||
recv_queue.add_used(&mem, desc_index, len);
|
recv_queue.add_used(&mem, desc_index, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
used_count > 0
|
used_count > 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user