mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-03-20 07:58:55 +00:00
arch: AArch64: Porting GIC redist_regs implementation
This commit ports the implementation of GIC redistributor registers from Firecracker. Signed-off-by: Henry Wang <Henry.Wang@arm.com>
This commit is contained in:
parent
f53990c7e7
commit
bfde6977c8
@ -5,8 +5,10 @@ pub mod dist_regs;
|
|||||||
mod gicv2;
|
mod gicv2;
|
||||||
mod gicv3;
|
mod gicv3;
|
||||||
mod gicv3_its;
|
mod gicv3_its;
|
||||||
|
pub mod redist_regs;
|
||||||
|
|
||||||
pub use self::dist_regs::{get_dist_regs, read_ctlr, set_dist_regs, write_ctlr};
|
pub use self::dist_regs::{get_dist_regs, read_ctlr, set_dist_regs, write_ctlr};
|
||||||
|
pub use self::redist_regs::{get_redist_regs, set_redist_regs};
|
||||||
use std::result;
|
use std::result;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
179
arch/src/aarch64/gic/redist_regs.rs
Normal file
179
arch/src/aarch64/gic/redist_regs.rs
Normal file
@ -0,0 +1,179 @@
|
|||||||
|
// Copyright 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use super::{Error, Result};
|
||||||
|
use hypervisor::kvm::kvm_bindings::{kvm_device_attr, KVM_DEV_ARM_VGIC_GRP_REDIST_REGS};
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
// Relevant redistributor registers that we want to save/restore.
|
||||||
|
const GICR_CTLR: u32 = 0x0000;
|
||||||
|
const GICR_STATUSR: u32 = 0x0010;
|
||||||
|
const GICR_WAKER: u32 = 0x0014;
|
||||||
|
const GICR_PROPBASER: u32 = 0x0070;
|
||||||
|
const GICR_PENDBASER: u32 = 0x0078;
|
||||||
|
|
||||||
|
/* SGI and PPI Redistributor registers, offsets from RD_base */
|
||||||
|
/*
|
||||||
|
* Redistributor frame offsets from RD_base which is actually SZ_
|
||||||
|
*/
|
||||||
|
const GICR_SGI_OFFSET: u32 = 0x0001_0000;
|
||||||
|
const GICR_IGROUPR0: u32 = GICR_SGI_OFFSET + 0x0080;
|
||||||
|
const GICR_ICENABLER0: u32 = GICR_SGI_OFFSET + 0x0180;
|
||||||
|
const GICR_ISENABLER0: u32 = GICR_SGI_OFFSET + 0x0100;
|
||||||
|
const GICR_ISPENDR0: u32 = GICR_SGI_OFFSET + 0x0200;
|
||||||
|
const GICR_ICPENDR0: u32 = GICR_SGI_OFFSET + 0x0280;
|
||||||
|
const GICR_ISACTIVER0: u32 = GICR_SGI_OFFSET + 0x0300;
|
||||||
|
const GICR_ICACTIVER0: u32 = GICR_SGI_OFFSET + 0x0380;
|
||||||
|
const GICR_IPRIORITYR0: u32 = GICR_SGI_OFFSET + 0x0400;
|
||||||
|
const GICR_ICFGR0: u32 = GICR_SGI_OFFSET + 0x0C00;
|
||||||
|
|
||||||
|
const KVM_DEV_ARM_VGIC_V3_MPIDR_SHIFT: u32 = 32;
|
||||||
|
const KVM_DEV_ARM_VGIC_V3_MPIDR_MASK: u64 = 0xffffffff << KVM_DEV_ARM_VGIC_V3_MPIDR_SHIFT as u64;
|
||||||
|
|
||||||
|
/// This is how we represent the registers of a distributor.
|
||||||
|
/// It is relrvant their offset from the base address of the
|
||||||
|
/// distributor.
|
||||||
|
/// Each register has a different number
|
||||||
|
/// of bits_per_irq and is therefore variable length.
|
||||||
|
/// First 32 interrupts (0-32) are private to each CPU (SGIs and PPIs).
|
||||||
|
/// and so we save the first irq to identify between the type of the interrupt
|
||||||
|
/// that the respective register deals with.
|
||||||
|
struct RdistReg {
|
||||||
|
/// Offset from distributor address.
|
||||||
|
base: u32,
|
||||||
|
/// Length of the register.
|
||||||
|
length: u8,
|
||||||
|
}
|
||||||
|
|
||||||
|
// All or at least the registers we are interested in are 32 bit, so
|
||||||
|
// we use a constant for size(u32).
|
||||||
|
const REG_SIZE: u8 = 4;
|
||||||
|
|
||||||
|
// Creates a vgic redistributor register.
|
||||||
|
macro_rules! VGIC_RDIST_REG {
|
||||||
|
($base:expr, $len:expr) => {
|
||||||
|
RdistReg {
|
||||||
|
base: $base,
|
||||||
|
length: $len,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// List with relevant distributor registers that we will be restoring.
|
||||||
|
static VGIC_RDIST_REGS: &'static [RdistReg] = &[
|
||||||
|
VGIC_RDIST_REG!(GICR_CTLR, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_STATUSR, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_WAKER, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_PROPBASER, 8),
|
||||||
|
VGIC_RDIST_REG!(GICR_PENDBASER, 8),
|
||||||
|
];
|
||||||
|
|
||||||
|
// List with relevant distributor registers that we will be restoring.
|
||||||
|
static VGIC_SGI_REGS: &'static [RdistReg] = &[
|
||||||
|
VGIC_RDIST_REG!(GICR_IGROUPR0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_ICENABLER0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_ISENABLER0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_ICFGR0, 8),
|
||||||
|
VGIC_RDIST_REG!(GICR_ICPENDR0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_ISPENDR0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_ICACTIVER0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_ISACTIVER0, 4),
|
||||||
|
VGIC_RDIST_REG!(GICR_IPRIORITYR0, 32),
|
||||||
|
];
|
||||||
|
|
||||||
|
fn redist_attr_access(
|
||||||
|
gic: &Arc<dyn hypervisor::Device>,
|
||||||
|
offset: u32,
|
||||||
|
typer: u64,
|
||||||
|
val: &u32,
|
||||||
|
set: bool,
|
||||||
|
) -> Result<()> {
|
||||||
|
let mut gic_dist_attr = kvm_device_attr {
|
||||||
|
group: KVM_DEV_ARM_VGIC_GRP_REDIST_REGS,
|
||||||
|
attr: (typer & KVM_DEV_ARM_VGIC_V3_MPIDR_MASK) | (offset as u64), // this needs the mpidr
|
||||||
|
addr: val as *const u32 as u64,
|
||||||
|
flags: 0,
|
||||||
|
};
|
||||||
|
if set {
|
||||||
|
#[allow(clippy::unnecessary_mut_passed)]
|
||||||
|
gic.set_device_attr(&mut gic_dist_attr)
|
||||||
|
.map_err(Error::SetDeviceAttribute)?;
|
||||||
|
} else {
|
||||||
|
gic.get_device_attr(&mut gic_dist_attr)
|
||||||
|
.map_err(Error::GetDeviceAttribute)?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn access_redists_aux(
|
||||||
|
gic: &Arc<dyn hypervisor::Device>,
|
||||||
|
gicr_typer: &[u64],
|
||||||
|
state: &mut Vec<u32>,
|
||||||
|
reg_list: &'static [RdistReg],
|
||||||
|
idx: &mut usize,
|
||||||
|
set: bool,
|
||||||
|
) -> Result<()> {
|
||||||
|
for i in gicr_typer {
|
||||||
|
for rdreg in reg_list {
|
||||||
|
let mut base = rdreg.base;
|
||||||
|
let end = base + rdreg.length as u32;
|
||||||
|
|
||||||
|
while base < end {
|
||||||
|
let mut val = 0;
|
||||||
|
if set {
|
||||||
|
val = state[*idx];
|
||||||
|
redist_attr_access(gic, base, *i, &val, true)?;
|
||||||
|
*idx += 1;
|
||||||
|
} else {
|
||||||
|
redist_attr_access(gic, base, *i, &val, false)?;
|
||||||
|
state.push(val);
|
||||||
|
}
|
||||||
|
base += REG_SIZE as u32;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get redistributor registers.
|
||||||
|
pub fn get_redist_regs(gic: &Arc<dyn hypervisor::Device>, gicr_typer: &[u64]) -> Result<Vec<u32>> {
|
||||||
|
let mut state = Vec::new();
|
||||||
|
let mut idx: usize = 0;
|
||||||
|
access_redists_aux(
|
||||||
|
gic,
|
||||||
|
&gicr_typer,
|
||||||
|
&mut state,
|
||||||
|
VGIC_RDIST_REGS,
|
||||||
|
&mut idx,
|
||||||
|
false,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
access_redists_aux(gic, &gicr_typer, &mut state, VGIC_SGI_REGS, &mut idx, false)?;
|
||||||
|
Ok(state)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set redistributor registers.
|
||||||
|
pub fn set_redist_regs(
|
||||||
|
gic: &Arc<dyn hypervisor::Device>,
|
||||||
|
gicr_typer: &[u64],
|
||||||
|
state: &[u32],
|
||||||
|
) -> Result<()> {
|
||||||
|
let mut idx: usize = 0;
|
||||||
|
let mut mut_state = state.to_owned();
|
||||||
|
access_redists_aux(
|
||||||
|
gic,
|
||||||
|
gicr_typer,
|
||||||
|
&mut mut_state,
|
||||||
|
VGIC_RDIST_REGS,
|
||||||
|
&mut idx,
|
||||||
|
true,
|
||||||
|
)?;
|
||||||
|
access_redists_aux(
|
||||||
|
gic,
|
||||||
|
gicr_typer,
|
||||||
|
&mut mut_state,
|
||||||
|
VGIC_SGI_REGS,
|
||||||
|
&mut idx,
|
||||||
|
true,
|
||||||
|
)
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user