// Copyright 2020, ARM Limited. // // SPDX-License-Identifier: Apache-2.0 AND BSD-3-Clause use super::interrupt_controller::{Error, InterruptController}; extern crate arch; use anyhow::anyhow; use arch::layout; use hypervisor::{ arch::aarch64::gic::{Vgic, VgicConfig}, CpuState, }; use std::result; use std::sync::{Arc, Mutex}; use vm_device::interrupt::{ InterruptIndex, InterruptManager, InterruptSourceConfig, InterruptSourceGroup, LegacyIrqSourceConfig, MsiIrqGroupConfig, }; use vm_memory::address::Address; use vm_migration::{Migratable, MigratableError, Pausable, Snapshot, Snapshottable, Transportable}; use vmm_sys_util::eventfd::EventFd; type Result = result::Result; // Reserve 32 IRQs for legacy devices. pub const IRQ_LEGACY_BASE: usize = layout::IRQ_BASE as usize; pub const IRQ_LEGACY_COUNT: usize = 32; // Gic (Generic Interupt Controller) struct provides all the functionality of a // GIC device. It wraps a hypervisor-emulated GIC device (Vgic) provided by the // `hypervisor` crate. // Gic struct also implements InterruptController to provide interrupt delivery // service. pub struct Gic { interrupt_source_group: Arc, // The hypervisor agnostic virtual GIC vgic: Option>>, } impl Gic { pub fn new( vcpu_count: u8, interrupt_manager: Arc>, vm: Arc, ) -> Result { let interrupt_source_group = interrupt_manager .create_group(MsiIrqGroupConfig { base: IRQ_LEGACY_BASE as InterruptIndex, count: IRQ_LEGACY_COUNT as InterruptIndex, }) .map_err(Error::CreateInterruptSourceGroup)?; let vgic = vm .create_vgic(Gic::create_default_config(vcpu_count as u64)) .map_err(Error::CreateGic)?; let gic = Gic { interrupt_source_group, vgic: Some(vgic), }; gic.enable()?; Ok(gic) } fn enable(&self) -> Result<()> { // Set irqfd for legacy interrupts self.interrupt_source_group .enable() .map_err(Error::EnableInterrupt)?; // Set irq_routing for legacy interrupts. // irqchip: Hardcode to 0 as we support only 1 GIC // pin: Use irq number as pin for i in IRQ_LEGACY_BASE..(IRQ_LEGACY_BASE + IRQ_LEGACY_COUNT) { let config = LegacyIrqSourceConfig { irqchip: 0, pin: (i - IRQ_LEGACY_BASE) as u32, }; self.interrupt_source_group .update( i as InterruptIndex, InterruptSourceConfig::LegacyIrq(config), false, ) .map_err(Error::EnableInterrupt)?; } Ok(()) } /// Default config implied by arch::layout pub fn create_default_config(vcpu_count: u64) -> VgicConfig { let redists_size = layout::GIC_V3_REDIST_SIZE * vcpu_count; let redists_addr = layout::GIC_V3_DIST_START.raw_value() - redists_size; VgicConfig { vcpu_count, dist_addr: layout::GIC_V3_DIST_START.raw_value(), dist_size: layout::GIC_V3_DIST_SIZE, redists_addr, redists_size, msi_addr: redists_addr - layout::GIC_V3_ITS_SIZE, msi_size: layout::GIC_V3_ITS_SIZE, nr_irqs: layout::IRQ_NUM, } } pub fn get_vgic(&mut self) -> Result>> { Ok(self.vgic.clone().unwrap()) } pub fn set_gicr_typers(&mut self, vcpu_states: &[CpuState]) { let vgic = self.vgic.as_ref().unwrap().clone(); vgic.lock().unwrap().set_gicr_typers(vcpu_states); } } impl InterruptController for Gic { // This should be called anytime an interrupt needs to be injected into the // running guest. fn service_irq(&mut self, irq: usize) -> Result<()> { self.interrupt_source_group .trigger(irq as InterruptIndex) .map_err(Error::TriggerInterrupt)?; Ok(()) } fn notifier(&self, irq: usize) -> Option { self.interrupt_source_group.notifier(irq as InterruptIndex) } } pub const GIC_V3_ITS_SNAPSHOT_ID: &str = "gic-v3-its"; impl Snapshottable for Gic { fn id(&self) -> String { GIC_V3_ITS_SNAPSHOT_ID.to_string() } fn snapshot(&mut self) -> std::result::Result { let vgic = self.vgic.as_ref().unwrap().clone(); let state = vgic.lock().unwrap().state().unwrap(); Snapshot::new_from_state(&self.id(), &state) } fn restore(&mut self, snapshot: Snapshot) -> std::result::Result<(), MigratableError> { let vgic = self.vgic.as_ref().unwrap().clone(); vgic.lock() .unwrap() .set_state(&snapshot.to_state(&self.id())?) .map_err(|e| { MigratableError::Restore(anyhow!("Could not restore GICv3ITS state {:?}", e)) })?; Ok(()) } } impl Pausable for Gic { fn pause(&mut self) -> std::result::Result<(), MigratableError> { // Flush tables to guest RAM let vgic = self.vgic.as_ref().unwrap().clone(); vgic.lock().unwrap().save_data_tables().map_err(|e| { MigratableError::Pause(anyhow!( "Could not save GICv3ITS GIC pending tables {:?}", e )) })?; Ok(()) } } impl Transportable for Gic {} impl Migratable for Gic {}