1*bb4ee6a4SAndroid Build Coastguard Worker // Copyright 2023 The ChromiumOS Authors
2*bb4ee6a4SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*bb4ee6a4SAndroid Build Coastguard Worker // found in the LICENSE file.
4*bb4ee6a4SAndroid Build Coastguard Worker
5*bb4ee6a4SAndroid Build Coastguard Worker use std::sync::Arc;
6*bb4ee6a4SAndroid Build Coastguard Worker
7*bb4ee6a4SAndroid Build Coastguard Worker use base::error;
8*bb4ee6a4SAndroid Build Coastguard Worker use base::Error;
9*bb4ee6a4SAndroid Build Coastguard Worker use base::Event;
10*bb4ee6a4SAndroid Build Coastguard Worker use base::Result;
11*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::geniezone::geniezone_sys::*;
12*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::geniezone::GeniezoneVcpu;
13*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::geniezone::GeniezoneVm;
14*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::DeviceKind;
15*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::IrqRoute;
16*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::MPState;
17*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::Vcpu;
18*bb4ee6a4SAndroid Build Coastguard Worker use hypervisor::Vm;
19*bb4ee6a4SAndroid Build Coastguard Worker use resources::SystemAllocator;
20*bb4ee6a4SAndroid Build Coastguard Worker use sync::Mutex;
21*bb4ee6a4SAndroid Build Coastguard Worker
22*bb4ee6a4SAndroid Build Coastguard Worker use crate::Bus;
23*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqChip;
24*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqChipAArch64;
25*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqChipCap;
26*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqEdgeEvent;
27*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqEventIndex;
28*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqEventSource;
29*bb4ee6a4SAndroid Build Coastguard Worker use crate::IrqLevelEvent;
30*bb4ee6a4SAndroid Build Coastguard Worker use crate::VcpuRunState;
31*bb4ee6a4SAndroid Build Coastguard Worker
32*bb4ee6a4SAndroid Build Coastguard Worker /// Default ARM routing table. AARCH64_GIC_NR_SPIS pins go to VGIC.
default_irq_routing_table() -> Vec<IrqRoute>33*bb4ee6a4SAndroid Build Coastguard Worker fn default_irq_routing_table() -> Vec<IrqRoute> {
34*bb4ee6a4SAndroid Build Coastguard Worker let mut routes: Vec<IrqRoute> = Vec::new();
35*bb4ee6a4SAndroid Build Coastguard Worker
36*bb4ee6a4SAndroid Build Coastguard Worker for i in 0..AARCH64_GIC_NR_SPIS {
37*bb4ee6a4SAndroid Build Coastguard Worker routes.push(IrqRoute::gic_irq_route(i));
38*bb4ee6a4SAndroid Build Coastguard Worker }
39*bb4ee6a4SAndroid Build Coastguard Worker
40*bb4ee6a4SAndroid Build Coastguard Worker routes
41*bb4ee6a4SAndroid Build Coastguard Worker }
42*bb4ee6a4SAndroid Build Coastguard Worker
43*bb4ee6a4SAndroid Build Coastguard Worker /// IrqChip implementation where the entire IrqChip is emulated by GZVM.
44*bb4ee6a4SAndroid Build Coastguard Worker ///
45*bb4ee6a4SAndroid Build Coastguard Worker /// This implementation will use the GZVM API to create and configure the in-kernel irqchip.
46*bb4ee6a4SAndroid Build Coastguard Worker pub struct GeniezoneKernelIrqChip {
47*bb4ee6a4SAndroid Build Coastguard Worker pub(super) vm: GeniezoneVm,
48*bb4ee6a4SAndroid Build Coastguard Worker pub(super) vcpus: Arc<Mutex<Vec<Option<GeniezoneVcpu>>>>,
49*bb4ee6a4SAndroid Build Coastguard Worker device_kind: DeviceKind,
50*bb4ee6a4SAndroid Build Coastguard Worker pub(super) routes: Arc<Mutex<Vec<IrqRoute>>>,
51*bb4ee6a4SAndroid Build Coastguard Worker }
52*bb4ee6a4SAndroid Build Coastguard Worker
53*bb4ee6a4SAndroid Build Coastguard Worker // These constants indicate the address space used by the ARM vGIC.
54*bb4ee6a4SAndroid Build Coastguard Worker const AARCH64_GIC_DIST_SIZE: u64 = 0x10000;
55*bb4ee6a4SAndroid Build Coastguard Worker
56*bb4ee6a4SAndroid Build Coastguard Worker // These constants indicate the placement of the GIC registers in the physical
57*bb4ee6a4SAndroid Build Coastguard Worker // address space.
58*bb4ee6a4SAndroid Build Coastguard Worker const AARCH64_GIC_DIST_BASE: u64 = 0x40000000 - AARCH64_GIC_DIST_SIZE;
59*bb4ee6a4SAndroid Build Coastguard Worker const AARCH64_GIC_REDIST_SIZE: u64 = 0x20000;
60*bb4ee6a4SAndroid Build Coastguard Worker
61*bb4ee6a4SAndroid Build Coastguard Worker // This is the minimum number of SPI interrupts aligned to 32 + 32 for the
62*bb4ee6a4SAndroid Build Coastguard Worker // PPI (16) and GSI (16).
63*bb4ee6a4SAndroid Build Coastguard Worker // pub const AARCH64_GIC_NR_IRQS: u32 = 64;
64*bb4ee6a4SAndroid Build Coastguard Worker // Number of SPIs (32), which is the NR_IRQS (64) minus the number of PPIs (16) and GSIs (16)
65*bb4ee6a4SAndroid Build Coastguard Worker pub const AARCH64_GIC_NR_SPIS: u32 = 32;
66*bb4ee6a4SAndroid Build Coastguard Worker
67*bb4ee6a4SAndroid Build Coastguard Worker impl GeniezoneKernelIrqChip {
68*bb4ee6a4SAndroid Build Coastguard Worker /// Construct a new GzvmKernelIrqchip.
new(vm: GeniezoneVm, num_vcpus: usize) -> Result<GeniezoneKernelIrqChip>69*bb4ee6a4SAndroid Build Coastguard Worker pub fn new(vm: GeniezoneVm, num_vcpus: usize) -> Result<GeniezoneKernelIrqChip> {
70*bb4ee6a4SAndroid Build Coastguard Worker let dist_if_addr: u64 = AARCH64_GIC_DIST_BASE;
71*bb4ee6a4SAndroid Build Coastguard Worker let redist_addr: u64 = dist_if_addr - (AARCH64_GIC_REDIST_SIZE * num_vcpus as u64);
72*bb4ee6a4SAndroid Build Coastguard Worker let device_kind = DeviceKind::ArmVgicV3;
73*bb4ee6a4SAndroid Build Coastguard Worker
74*bb4ee6a4SAndroid Build Coastguard Worker // prepare gzvm_create_device and call ioctl
75*bb4ee6a4SAndroid Build Coastguard Worker let device_dis = gzvm_create_device {
76*bb4ee6a4SAndroid Build Coastguard Worker dev_type: gzvm_device_type_GZVM_DEV_TYPE_ARM_VGIC_V3_DIST,
77*bb4ee6a4SAndroid Build Coastguard Worker id: 0,
78*bb4ee6a4SAndroid Build Coastguard Worker flags: 0,
79*bb4ee6a4SAndroid Build Coastguard Worker dev_addr: dist_if_addr,
80*bb4ee6a4SAndroid Build Coastguard Worker dev_reg_size: AARCH64_GIC_DIST_SIZE,
81*bb4ee6a4SAndroid Build Coastguard Worker attr_addr: 0_u64,
82*bb4ee6a4SAndroid Build Coastguard Worker attr_size: 0_u64,
83*bb4ee6a4SAndroid Build Coastguard Worker };
84*bb4ee6a4SAndroid Build Coastguard Worker
85*bb4ee6a4SAndroid Build Coastguard Worker match vm.create_geniezone_device(device_dis) {
86*bb4ee6a4SAndroid Build Coastguard Worker Ok(()) => {}
87*bb4ee6a4SAndroid Build Coastguard Worker Err(e) => {
88*bb4ee6a4SAndroid Build Coastguard Worker error!("failed to create geniezone device with err: {}", e);
89*bb4ee6a4SAndroid Build Coastguard Worker return Err(e);
90*bb4ee6a4SAndroid Build Coastguard Worker }
91*bb4ee6a4SAndroid Build Coastguard Worker };
92*bb4ee6a4SAndroid Build Coastguard Worker
93*bb4ee6a4SAndroid Build Coastguard Worker let device_redis = gzvm_create_device {
94*bb4ee6a4SAndroid Build Coastguard Worker dev_type: gzvm_device_type_GZVM_DEV_TYPE_ARM_VGIC_V3_REDIST,
95*bb4ee6a4SAndroid Build Coastguard Worker id: 0,
96*bb4ee6a4SAndroid Build Coastguard Worker flags: 0,
97*bb4ee6a4SAndroid Build Coastguard Worker dev_addr: redist_addr,
98*bb4ee6a4SAndroid Build Coastguard Worker dev_reg_size: AARCH64_GIC_REDIST_SIZE,
99*bb4ee6a4SAndroid Build Coastguard Worker attr_addr: 0_u64,
100*bb4ee6a4SAndroid Build Coastguard Worker attr_size: 0_u64,
101*bb4ee6a4SAndroid Build Coastguard Worker };
102*bb4ee6a4SAndroid Build Coastguard Worker
103*bb4ee6a4SAndroid Build Coastguard Worker match vm.create_geniezone_device(device_redis) {
104*bb4ee6a4SAndroid Build Coastguard Worker Ok(()) => {}
105*bb4ee6a4SAndroid Build Coastguard Worker Err(e) => {
106*bb4ee6a4SAndroid Build Coastguard Worker error!("failed to create geniezone device with err: {}", e);
107*bb4ee6a4SAndroid Build Coastguard Worker return Err(e);
108*bb4ee6a4SAndroid Build Coastguard Worker }
109*bb4ee6a4SAndroid Build Coastguard Worker };
110*bb4ee6a4SAndroid Build Coastguard Worker
111*bb4ee6a4SAndroid Build Coastguard Worker Ok(GeniezoneKernelIrqChip {
112*bb4ee6a4SAndroid Build Coastguard Worker vm,
113*bb4ee6a4SAndroid Build Coastguard Worker vcpus: Arc::new(Mutex::new((0..num_vcpus).map(|_| None).collect())),
114*bb4ee6a4SAndroid Build Coastguard Worker device_kind,
115*bb4ee6a4SAndroid Build Coastguard Worker routes: Arc::new(Mutex::new(default_irq_routing_table())),
116*bb4ee6a4SAndroid Build Coastguard Worker })
117*bb4ee6a4SAndroid Build Coastguard Worker }
118*bb4ee6a4SAndroid Build Coastguard Worker /// Attempt to create a shallow clone of this aarch64 GzvmKernelIrqChip instance.
arch_try_clone(&self) -> Result<Self>119*bb4ee6a4SAndroid Build Coastguard Worker pub(super) fn arch_try_clone(&self) -> Result<Self> {
120*bb4ee6a4SAndroid Build Coastguard Worker Ok(GeniezoneKernelIrqChip {
121*bb4ee6a4SAndroid Build Coastguard Worker vm: self.vm.try_clone()?,
122*bb4ee6a4SAndroid Build Coastguard Worker vcpus: self.vcpus.clone(),
123*bb4ee6a4SAndroid Build Coastguard Worker device_kind: self.device_kind,
124*bb4ee6a4SAndroid Build Coastguard Worker routes: self.routes.clone(),
125*bb4ee6a4SAndroid Build Coastguard Worker })
126*bb4ee6a4SAndroid Build Coastguard Worker }
127*bb4ee6a4SAndroid Build Coastguard Worker }
128*bb4ee6a4SAndroid Build Coastguard Worker
129*bb4ee6a4SAndroid Build Coastguard Worker impl IrqChipAArch64 for GeniezoneKernelIrqChip {
try_box_clone(&self) -> Result<Box<dyn IrqChipAArch64>>130*bb4ee6a4SAndroid Build Coastguard Worker fn try_box_clone(&self) -> Result<Box<dyn IrqChipAArch64>> {
131*bb4ee6a4SAndroid Build Coastguard Worker Ok(Box::new(self.try_clone()?))
132*bb4ee6a4SAndroid Build Coastguard Worker }
133*bb4ee6a4SAndroid Build Coastguard Worker
as_irq_chip(&self) -> &dyn IrqChip134*bb4ee6a4SAndroid Build Coastguard Worker fn as_irq_chip(&self) -> &dyn IrqChip {
135*bb4ee6a4SAndroid Build Coastguard Worker self
136*bb4ee6a4SAndroid Build Coastguard Worker }
137*bb4ee6a4SAndroid Build Coastguard Worker
as_irq_chip_mut(&mut self) -> &mut dyn IrqChip138*bb4ee6a4SAndroid Build Coastguard Worker fn as_irq_chip_mut(&mut self) -> &mut dyn IrqChip {
139*bb4ee6a4SAndroid Build Coastguard Worker self
140*bb4ee6a4SAndroid Build Coastguard Worker }
141*bb4ee6a4SAndroid Build Coastguard Worker
get_vgic_version(&self) -> DeviceKind142*bb4ee6a4SAndroid Build Coastguard Worker fn get_vgic_version(&self) -> DeviceKind {
143*bb4ee6a4SAndroid Build Coastguard Worker self.device_kind
144*bb4ee6a4SAndroid Build Coastguard Worker }
145*bb4ee6a4SAndroid Build Coastguard Worker
finalize(&self) -> Result<()>146*bb4ee6a4SAndroid Build Coastguard Worker fn finalize(&self) -> Result<()> {
147*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
148*bb4ee6a4SAndroid Build Coastguard Worker }
149*bb4ee6a4SAndroid Build Coastguard Worker }
150*bb4ee6a4SAndroid Build Coastguard Worker
151*bb4ee6a4SAndroid Build Coastguard Worker /// This IrqChip only works with Geniezone so we only implement it for GeniezoneVcpu.
152*bb4ee6a4SAndroid Build Coastguard Worker impl IrqChip for GeniezoneKernelIrqChip {
153*bb4ee6a4SAndroid Build Coastguard Worker /// Add a vcpu to the irq chip.
add_vcpu(&mut self, vcpu_id: usize, vcpu: &dyn Vcpu) -> Result<()>154*bb4ee6a4SAndroid Build Coastguard Worker fn add_vcpu(&mut self, vcpu_id: usize, vcpu: &dyn Vcpu) -> Result<()> {
155*bb4ee6a4SAndroid Build Coastguard Worker let vcpu: &GeniezoneVcpu = vcpu
156*bb4ee6a4SAndroid Build Coastguard Worker .downcast_ref()
157*bb4ee6a4SAndroid Build Coastguard Worker .expect("GeniezoneKernelIrqChip::add_vcpu called with non-GeniezoneVcpu");
158*bb4ee6a4SAndroid Build Coastguard Worker self.vcpus.lock()[vcpu_id] = Some(vcpu.try_clone()?);
159*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
160*bb4ee6a4SAndroid Build Coastguard Worker }
161*bb4ee6a4SAndroid Build Coastguard Worker
162*bb4ee6a4SAndroid Build Coastguard Worker /// Register an event with edge-trigger semantic that can trigger an interrupt
163*bb4ee6a4SAndroid Build Coastguard Worker /// for a particular GSI.
register_edge_irq_event( &mut self, irq: u32, irq_event: &IrqEdgeEvent, _source: IrqEventSource, ) -> Result<Option<IrqEventIndex>>164*bb4ee6a4SAndroid Build Coastguard Worker fn register_edge_irq_event(
165*bb4ee6a4SAndroid Build Coastguard Worker &mut self,
166*bb4ee6a4SAndroid Build Coastguard Worker irq: u32,
167*bb4ee6a4SAndroid Build Coastguard Worker irq_event: &IrqEdgeEvent,
168*bb4ee6a4SAndroid Build Coastguard Worker _source: IrqEventSource,
169*bb4ee6a4SAndroid Build Coastguard Worker ) -> Result<Option<IrqEventIndex>> {
170*bb4ee6a4SAndroid Build Coastguard Worker self.vm.register_irqfd(irq, irq_event.get_trigger(), None)?;
171*bb4ee6a4SAndroid Build Coastguard Worker Ok(None)
172*bb4ee6a4SAndroid Build Coastguard Worker }
173*bb4ee6a4SAndroid Build Coastguard Worker
174*bb4ee6a4SAndroid Build Coastguard Worker /// Unregister an event with edge-trigger semantic for a particular GSI.
unregister_edge_irq_event(&mut self, irq: u32, irq_event: &IrqEdgeEvent) -> Result<()>175*bb4ee6a4SAndroid Build Coastguard Worker fn unregister_edge_irq_event(&mut self, irq: u32, irq_event: &IrqEdgeEvent) -> Result<()> {
176*bb4ee6a4SAndroid Build Coastguard Worker self.vm.unregister_irqfd(irq, irq_event.get_trigger())
177*bb4ee6a4SAndroid Build Coastguard Worker }
178*bb4ee6a4SAndroid Build Coastguard Worker
179*bb4ee6a4SAndroid Build Coastguard Worker /// Register an event with level-trigger semantic that can trigger an interrupt
180*bb4ee6a4SAndroid Build Coastguard Worker /// for a particular GSI.
register_level_irq_event( &mut self, irq: u32, irq_event: &IrqLevelEvent, _source: IrqEventSource, ) -> Result<Option<IrqEventIndex>>181*bb4ee6a4SAndroid Build Coastguard Worker fn register_level_irq_event(
182*bb4ee6a4SAndroid Build Coastguard Worker &mut self,
183*bb4ee6a4SAndroid Build Coastguard Worker irq: u32,
184*bb4ee6a4SAndroid Build Coastguard Worker irq_event: &IrqLevelEvent,
185*bb4ee6a4SAndroid Build Coastguard Worker _source: IrqEventSource,
186*bb4ee6a4SAndroid Build Coastguard Worker ) -> Result<Option<IrqEventIndex>> {
187*bb4ee6a4SAndroid Build Coastguard Worker self.vm
188*bb4ee6a4SAndroid Build Coastguard Worker .register_irqfd(irq, irq_event.get_trigger(), Some(irq_event.get_resample()))?;
189*bb4ee6a4SAndroid Build Coastguard Worker Ok(None)
190*bb4ee6a4SAndroid Build Coastguard Worker }
191*bb4ee6a4SAndroid Build Coastguard Worker
192*bb4ee6a4SAndroid Build Coastguard Worker /// Unregister an event with level-trigger semantic for a particular GSI.
unregister_level_irq_event(&mut self, irq: u32, irq_event: &IrqLevelEvent) -> Result<()>193*bb4ee6a4SAndroid Build Coastguard Worker fn unregister_level_irq_event(&mut self, irq: u32, irq_event: &IrqLevelEvent) -> Result<()> {
194*bb4ee6a4SAndroid Build Coastguard Worker self.vm.unregister_irqfd(irq, irq_event.get_trigger())
195*bb4ee6a4SAndroid Build Coastguard Worker }
196*bb4ee6a4SAndroid Build Coastguard Worker
197*bb4ee6a4SAndroid Build Coastguard Worker /// Route an IRQ line to an interrupt controller, or to a particular MSI vector.
route_irq(&mut self, route: IrqRoute) -> Result<()>198*bb4ee6a4SAndroid Build Coastguard Worker fn route_irq(&mut self, route: IrqRoute) -> Result<()> {
199*bb4ee6a4SAndroid Build Coastguard Worker let mut routes = self.routes.lock();
200*bb4ee6a4SAndroid Build Coastguard Worker routes.retain(|r| r.gsi != route.gsi);
201*bb4ee6a4SAndroid Build Coastguard Worker
202*bb4ee6a4SAndroid Build Coastguard Worker routes.push(route);
203*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
204*bb4ee6a4SAndroid Build Coastguard Worker }
205*bb4ee6a4SAndroid Build Coastguard Worker
206*bb4ee6a4SAndroid Build Coastguard Worker /// Replace all irq routes with the supplied routes
set_irq_routes(&mut self, routes: &[IrqRoute]) -> Result<()>207*bb4ee6a4SAndroid Build Coastguard Worker fn set_irq_routes(&mut self, routes: &[IrqRoute]) -> Result<()> {
208*bb4ee6a4SAndroid Build Coastguard Worker let mut current_routes = self.routes.lock();
209*bb4ee6a4SAndroid Build Coastguard Worker *current_routes = routes.to_vec();
210*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
211*bb4ee6a4SAndroid Build Coastguard Worker }
212*bb4ee6a4SAndroid Build Coastguard Worker
213*bb4ee6a4SAndroid Build Coastguard Worker /// Return a vector of all registered irq numbers and their associated events and event
214*bb4ee6a4SAndroid Build Coastguard Worker /// indices. These should be used by the main thread to wait for irq events.
215*bb4ee6a4SAndroid Build Coastguard Worker /// For the GeniezoneKernelIrqChip, the kernel handles listening to irq events being triggered
216*bb4ee6a4SAndroid Build Coastguard Worker /// by devices, so this function always returns an empty Vec.
irq_event_tokens(&self) -> Result<Vec<(IrqEventIndex, IrqEventSource, Event)>>217*bb4ee6a4SAndroid Build Coastguard Worker fn irq_event_tokens(&self) -> Result<Vec<(IrqEventIndex, IrqEventSource, Event)>> {
218*bb4ee6a4SAndroid Build Coastguard Worker Ok(Vec::new())
219*bb4ee6a4SAndroid Build Coastguard Worker }
220*bb4ee6a4SAndroid Build Coastguard Worker
221*bb4ee6a4SAndroid Build Coastguard Worker /// Either assert or deassert an IRQ line. Sends to either an interrupt controller, or does
222*bb4ee6a4SAndroid Build Coastguard Worker /// a send_msi if the irq is associated with an MSI.
223*bb4ee6a4SAndroid Build Coastguard Worker /// For the GeniezoneKernelIrqChip this simply calls the GZVM_SET_IRQ_LINE ioctl.
service_irq(&mut self, irq: u32, level: bool) -> Result<()>224*bb4ee6a4SAndroid Build Coastguard Worker fn service_irq(&mut self, irq: u32, level: bool) -> Result<()> {
225*bb4ee6a4SAndroid Build Coastguard Worker self.vm.set_irq_line(irq, level)
226*bb4ee6a4SAndroid Build Coastguard Worker }
227*bb4ee6a4SAndroid Build Coastguard Worker
228*bb4ee6a4SAndroid Build Coastguard Worker /// Service an IRQ event by asserting then deasserting an IRQ line. The associated Event
229*bb4ee6a4SAndroid Build Coastguard Worker /// that triggered the irq event will be read from. If the irq is associated with a resample
230*bb4ee6a4SAndroid Build Coastguard Worker /// Event, then the deassert will only happen after an EOI is broadcast for a vector
231*bb4ee6a4SAndroid Build Coastguard Worker /// associated with the irq line.
232*bb4ee6a4SAndroid Build Coastguard Worker /// This function should never be called on GeniezoneKernelIrqChip.
service_irq_event(&mut self, _event_index: IrqEventIndex) -> Result<()>233*bb4ee6a4SAndroid Build Coastguard Worker fn service_irq_event(&mut self, _event_index: IrqEventIndex) -> Result<()> {
234*bb4ee6a4SAndroid Build Coastguard Worker error!("service_irq_event should never be called for GeniezoneKernelIrqChip");
235*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
236*bb4ee6a4SAndroid Build Coastguard Worker }
237*bb4ee6a4SAndroid Build Coastguard Worker
238*bb4ee6a4SAndroid Build Coastguard Worker /// Broadcast an end of interrupt.
239*bb4ee6a4SAndroid Build Coastguard Worker /// This should never be called on a GeniezoneKernelIrqChip because a Geniezone vcpu should
240*bb4ee6a4SAndroid Build Coastguard Worker /// never exit with the GZVM_EXIT_EOI_BROADCAST reason when an in-kernel irqchip exists.
broadcast_eoi(&self, _vector: u8) -> Result<()>241*bb4ee6a4SAndroid Build Coastguard Worker fn broadcast_eoi(&self, _vector: u8) -> Result<()> {
242*bb4ee6a4SAndroid Build Coastguard Worker error!("broadcast_eoi should never be called for GeniezoneKernelIrqChip");
243*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
244*bb4ee6a4SAndroid Build Coastguard Worker }
245*bb4ee6a4SAndroid Build Coastguard Worker
246*bb4ee6a4SAndroid Build Coastguard Worker /// Injects any pending interrupts for `vcpu`.
247*bb4ee6a4SAndroid Build Coastguard Worker /// For GeniezoneKernelIrqChip this is a no-op because Geniezone is responsible for injecting
248*bb4ee6a4SAndroid Build Coastguard Worker /// all interrupts.
inject_interrupts(&self, _vcpu: &dyn Vcpu) -> Result<()>249*bb4ee6a4SAndroid Build Coastguard Worker fn inject_interrupts(&self, _vcpu: &dyn Vcpu) -> Result<()> {
250*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
251*bb4ee6a4SAndroid Build Coastguard Worker }
252*bb4ee6a4SAndroid Build Coastguard Worker
253*bb4ee6a4SAndroid Build Coastguard Worker /// Notifies the irq chip that the specified VCPU has executed a halt instruction.
254*bb4ee6a4SAndroid Build Coastguard Worker /// For GeniezoneKernelIrqChip this is a no-op because Geniezone handles VCPU blocking.
halted(&self, _vcpu_id: usize)255*bb4ee6a4SAndroid Build Coastguard Worker fn halted(&self, _vcpu_id: usize) {}
256*bb4ee6a4SAndroid Build Coastguard Worker
257*bb4ee6a4SAndroid Build Coastguard Worker /// Blocks until `vcpu` is in a runnable state or until interrupted by
258*bb4ee6a4SAndroid Build Coastguard Worker /// `IrqChip::kick_halted_vcpus`. Returns `VcpuRunState::Runnable if vcpu is runnable, or
259*bb4ee6a4SAndroid Build Coastguard Worker /// `VcpuRunState::Interrupted` if the wait was interrupted.
260*bb4ee6a4SAndroid Build Coastguard Worker /// For GeniezoneKernelIrqChip this is a no-op and always returns Runnable because Geniezone
261*bb4ee6a4SAndroid Build Coastguard Worker /// handles VCPU blocking.
wait_until_runnable(&self, _vcpu: &dyn Vcpu) -> Result<VcpuRunState>262*bb4ee6a4SAndroid Build Coastguard Worker fn wait_until_runnable(&self, _vcpu: &dyn Vcpu) -> Result<VcpuRunState> {
263*bb4ee6a4SAndroid Build Coastguard Worker Ok(VcpuRunState::Runnable)
264*bb4ee6a4SAndroid Build Coastguard Worker }
265*bb4ee6a4SAndroid Build Coastguard Worker
266*bb4ee6a4SAndroid Build Coastguard Worker /// Makes unrunnable VCPUs return immediately from `wait_until_runnable`.
267*bb4ee6a4SAndroid Build Coastguard Worker /// For GeniezoneKernelIrqChip this is a no-op because Geniezone handles VCPU blocking.
kick_halted_vcpus(&self)268*bb4ee6a4SAndroid Build Coastguard Worker fn kick_halted_vcpus(&self) {}
269*bb4ee6a4SAndroid Build Coastguard Worker
270*bb4ee6a4SAndroid Build Coastguard Worker /// Get the current MP state of the specified VCPU.
get_mp_state(&self, _vcpu_id: usize) -> Result<MPState>271*bb4ee6a4SAndroid Build Coastguard Worker fn get_mp_state(&self, _vcpu_id: usize) -> Result<MPState> {
272*bb4ee6a4SAndroid Build Coastguard Worker Err(Error::new(libc::ENOENT))
273*bb4ee6a4SAndroid Build Coastguard Worker }
274*bb4ee6a4SAndroid Build Coastguard Worker
275*bb4ee6a4SAndroid Build Coastguard Worker /// Set the current MP state of the specified VCPU.
set_mp_state(&mut self, _vcpu_id: usize, _state: &MPState) -> Result<()>276*bb4ee6a4SAndroid Build Coastguard Worker fn set_mp_state(&mut self, _vcpu_id: usize, _state: &MPState) -> Result<()> {
277*bb4ee6a4SAndroid Build Coastguard Worker Err(Error::new(libc::ENOENT))
278*bb4ee6a4SAndroid Build Coastguard Worker }
279*bb4ee6a4SAndroid Build Coastguard Worker
280*bb4ee6a4SAndroid Build Coastguard Worker /// Attempt to clone this IrqChip instance.
try_clone(&self) -> Result<Self>281*bb4ee6a4SAndroid Build Coastguard Worker fn try_clone(&self) -> Result<Self> {
282*bb4ee6a4SAndroid Build Coastguard Worker // Because the GeniezoneKernelIrqChip struct contains arch-specific fields we leave the
283*bb4ee6a4SAndroid Build Coastguard Worker // cloning to arch-specific implementations
284*bb4ee6a4SAndroid Build Coastguard Worker self.arch_try_clone()
285*bb4ee6a4SAndroid Build Coastguard Worker }
286*bb4ee6a4SAndroid Build Coastguard Worker
287*bb4ee6a4SAndroid Build Coastguard Worker /// Finalize irqchip setup. Should be called once all devices have registered irq events and
288*bb4ee6a4SAndroid Build Coastguard Worker /// been added to the io_bus and mmio_bus.
289*bb4ee6a4SAndroid Build Coastguard Worker /// GeniezoneKernelIrqChip does not need to do anything here.
finalize_devices( &mut self, _resources: &mut SystemAllocator, _io_bus: &Bus, _mmio_bus: &Bus, ) -> Result<()>290*bb4ee6a4SAndroid Build Coastguard Worker fn finalize_devices(
291*bb4ee6a4SAndroid Build Coastguard Worker &mut self,
292*bb4ee6a4SAndroid Build Coastguard Worker _resources: &mut SystemAllocator,
293*bb4ee6a4SAndroid Build Coastguard Worker _io_bus: &Bus,
294*bb4ee6a4SAndroid Build Coastguard Worker _mmio_bus: &Bus,
295*bb4ee6a4SAndroid Build Coastguard Worker ) -> Result<()> {
296*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
297*bb4ee6a4SAndroid Build Coastguard Worker }
298*bb4ee6a4SAndroid Build Coastguard Worker
299*bb4ee6a4SAndroid Build Coastguard Worker /// The GeniezoneKernelIrqChip doesn't process irq events itself so this function does nothing.
process_delayed_irq_events(&mut self) -> Result<()>300*bb4ee6a4SAndroid Build Coastguard Worker fn process_delayed_irq_events(&mut self) -> Result<()> {
301*bb4ee6a4SAndroid Build Coastguard Worker Ok(())
302*bb4ee6a4SAndroid Build Coastguard Worker }
303*bb4ee6a4SAndroid Build Coastguard Worker
irq_delayed_event_token(&self) -> Result<Option<Event>>304*bb4ee6a4SAndroid Build Coastguard Worker fn irq_delayed_event_token(&self) -> Result<Option<Event>> {
305*bb4ee6a4SAndroid Build Coastguard Worker Ok(None)
306*bb4ee6a4SAndroid Build Coastguard Worker }
307*bb4ee6a4SAndroid Build Coastguard Worker
check_capability(&self, _c: IrqChipCap) -> bool308*bb4ee6a4SAndroid Build Coastguard Worker fn check_capability(&self, _c: IrqChipCap) -> bool {
309*bb4ee6a4SAndroid Build Coastguard Worker false
310*bb4ee6a4SAndroid Build Coastguard Worker }
311*bb4ee6a4SAndroid Build Coastguard Worker }
312