use crate::KvmPartitionInner;
use pal_event::Event;
use parking_lot::Mutex;
use std::os::unix::prelude::*;
use std::sync::Arc;
use std::sync::Weak;
use std::sync::atomic::AtomicBool;
use std::sync::atomic::Ordering;
const NUM_GSIS: usize = 2048;
#[derive(Debug)]
pub struct GsiRouting {
states: Box<[GsiState; NUM_GSIS]>,
}
impl GsiRouting {
pub fn new() -> Self {
Self {
states: Box::new([GsiState::Unallocated; NUM_GSIS]),
}
}
pub fn claim(&mut self, gsi: u32) {
let gsi = gsi as usize;
assert_eq!(self.states[gsi], GsiState::Unallocated);
self.states[gsi] = GsiState::Disabled;
}
pub fn alloc(&mut self) -> Option<u32> {
let gsi = self.states.iter().position(|state| !state.is_allocated())?;
self.states[gsi] = GsiState::Disabled;
Some(gsi as u32)
}
pub fn free(&mut self, gsi: u32) {
let gsi = gsi as usize;
assert_eq!(self.states[gsi], GsiState::Disabled);
self.states[gsi] = GsiState::Unallocated;
}
pub fn set(&mut self, gsi: u32, entry: Option<kvm::RoutingEntry>) -> bool {
let new_state = entry.map_or(GsiState::Disabled, GsiState::Enabled);
let state = &mut self.states[gsi as usize];
assert!(state.is_allocated());
if *state != new_state {
*state = new_state;
true
} else {
false
}
}
pub fn update_routes(&mut self, kvm: &kvm::Partition) {
let routing: Vec<_> = self
.states
.iter()
.enumerate()
.filter_map(|(gsi, state)| match state {
GsiState::Unallocated | GsiState::Disabled => None,
GsiState::Enabled(entry) => Some((gsi as u32, *entry)),
})
.collect();
kvm.set_gsi_routes(&routing).expect("should not fail");
}
}
impl KvmPartitionInner {
pub(crate) fn new_route(self: &Arc<Self>, irqfd_event: Option<Event>) -> Option<GsiRoute> {
let gsi = self.gsi_routing.lock().alloc()?;
Some(GsiRoute {
partition: Arc::downgrade(self),
gsi,
irqfd_event,
enabled: false.into(),
enable_mutex: Mutex::new(()),
})
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
enum GsiState {
Unallocated,
Disabled,
Enabled(kvm::RoutingEntry),
}
impl GsiState {
fn is_allocated(&self) -> bool {
!matches!(self, GsiState::Unallocated)
}
}
#[derive(Debug)]
pub struct GsiRoute {
partition: Weak<KvmPartitionInner>,
gsi: u32,
irqfd_event: Option<Event>,
enabled: AtomicBool,
enable_mutex: Mutex<()>, }
impl Drop for GsiRoute {
fn drop(&mut self) {
self.disable();
self.set_entry(None);
if let Some(partition) = self.partition.upgrade() {
partition.gsi_routing.lock().free(self.gsi);
}
}
}
impl GsiRoute {
fn set_entry(&self, new_entry: Option<kvm::RoutingEntry>) -> Option<Arc<KvmPartitionInner>> {
let partition = self.partition.upgrade();
if let Some(partition) = &partition {
let mut routing = partition.gsi_routing.lock();
if routing.set(self.gsi, new_entry) {
routing.update_routes(&partition.kvm);
}
}
partition
}
pub fn enable(&self, entry: kvm::RoutingEntry) {
let partition = self.set_entry(Some(entry));
let _lock = self.enable_mutex.lock();
if !self.enabled.load(Ordering::Relaxed) {
if let (Some(partition), Some(event)) = (&partition, &self.irqfd_event) {
partition
.kvm
.irqfd(self.gsi, event.as_fd().as_raw_fd(), true)
.expect("should not fail");
}
self.enabled.store(true, Ordering::Relaxed);
}
}
pub fn disable(&self) {
let _lock = self.enable_mutex.lock();
if self.enabled.load(Ordering::Relaxed) {
if let Some(irqfd_event) = &self.irqfd_event {
if let Some(partition) = self.partition.upgrade() {
partition
.kvm
.irqfd(self.gsi, irqfd_event.as_fd().as_raw_fd(), false)
.expect("should not fail");
}
}
self.enabled.store(false, Ordering::Relaxed);
}
}
pub fn irqfd_event(&self) -> Option<&Event> {
self.irqfd_event.as_ref()
}
pub fn _signal(&self) {
if self.enabled.load(Ordering::Relaxed) {
if let Some(event) = &self.irqfd_event {
event.signal();
} else if let Some(partition) = self.partition.upgrade() {
assert!(cfg!(target_arch = "x86_64"));
partition
.kvm
.irq_line(self.gsi, true)
.expect("interrupt delivery failure");
}
}
}
}