vm-virtio: Add support for notifying about virtio config update

As per the VIRTIO specification, every virtio device configuration can
be updated while the guest is running. The guest needs to be notified
when this happens, and it can be done in two different ways, depending
on the type of interrupt being used for those devices.

In case the device uses INTx, the allocated IRQ pin is shared between
queues and configuration updates. The way for the guest to differentiate
between an interrupt meant for a virtqueue or meant for a configuration
update is tied to the value of the ISR status field. This field is a
simple 32 bits bitmask where only bit 0 and 1 can be changed, the rest
is reserved.

In case the device uses MSI/MSI-X, the driver should allocate a
dedicated vector for configuration updates. This case is much simpler as
it only requires the device to send the appropriate MSI vector.

The cloud-hypervisor codebase was not supporting the update of a virtio
device configuration. This patch extends the existing VirtioInterrupt
closure to accept a type that can be Config or Queue, so that based on
this type, the closure implementation can make the right choice about
which interrupt pin or vector to trigger.

Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
This commit is contained in:
Sebastien Boeuf 2019-07-26 11:48:07 -07:00 committed by Rob Bradford
parent 93b77530c7
commit 98d7955e34
9 changed files with 93 additions and 95 deletions

View File

@ -17,14 +17,13 @@ use std::os::linux::fs::MetadataExt;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::path::PathBuf; use std::path::PathBuf;
use std::result; use std::result;
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc; use std::sync::Arc;
use std::thread; use std::thread;
use super::Error as DeviceError; use super::Error as DeviceError;
use super::{ use super::{
ActivateError, ActivateResult, DescriptorChain, DeviceEventT, Queue, VirtioDevice, ActivateError, ActivateResult, DescriptorChain, DeviceEventT, Queue, VirtioDevice,
VirtioDeviceType, INTERRUPT_STATUS_USED_RING, VirtioDeviceType, VirtioInterruptType,
}; };
use crate::VirtioInterrupt; use crate::VirtioInterrupt;
use virtio_bindings::virtio_blk::*; use virtio_bindings::virtio_blk::*;
@ -325,7 +324,6 @@ struct BlockEpollHandler<T: DiskFile> {
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
disk_image: T, disk_image: T,
disk_nsectors: u64, disk_nsectors: u64,
interrupt_status: Arc<AtomicUsize>,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
disk_image_id: Vec<u8>, disk_image_id: Vec<u8>,
} }
@ -376,12 +374,12 @@ impl<T: DiskFile> BlockEpollHandler<T> {
} }
fn signal_used_queue(&self, queue_index: usize) -> result::Result<(), DeviceError> { fn signal_used_queue(&self, queue_index: usize) -> result::Result<(), DeviceError> {
self.interrupt_status (self.interrupt_cb)(&VirtioInterruptType::Queue, Some(&self.queues[queue_index])).map_err(
.fetch_or(INTERRUPT_STATUS_USED_RING as usize, Ordering::SeqCst); |e| {
(self.interrupt_cb)(&self.queues[queue_index]).map_err(|e| { error!("Failed to signal used queue: {:?}", e);
error!("Failed to signal used queue: {:?}", e); DeviceError::FailedSignalingUsedQueue(e)
DeviceError::FailedSignalingUsedQueue(e) },
}) )
} }
#[allow(dead_code)] #[allow(dead_code)]
@ -600,7 +598,6 @@ impl<T: 'static + DiskFile + Send> VirtioDevice for Block<T> {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
queues: Vec<Queue>, queues: Vec<Queue>,
mut queue_evts: Vec<EventFd>, mut queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -644,7 +641,6 @@ impl<T: 'static + DiskFile + Send> VirtioDevice for Block<T> {
mem, mem,
disk_image, disk_image,
disk_nsectors: self.disk_nsectors, disk_nsectors: self.disk_nsectors,
interrupt_status: status,
interrupt_cb, interrupt_cb,
disk_image_id, disk_image_id,
}; };

View File

@ -10,14 +10,13 @@ use std::io;
use std::io::Write; use std::io::Write;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::result; use std::result;
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::{Arc, Mutex}; use std::sync::{Arc, Mutex};
use std::thread; use std::thread;
use super::Error as DeviceError; use super::Error as DeviceError;
use super::{ use super::{
ActivateError, ActivateResult, DeviceEventT, Queue, VirtioDevice, VirtioDeviceType, ActivateError, ActivateResult, DeviceEventT, Queue, VirtioDevice, VirtioDeviceType,
INTERRUPT_STATUS_USED_RING, VIRTIO_F_VERSION_1, VirtioInterruptType, VIRTIO_F_VERSION_1,
}; };
use crate::VirtioInterrupt; use crate::VirtioInterrupt;
use vm_memory::{Bytes, GuestMemoryMmap}; use vm_memory::{Bytes, GuestMemoryMmap};
@ -38,7 +37,6 @@ const KILL_EVENT: DeviceEventT = 3;
struct ConsoleEpollHandler { struct ConsoleEpollHandler {
queues: Vec<Queue>, queues: Vec<Queue>,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_status: Arc<AtomicUsize>,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
in_buffer: Arc<Mutex<VecDeque<u8>>>, in_buffer: Arc<Mutex<VecDeque<u8>>>,
out: Box<io::Write + Send>, out: Box<io::Write + Send>,
@ -128,9 +126,7 @@ impl ConsoleEpollHandler {
} }
fn signal_used_queue(&self) -> result::Result<(), DeviceError> { fn signal_used_queue(&self) -> result::Result<(), DeviceError> {
self.interrupt_status (self.interrupt_cb)(&VirtioInterruptType::Queue, Some(&self.queues[0])).map_err(|e| {
.fetch_or(INTERRUPT_STATUS_USED_RING as usize, Ordering::SeqCst);
(self.interrupt_cb)(&self.queues[0]).map_err(|e| {
error!("Failed to signal used queue: {:?}", e); error!("Failed to signal used queue: {:?}", e);
DeviceError::FailedSignalingUsedQueue(e) DeviceError::FailedSignalingUsedQueue(e)
}) })
@ -333,7 +329,6 @@ impl VirtioDevice for Console {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
queues: Vec<Queue>, queues: Vec<Queue>,
mut queue_evts: Vec<EventFd>, mut queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -360,7 +355,6 @@ impl VirtioDevice for Console {
let mut handler = ConsoleEpollHandler { let mut handler = ConsoleEpollHandler {
queues, queues,
mem, mem,
interrupt_status: status,
interrupt_cb, interrupt_cb,
in_buffer: self.input.in_buffer.clone(), in_buffer: self.input.in_buffer.clone(),
out, out,

View File

@ -8,12 +8,20 @@
use super::*; use super::*;
use pci::{PciBarConfiguration, PciCapability}; use pci::{PciBarConfiguration, PciCapability};
use std::sync::atomic::AtomicUsize;
use std::sync::Arc; use std::sync::Arc;
use vm_memory::GuestMemoryMmap; use vm_memory::GuestMemoryMmap;
use vmm_sys_util::EventFd; use vmm_sys_util::EventFd;
pub type VirtioInterrupt = Box<Fn(&Queue) -> std::result::Result<(), std::io::Error> + Send + Sync>; pub enum VirtioInterruptType {
Config,
Queue,
}
pub type VirtioInterrupt = Box<
Fn(&VirtioInterruptType, Option<&Queue>) -> std::result::Result<(), std::io::Error>
+ Send
+ Sync,
>;
/// Trait for virtio devices to be driven by a virtio transport. /// Trait for virtio devices to be driven by a virtio transport.
/// ///
@ -49,7 +57,6 @@ pub trait VirtioDevice: Send {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_evt: Arc<VirtioInterrupt>, interrupt_evt: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
queues: Vec<Queue>, queues: Vec<Queue>,
queue_evts: Vec<EventFd>, queue_evts: Vec<EventFd>,
) -> ActivateResult; ) -> ActivateResult;

View File

@ -2,11 +2,8 @@
// SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: Apache-2.0
use super::Error as DeviceError; use super::Error as DeviceError;
use super::{ use super::{ActivateError, ActivateResult, Queue, VirtioDevice, VirtioDeviceType};
ActivateError, ActivateResult, Queue, VirtioDevice, VirtioDeviceType, use crate::{VirtioInterrupt, VirtioInterruptType, VIRTIO_F_VERSION_1_BITMASK};
INTERRUPT_STATUS_USED_RING,
};
use crate::{VirtioInterrupt, VIRTIO_F_VERSION_1_BITMASK};
use epoll; use epoll;
use libc::EFD_NONBLOCK; use libc::EFD_NONBLOCK;
use std::cmp; use std::cmp;
@ -14,7 +11,6 @@ use std::io;
use std::io::Write; use std::io::Write;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::result; use std::result;
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc; use std::sync::Arc;
use std::thread; use std::thread;
use vhost_rs::vhost_user::message::{VhostUserProtocolFeatures, VhostUserVirtioFeatures}; use vhost_rs::vhost_user::message::{VhostUserProtocolFeatures, VhostUserVirtioFeatures};
@ -92,7 +88,6 @@ type Result<T> = result::Result<T, Error>;
struct FsEpollHandler { struct FsEpollHandler {
vu_call_evt_queue_list: Vec<(EventFd, Queue)>, vu_call_evt_queue_list: Vec<(EventFd, Queue)>,
interrupt_status: Arc<AtomicUsize>,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
kill_evt: EventFd, kill_evt: EventFd,
} }
@ -137,16 +132,15 @@ impl FsEpollHandler {
if let Err(e) = self.vu_call_evt_queue_list[x].0.read() { if let Err(e) = self.vu_call_evt_queue_list[x].0.read() {
error!("Failed to get queue event: {:?}", e); error!("Failed to get queue event: {:?}", e);
break 'epoll; break 'epoll;
} else { } else if let Err(e) = (self.interrupt_cb)(
self.interrupt_status &VirtioInterruptType::Queue,
.fetch_or(INTERRUPT_STATUS_USED_RING as usize, Ordering::SeqCst); Some(&self.vu_call_evt_queue_list[x].1),
if let Err(e) = (self.interrupt_cb)(&self.vu_call_evt_queue_list[x].1) { ) {
error!( error!(
"Failed to signal used queue: {:?}", "Failed to signal used queue: {:?}",
DeviceError::FailedSignalingUsedQueue(e) DeviceError::FailedSignalingUsedQueue(e)
); );
break 'epoll; break 'epoll;
}
} }
} }
x if (x == kill_evt_index) => { x if (x == kill_evt_index) => {
@ -390,7 +384,6 @@ impl VirtioDevice for Fs {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
queues: Vec<Queue>, queues: Vec<Queue>,
queue_evts: Vec<EventFd>, queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -419,7 +412,6 @@ impl VirtioDevice for Fs {
let mut handler = FsEpollHandler { let mut handler = FsEpollHandler {
vu_call_evt_queue_list, vu_call_evt_queue_list,
interrupt_status: status,
interrupt_cb, interrupt_cb,
kill_evt, kill_evt,
}; };

View File

@ -15,7 +15,6 @@ use std::mem;
use std::net::Ipv4Addr; use std::net::Ipv4Addr;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::result; use std::result;
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc; use std::sync::Arc;
use std::thread; use std::thread;
use std::vec::Vec; use std::vec::Vec;
@ -25,7 +24,7 @@ use net_gen;
use super::Error as DeviceError; use super::Error as DeviceError;
use super::{ use super::{
ActivateError, ActivateResult, DeviceEventT, Queue, VirtioDevice, VirtioDeviceType, ActivateError, ActivateResult, DeviceEventT, Queue, VirtioDevice, VirtioDeviceType,
INTERRUPT_STATUS_USED_RING, VirtioInterruptType,
}; };
use crate::VirtioInterrupt; use crate::VirtioInterrupt;
use net_util::{MacAddr, Tap, TapError, MAC_ADDR_LEN}; use net_util::{MacAddr, Tap, TapError, MAC_ADDR_LEN};
@ -120,16 +119,13 @@ struct NetEpollHandler {
tap: Tap, tap: Tap,
rx: RxVirtio, rx: RxVirtio,
tx: TxVirtio, tx: TxVirtio,
interrupt_status: Arc<AtomicUsize>,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
kill_evt: EventFd, kill_evt: EventFd,
} }
impl NetEpollHandler { impl NetEpollHandler {
fn signal_used_queue(&self, queue: &Queue) -> result::Result<(), DeviceError> { fn signal_used_queue(&self, queue: &Queue) -> result::Result<(), DeviceError> {
self.interrupt_status (self.interrupt_cb)(&VirtioInterruptType::Queue, Some(queue)).map_err(|e| {
.fetch_or(INTERRUPT_STATUS_USED_RING as usize, Ordering::SeqCst);
(self.interrupt_cb)(queue).map_err(|e| {
error!("Failed to signal used queue: {:?}", e); error!("Failed to signal used queue: {:?}", e);
DeviceError::FailedSignalingUsedQueue(e) DeviceError::FailedSignalingUsedQueue(e)
}) })
@ -538,7 +534,6 @@ impl VirtioDevice for Net {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
mut queues: Vec<Queue>, mut queues: Vec<Queue>,
mut queue_evts: Vec<EventFd>, mut queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -571,7 +566,6 @@ impl VirtioDevice for Net {
tap, tap,
rx: RxVirtio::new(rx_queue, rx_queue_evt), rx: RxVirtio::new(rx_queue, rx_queue_evt),
tx: TxVirtio::new(tx_queue, tx_queue_evt), tx: TxVirtio::new(tx_queue, tx_queue_evt),
interrupt_status: status,
interrupt_cb, interrupt_cb,
kill_evt, kill_evt,
}; };

View File

@ -15,16 +15,15 @@ use std::io::{self, Write};
use std::mem::size_of; use std::mem::size_of;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::result; use std::result;
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc; use std::sync::Arc;
use std::thread; use std::thread;
use super::Error as DeviceError; use super::Error as DeviceError;
use super::{ use super::{
ActivateError, ActivateResult, DescriptorChain, DeviceEventT, Queue, VirtioDevice, ActivateError, ActivateResult, DescriptorChain, DeviceEventT, Queue, VirtioDevice,
VirtioDeviceType, INTERRUPT_STATUS_USED_RING, VIRTIO_F_VERSION_1, VirtioDeviceType, VIRTIO_F_VERSION_1,
}; };
use crate::VirtioInterrupt; use crate::{VirtioInterrupt, VirtioInterruptType};
use vm_memory::{ use vm_memory::{
Address, ByteValued, Bytes, GuestAddress, GuestMemoryError, GuestMemoryMmap, GuestUsize, Address, ByteValued, Bytes, GuestAddress, GuestMemoryError, GuestMemoryMmap, GuestUsize,
}; };
@ -157,7 +156,6 @@ struct PmemEpollHandler {
queue: Queue, queue: Queue,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
disk: File, disk: File,
interrupt_status: Arc<AtomicUsize>,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
queue_evt: EventFd, queue_evt: EventFd,
kill_evt: EventFd, kill_evt: EventFd,
@ -209,9 +207,7 @@ impl PmemEpollHandler {
} }
fn signal_used_queue(&self) -> result::Result<(), DeviceError> { fn signal_used_queue(&self) -> result::Result<(), DeviceError> {
self.interrupt_status (self.interrupt_cb)(&VirtioInterruptType::Queue, Some(&self.queue)).map_err(|e| {
.fetch_or(INTERRUPT_STATUS_USED_RING as usize, Ordering::SeqCst);
(self.interrupt_cb)(&self.queue).map_err(|e| {
error!("Failed to signal used queue: {:?}", e); error!("Failed to signal used queue: {:?}", e);
DeviceError::FailedSignalingUsedQueue(e) DeviceError::FailedSignalingUsedQueue(e)
}) })
@ -374,7 +370,6 @@ impl VirtioDevice for Pmem {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
mut queues: Vec<Queue>, mut queues: Vec<Queue>,
mut queue_evts: Vec<EventFd>, mut queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -402,7 +397,6 @@ impl VirtioDevice for Pmem {
queue: queues.remove(0), queue: queues.remove(0),
mem, mem,
disk, disk,
interrupt_status: status,
interrupt_cb, interrupt_cb,
queue_evt: queue_evts.remove(0), queue_evt: queue_evts.remove(0),
kill_evt, kill_evt,

View File

@ -9,16 +9,15 @@ use std::fs::File;
use std::io; use std::io;
use std::os::unix::io::AsRawFd; use std::os::unix::io::AsRawFd;
use std::result; use std::result;
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::Arc; use std::sync::Arc;
use std::thread; use std::thread;
use super::Error as DeviceError; use super::Error as DeviceError;
use super::{ use super::{
ActivateError, ActivateResult, DeviceEventT, Queue, VirtioDevice, VirtioDeviceType, ActivateError, ActivateResult, DeviceEventT, Queue, VirtioDevice, VirtioDeviceType,
INTERRUPT_STATUS_USED_RING, VIRTIO_F_VERSION_1, VIRTIO_F_VERSION_1,
}; };
use crate::VirtioInterrupt; use crate::{VirtioInterrupt, VirtioInterruptType};
use vm_memory::{Bytes, GuestMemoryMmap}; use vm_memory::{Bytes, GuestMemoryMmap};
use vmm_sys_util::EventFd; use vmm_sys_util::EventFd;
@ -35,7 +34,6 @@ struct RngEpollHandler {
queues: Vec<Queue>, queues: Vec<Queue>,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
random_file: File, random_file: File,
interrupt_status: Arc<AtomicUsize>,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
queue_evt: EventFd, queue_evt: EventFd,
kill_evt: EventFd, kill_evt: EventFd,
@ -77,9 +75,7 @@ impl RngEpollHandler {
} }
fn signal_used_queue(&self) -> result::Result<(), DeviceError> { fn signal_used_queue(&self) -> result::Result<(), DeviceError> {
self.interrupt_status (self.interrupt_cb)(&VirtioInterruptType::Queue, Some(&self.queues[0])).map_err(|e| {
.fetch_or(INTERRUPT_STATUS_USED_RING as usize, Ordering::SeqCst);
(self.interrupt_cb)(&self.queues[0]).map_err(|e| {
error!("Failed to signal used queue: {:?}", e); error!("Failed to signal used queue: {:?}", e);
DeviceError::FailedSignalingUsedQueue(e) DeviceError::FailedSignalingUsedQueue(e)
}) })
@ -229,7 +225,6 @@ impl VirtioDevice for Rng {
&mut self, &mut self,
mem: GuestMemoryMmap, mem: GuestMemoryMmap,
interrupt_cb: Arc<VirtioInterrupt>, interrupt_cb: Arc<VirtioInterrupt>,
status: Arc<AtomicUsize>,
queues: Vec<Queue>, queues: Vec<Queue>,
mut queue_evts: Vec<EventFd>, mut queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -257,7 +252,6 @@ impl VirtioDevice for Rng {
queues, queues,
mem, mem,
random_file, random_file,
interrupt_status: status,
interrupt_cb, interrupt_cb,
queue_evt: queue_evts.remove(0), queue_evt: queue_evts.remove(0),
kill_evt, kill_evt,

View File

@ -8,6 +8,8 @@
extern crate byteorder; extern crate byteorder;
use byteorder::{ByteOrder, LittleEndian}; use byteorder::{ByteOrder, LittleEndian};
use std::sync::atomic::{AtomicU16, Ordering};
use std::sync::Arc;
use vm_memory::GuestAddress; use vm_memory::GuestAddress;
use crate::{Queue, VirtioDevice}; use crate::{Queue, VirtioDevice};
@ -40,7 +42,7 @@ pub struct VirtioPciCommonConfig {
pub device_feature_select: u32, pub device_feature_select: u32,
pub driver_feature_select: u32, pub driver_feature_select: u32,
pub queue_select: u16, pub queue_select: u16,
pub msix_config: u16, pub msix_config: Arc<AtomicU16>,
} }
impl VirtioPciCommonConfig { impl VirtioPciCommonConfig {
@ -120,7 +122,7 @@ impl VirtioPciCommonConfig {
fn read_common_config_word(&self, offset: u64, queues: &[Queue]) -> u16 { fn read_common_config_word(&self, offset: u64, queues: &[Queue]) -> u16 {
debug!("read_common_config_word: offset 0x{:x}", offset); debug!("read_common_config_word: offset 0x{:x}", offset);
match offset { match offset {
0x10 => self.msix_config, 0x10 => self.msix_config.load(Ordering::SeqCst),
0x12 => queues.len() as u16, // num_queues 0x12 => queues.len() as u16, // num_queues
0x16 => self.queue_select, 0x16 => self.queue_select,
0x18 => self.with_queue(queues, |q| q.size).unwrap_or(0), 0x18 => self.with_queue(queues, |q| q.size).unwrap_or(0),
@ -143,7 +145,7 @@ impl VirtioPciCommonConfig {
fn write_common_config_word(&mut self, offset: u64, value: u16, queues: &mut Vec<Queue>) { fn write_common_config_word(&mut self, offset: u64, value: u16, queues: &mut Vec<Queue>) {
debug!("write_common_config_word: offset 0x{:x}", offset); debug!("write_common_config_word: offset 0x{:x}", offset);
match offset { match offset {
0x10 => self.msix_config = value, 0x10 => self.msix_config.store(value, Ordering::SeqCst),
0x16 => self.queue_select = value, 0x16 => self.queue_select = value,
0x18 => self.with_queue_mut(queues, |q| q.size = value), 0x18 => self.with_queue_mut(queues, |q| q.size = value),
0x1a => self.with_queue_mut(queues, |q| q.vector = value), 0x1a => self.with_queue_mut(queues, |q| q.vector = value),
@ -272,7 +274,6 @@ mod tests {
&mut self, &mut self,
_mem: GuestMemoryMmap, _mem: GuestMemoryMmap,
_interrupt_evt: Arc<VirtioInterrupt>, _interrupt_evt: Arc<VirtioInterrupt>,
_status: Arc<AtomicUsize>,
_queues: Vec<Queue>, _queues: Vec<Queue>,
_queue_evts: Vec<EventFd>, _queue_evts: Vec<EventFd>,
) -> ActivateResult { ) -> ActivateResult {
@ -298,7 +299,7 @@ mod tests {
device_feature_select: 0x0, device_feature_select: 0x0,
driver_feature_select: 0x0, driver_feature_select: 0x0,
queue_select: 0xff, queue_select: 0xff,
msix_config: 0, msix_config: Arc::new(AtomicU16::new(0)),
}; };
let dev = &mut DummyDevice(0) as &mut dyn VirtioDevice; let dev = &mut DummyDevice(0) as &mut dyn VirtioDevice;

View File

@ -13,7 +13,7 @@ extern crate vm_memory;
extern crate vmm_sys_util; extern crate vmm_sys_util;
use libc::EFD_NONBLOCK; use libc::EFD_NONBLOCK;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicU16, AtomicUsize, Ordering};
use std::sync::Arc; use std::sync::Arc;
use std::sync::Mutex; use std::sync::Mutex;
@ -30,8 +30,9 @@ use vmm_sys_util::{EventFd, Result};
use super::VirtioPciCommonConfig; use super::VirtioPciCommonConfig;
use crate::{ use crate::{
Queue, VirtioDevice, VirtioDeviceType, VirtioInterrupt, DEVICE_ACKNOWLEDGE, DEVICE_DRIVER, Queue, VirtioDevice, VirtioDeviceType, VirtioInterrupt, VirtioInterruptType,
DEVICE_DRIVER_OK, DEVICE_FAILED, DEVICE_FEATURES_OK, DEVICE_INIT, DEVICE_ACKNOWLEDGE, DEVICE_DRIVER, DEVICE_DRIVER_OK, DEVICE_FAILED, DEVICE_FEATURES_OK,
DEVICE_INIT, INTERRUPT_STATUS_CONFIG_CHANGED, INTERRUPT_STATUS_USED_RING,
}; };
#[allow(clippy::enum_variant_names)] #[allow(clippy::enum_variant_names)]
@ -254,7 +255,7 @@ impl VirtioPciDevice {
device_feature_select: 0, device_feature_select: 0,
driver_feature_select: 0, driver_feature_select: 0,
queue_select: 0, queue_select: 0,
msix_config: 0, msix_config: Arc::new(AtomicU16::new(0)),
}, },
msix_config, msix_config,
msix_num, msix_num,
@ -376,10 +377,20 @@ impl PciDevice for VirtioPciDevice {
) { ) {
self.configuration.set_irq(irq_num as u8, irq_pin); self.configuration.set_irq(irq_num as u8, irq_pin);
let cb = Arc::new(Box::new(move |_queue: &Queue| { let interrupt_status = self.interrupt_status.clone();
let param = InterruptParameters { msix: None }; let cb = Arc::new(Box::new(
(irq_cb)(param) move |int_type: &VirtioInterruptType, _queue: Option<&Queue>| {
}) as VirtioInterrupt); let param = InterruptParameters { msix: None };
let status = match int_type {
VirtioInterruptType::Config => INTERRUPT_STATUS_CONFIG_CHANGED,
VirtioInterruptType::Queue => INTERRUPT_STATUS_USED_RING,
};
interrupt_status.fetch_or(status as usize, Ordering::SeqCst);
(irq_cb)(param)
},
) as VirtioInterrupt);
self.interrupt_cb = Some(cb); self.interrupt_cb = Some(cb);
} }
@ -393,22 +404,38 @@ impl PciDevice for VirtioPciDevice {
let msix_config_clone = msix_config.clone(); let msix_config_clone = msix_config.clone();
let cb = Arc::new(Box::new(move |queue: &Queue| { let common_config_msi_vector = self.common_config.msix_config.clone();
let config = &mut msix_config_clone.lock().unwrap(); let cb = Arc::new(Box::new(
let entry = &config.table_entries[queue.vector as usize]; move |int_type: &VirtioInterruptType, queue: Option<&Queue>| {
let vector = match int_type {
VirtioInterruptType::Config => {
common_config_msi_vector.load(Ordering::SeqCst)
}
VirtioInterruptType::Queue => {
if let Some(q) = queue {
q.vector
} else {
0
}
}
};
// In case the vector control register associated with the entry let config = &mut msix_config_clone.lock().unwrap();
// has its first bit set, this means the vector is masked and the let entry = &config.table_entries[vector as usize];
// device should not inject the interrupt.
// Instead, the Pending Bit Array table is updated to reflect there
// is a pending interrupt for this specific vector.
if config.masked() || entry.masked() {
config.set_pba_bit(queue.vector, false);
return Ok(());
}
(msi_cb)(InterruptParameters { msix: Some(entry) }) // In case the vector control register associated with the entry
}) as VirtioInterrupt); // has its first bit set, this means the vector is masked and the
// device should not inject the interrupt.
// Instead, the Pending Bit Array table is updated to reflect there
// is a pending interrupt for this specific vector.
if config.masked() || entry.masked() {
config.set_pba_bit(vector, false);
return Ok(());
}
(msi_cb)(InterruptParameters { msix: Some(entry) })
},
) as VirtioInterrupt);
self.interrupt_cb = Some(cb); self.interrupt_cb = Some(cb);
} }
@ -585,7 +612,6 @@ impl PciDevice for VirtioPciDevice {
.activate( .activate(
mem, mem,
interrupt_cb, interrupt_cb,
self.interrupt_status.clone(),
self.queues.clone(), self.queues.clone(),
self.queue_evts.split_off(0), self.queue_evts.split_off(0),
) )