hypervisor: kvm: Introduce AIA creation

Implement definition required to work with KVM in-kernel AIA device,
construction procedure of AIA.

Signed-off-by: Ruoqing He <heruoqing@iscas.ac.cn>
This commit is contained in:
Ruoqing He 2024-10-09 20:41:15 +08:00 committed by Rob Bradford
parent 2df8d2fa1e
commit c5774685f4
2 changed files with 275 additions and 0 deletions

View File

@ -0,0 +1,273 @@
// Copyright © 2024 Institute of Software, CAS. All rights reserved.
//
// SPDX-License-Identifier: Apache-2.0
use std::any::Any;
use kvm_ioctls::DeviceFd;
use serde::{Deserialize, Serialize};
use crate::arch::riscv64::aia::{Error, Result, Vaia, VaiaConfig};
use crate::device::HypervisorDeviceError;
use crate::kvm::KvmVm;
use crate::Vm;
pub struct KvmAiaImsics {
/// The KVM device for the Aia
device: DeviceFd,
/// AIA APLIC address
aplic_addr: u64,
/// AIA APLIC size
aplic_size: u64,
/// AIA IMSIC address
imsic_addr: u64,
/// AIA IMSIC size
imsic_size: u64,
/// Number of CPUs handled by the device
vcpu_count: u64,
}
#[derive(Clone, Default, Serialize, Deserialize)]
pub struct AiaImsicsState {}
impl KvmAiaImsics {
/// Device trees specific constants
fn version() -> u32 {
kvm_bindings::kvm_device_type_KVM_DEV_TYPE_RISCV_AIA
}
/// Setup the device-specific attributes
fn init_device_attributes(&mut self, nr_irqs: u32) -> Result<()> {
// AIA part attributes
// Getting the working mode of RISC-V AIA, defaults to EMUL, passible
// variants are EMUL, HW_ACCL, AUTO
let mut aia_mode = kvm_bindings::KVM_DEV_RISCV_AIA_MODE_EMUL;
Self::get_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_CONFIG,
u64::from(kvm_bindings::KVM_DEV_RISCV_AIA_CONFIG_MODE),
&mut aia_mode as *mut u32 as u64,
0,
)?;
// Report AIA MODE
// Setting up the number of wired interrupt sources
Self::set_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_CONFIG,
u64::from(kvm_bindings::KVM_DEV_RISCV_AIA_CONFIG_SRCS),
&nr_irqs as *const u32 as u64,
0,
)?;
// Getting the number of ids
let mut aia_nr_ids: u32 = 0;
Self::get_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_CONFIG,
u64::from(kvm_bindings::KVM_DEV_RISCV_AIA_CONFIG_IDS),
&mut aia_nr_ids as *mut u32 as u64,
0,
)?;
// Report NR_IDS
// Setting up hart_bits
let max_hart_index = self.vcpu_count - 1;
let hart_bits = std::cmp::max(64 - max_hart_index.leading_zeros(), 1);
Self::set_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_CONFIG,
u64::from(kvm_bindings::KVM_DEV_RISCV_AIA_CONFIG_HART_BITS),
&hart_bits as *const u32 as u64,
0,
)?;
// Designate addresses of APLIC and IMSICS
// Setting up RISC-V APLIC
Self::set_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_ADDR,
u64::from(kvm_bindings::KVM_DEV_RISCV_AIA_ADDR_APLIC),
&self.aplic_addr as *const u64 as u64,
0,
)?;
// Helpers to calculate address and attribute of IMSIC of each vCPU
let riscv_imsic_addr_of =
|cpu_index: u64| -> u64 { self.imsic_addr + cpu_index * self.imsic_size };
let riscv_imsic_attr_of = |cpu_index: u64| -> u64 { cpu_index };
// Setting up RISC-V IMSICs
for cpu_index in 0..self.vcpu_count {
let cpu_imsic_addr = riscv_imsic_addr_of(cpu_index);
Self::set_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_ADDR,
riscv_imsic_attr_of(cpu_index),
&cpu_imsic_addr as *const u64 as u64,
0,
)?;
}
// Finalizing the AIA device
Self::set_device_attribute(
&self.device,
kvm_bindings::KVM_DEV_RISCV_AIA_GRP_CTRL,
u64::from(kvm_bindings::KVM_DEV_RISCV_AIA_CTRL_INIT),
0,
0,
)
}
/// Create a KVM Vaia device
fn create_device(vm: &KvmVm) -> Result<DeviceFd> {
let mut aia_device = kvm_bindings::kvm_create_device {
type_: Self::version(),
fd: 0,
flags: 0,
};
let device_fd = vm
.create_device(&mut aia_device)
.map_err(Error::CreateAia)?;
// We know for sure this is a KVM fd
Ok(device_fd.to_kvm().unwrap())
}
/// Get an AIA device attribute
fn get_device_attribute(
device: &DeviceFd,
group: u32,
attr: u64,
addr: u64,
flags: u32,
) -> Result<()> {
let mut attr = kvm_bindings::kvm_device_attr {
flags,
group,
attr,
addr,
};
// SAFETY: attr.addr is safe to write to.
unsafe {
device.get_device_attr(&mut attr).map_err(|e| {
Error::GetDeviceAttribute(HypervisorDeviceError::GetDeviceAttribute(e.into()))
})
}
}
/// Set an AIA device attribute
fn set_device_attribute(
device: &DeviceFd,
group: u32,
attr: u64,
addr: u64,
flags: u32,
) -> Result<()> {
let attr = kvm_bindings::kvm_device_attr {
flags,
group,
attr,
addr,
};
device.set_device_attr(&attr).map_err(|e| {
Error::SetDeviceAttribute(HypervisorDeviceError::SetDeviceAttribute(e.into()))
})
}
/// Method to initialize the AIA device
pub fn new(vm: &dyn Vm, config: VaiaConfig) -> Result<KvmAiaImsics> {
// This is inside KVM module
let vm = vm.as_any().downcast_ref::<KvmVm>().expect("Wrong VM type?");
let vaia = Self::create_device(vm)?;
let mut aia_device = KvmAiaImsics {
device: vaia,
vcpu_count: config.vcpu_count,
aplic_addr: config.aplic_addr,
aplic_size: config.aplic_size,
imsic_addr: config.imsic_addr,
imsic_size: config.imsic_size,
};
aia_device.init_device_attributes(config.nr_irqs)?;
Ok(aia_device)
}
}
impl Vaia for KvmAiaImsics {
fn aplic_compatibility(&self) -> &str {
"riscv,aplic"
}
fn aplic_properties(&self) -> [u64; 4] {
[0, self.aplic_addr, 0, self.aplic_size]
}
fn imsic_compatibility(&self) -> &str {
"riscv,imsics"
}
fn imsic_properties(&self) -> [u64; 4] {
[0, self.imsic_addr, 0, self.imsic_size * self.vcpu_count]
}
fn vcpu_count(&self) -> u64 {
self.vcpu_count
}
fn msi_compatible(&self) -> bool {
true
}
fn as_any_concrete_mut(&mut self) -> &mut dyn Any {
self
}
/// Save the state of AIA.
fn state(&self) -> Result<AiaImsicsState> {
unimplemented!()
}
/// Restore the state of AIA_IMSICs.
fn set_state(&mut self, _state: &AiaImsicsState) -> Result<()> {
unimplemented!()
}
}
#[cfg(test)]
mod tests {
use crate::arch::riscv64::aia::VaiaConfig;
use crate::kvm::KvmAiaImsics;
fn create_test_vaia_config() -> VaiaConfig {
VaiaConfig {
vcpu_count: 1,
aplic_addr: 0xd000000,
aplic_size: 0x4000,
imsic_addr: 0x2800000,
imsic_size: 0x1000,
nr_irqs: 256,
}
}
#[test]
fn test_create_aia() {
let hv = crate::new().unwrap();
let vm = hv.create_vm().unwrap();
assert!(KvmAiaImsics::new(&*vm, create_test_vaia_config()).is_ok());
}
}

View File

@ -2,6 +2,8 @@
// //
// SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: Apache-2.0
pub mod aia;
use kvm_bindings::{ use kvm_bindings::{
kvm_mp_state, kvm_one_reg, kvm_riscv_core, KVM_REG_RISCV_CORE, KVM_REG_RISCV_TYPE_MASK, kvm_mp_state, kvm_one_reg, kvm_riscv_core, KVM_REG_RISCV_CORE, KVM_REG_RISCV_TYPE_MASK,
KVM_REG_SIZE_MASK, KVM_REG_SIZE_U64, KVM_REG_SIZE_MASK, KVM_REG_SIZE_U64,