vmm: Enable device manager for AArch64

Screened IO bus because it is not for AArch64.
Enabled Serial, RTC and Virtio devices with MMIO transport option.

Signed-off-by: Michael Zhao <michael.zhao@arm.com>
This commit is contained in:
Michael Zhao 2020-06-09 14:17:42 +08:00 committed by Rob Bradford
parent e9488846f1
commit eeeb45bbb9
2 changed files with 241 additions and 57 deletions

View File

@ -230,7 +230,7 @@ struct InterruptSourceOverride {
pub struct Vcpu { pub struct Vcpu {
fd: VcpuFd, fd: VcpuFd,
id: u8, id: u8,
#[cfg_attr(target_arch = "aarch64", allow(dead_code))] #[cfg(target_arch = "x86_64")]
io_bus: Arc<devices::Bus>, io_bus: Arc<devices::Bus>,
mmio_bus: Arc<devices::Bus>, mmio_bus: Arc<devices::Bus>,
#[cfg_attr(target_arch = "aarch64", allow(dead_code))] #[cfg_attr(target_arch = "aarch64", allow(dead_code))]
@ -267,7 +267,7 @@ impl Vcpu {
pub fn new( pub fn new(
id: u8, id: u8,
fd: &Arc<VmFd>, fd: &Arc<VmFd>,
io_bus: Arc<devices::Bus>, #[cfg(target_arch = "x86_64")] io_bus: Arc<devices::Bus>,
mmio_bus: Arc<devices::Bus>, mmio_bus: Arc<devices::Bus>,
interrupt_controller: Option<Arc<Mutex<dyn InterruptController>>>, interrupt_controller: Option<Arc<Mutex<dyn InterruptController>>>,
creation_ts: std::time::Instant, creation_ts: std::time::Instant,
@ -277,6 +277,7 @@ impl Vcpu {
Ok(Arc::new(Mutex::new(Vcpu { Ok(Arc::new(Mutex::new(Vcpu {
fd: kvm_vcpu, fd: kvm_vcpu,
id, id,
#[cfg(target_arch = "x86_64")]
io_bus, io_bus,
mmio_bus, mmio_bus,
interrupt_controller, interrupt_controller,
@ -504,6 +505,7 @@ impl Snapshottable for Vcpu {
pub struct CpuManager { pub struct CpuManager {
boot_vcpus: u8, boot_vcpus: u8,
max_vcpus: u8, max_vcpus: u8,
#[cfg(target_arch = "x86_64")]
io_bus: Arc<devices::Bus>, io_bus: Arc<devices::Bus>,
#[cfg_attr(target_arch = "aarch64", allow(dead_code))] #[cfg_attr(target_arch = "aarch64", allow(dead_code))]
mmio_bus: Arc<devices::Bus>, mmio_bus: Arc<devices::Bus>,
@ -657,6 +659,7 @@ impl CpuManager {
let cpu_manager = Arc::new(Mutex::new(CpuManager { let cpu_manager = Arc::new(Mutex::new(CpuManager {
boot_vcpus: config.boot_vcpus, boot_vcpus: config.boot_vcpus,
max_vcpus: config.max_vcpus, max_vcpus: config.max_vcpus,
#[cfg(target_arch = "x86_64")]
io_bus: device_manager.io_bus().clone(), io_bus: device_manager.io_bus().clone(),
mmio_bus: device_manager.mmio_bus().clone(), mmio_bus: device_manager.mmio_bus().clone(),
interrupt_controller: device_manager.interrupt_controller().clone(), interrupt_controller: device_manager.interrupt_controller().clone(),
@ -679,6 +682,7 @@ impl CpuManager {
.allocate_io_addresses(Some(GuestAddress(0x0cd8)), 0x8, None) .allocate_io_addresses(Some(GuestAddress(0x0cd8)), 0x8, None)
.ok_or(Error::AllocateIOPort)?; .ok_or(Error::AllocateIOPort)?;
#[cfg(target_arch = "x86_64")]
cpu_manager cpu_manager
.lock() .lock()
.unwrap() .unwrap()
@ -742,6 +746,7 @@ impl CpuManager {
let vcpu = Vcpu::new( let vcpu = Vcpu::new(
cpu_id, cpu_id,
&self.fd, &self.fd,
#[cfg(target_arch = "x86_64")]
self.io_bus.clone(), self.io_bus.clone(),
self.mmio_bus.clone(), self.mmio_bus.clone(),
interrupt_controller, interrupt_controller,

View File

@ -22,16 +22,20 @@ use crate::{device_node, DEVICE_MANAGER_SNAPSHOT_ID};
#[cfg(feature = "acpi")] #[cfg(feature = "acpi")]
use acpi_tables::{aml, aml::Aml}; use acpi_tables::{aml, aml::Aml};
use anyhow::anyhow; use anyhow::anyhow;
#[cfg(target_arch = "aarch64")]
use arch::aarch64::DeviceInfoForFDT;
#[cfg(feature = "acpi")] #[cfg(feature = "acpi")]
use arch::layout; use arch::layout;
#[cfg(target_arch = "x86_64")] #[cfg(target_arch = "x86_64")]
use arch::layout::{APIC_START, IOAPIC_SIZE, IOAPIC_START}; use arch::layout::{APIC_START, IOAPIC_SIZE, IOAPIC_START};
#[cfg(target_arch = "aarch64")] #[cfg(target_arch = "aarch64")]
use arch::DeviceType;
#[cfg(target_arch = "aarch64")]
use devices::gic; use devices::gic;
#[cfg(target_arch = "x86_64")] #[cfg(target_arch = "x86_64")]
use devices::ioapic; use devices::ioapic;
use devices::{ use devices::{
interrupt_controller, interrupt_controller::InterruptController, BusDevice, interrupt_controller, interrupt_controller::InterruptController, legacy::Serial, BusDevice,
HotPlugNotificationFlags, HotPlugNotificationFlags,
}; };
use kvm_ioctls::*; use kvm_ioctls::*;
@ -382,7 +386,7 @@ pub fn get_win_size() -> (u16, u16) {
#[derive(Default)] #[derive(Default)]
pub struct Console { pub struct Console {
// Serial port on 0x3f8 // Serial port on 0x3f8
serial: Option<Arc<Mutex<devices::legacy::Serial>>>, serial: Option<Arc<Mutex<Serial>>>,
console_input: Option<Arc<vm_virtio::ConsoleInput>>, console_input: Option<Arc<vm_virtio::ConsoleInput>>,
input_enabled: bool, input_enabled: bool,
} }
@ -421,6 +425,7 @@ impl Console {
struct AddressManager { struct AddressManager {
allocator: Arc<Mutex<SystemAllocator>>, allocator: Arc<Mutex<SystemAllocator>>,
#[cfg(target_arch = "x86_64")]
io_bus: Arc<devices::Bus>, io_bus: Arc<devices::Bus>,
mmio_bus: Arc<devices::Bus>, mmio_bus: Arc<devices::Bus>,
vm_fd: Arc<VmFd>, vm_fd: Arc<VmFd>,
@ -630,6 +635,28 @@ struct DeviceManagerState {
device_id_cnt: Wrapping<usize>, device_id_cnt: Wrapping<usize>,
} }
/// Private structure for storing information about the MMIO device registered at some address on the bus.
#[derive(Clone, Debug)]
#[cfg(target_arch = "aarch64")]
pub struct MMIODeviceInfo {
addr: u64,
irq: u32,
len: u64,
}
#[cfg(target_arch = "aarch64")]
impl DeviceInfoForFDT for MMIODeviceInfo {
fn addr(&self) -> u64 {
self.addr
}
fn irq(&self) -> u32 {
self.irq
}
fn length(&self) -> u64 {
self.len
}
}
pub struct DeviceManager { pub struct DeviceManager {
// Manage address space related to devices // Manage address space related to devices
address_manager: Arc<AddressManager>, address_manager: Arc<AddressManager>,
@ -713,8 +740,13 @@ pub struct DeviceManager {
// Exit event // Exit event
#[cfg(feature = "acpi")] #[cfg(feature = "acpi")]
exit_evt: EventFd, exit_evt: EventFd,
// Reset event // Reset event
#[cfg(target_arch = "x86_64")]
reset_evt: EventFd, reset_evt: EventFd,
#[cfg(target_arch = "aarch64")]
id_to_dev_info: HashMap<(DeviceType, String), MMIODeviceInfo>,
} }
impl DeviceManager { impl DeviceManager {
@ -723,13 +755,14 @@ impl DeviceManager {
config: Arc<Mutex<VmConfig>>, config: Arc<Mutex<VmConfig>>,
memory_manager: Arc<Mutex<MemoryManager>>, memory_manager: Arc<Mutex<MemoryManager>>,
_exit_evt: &EventFd, _exit_evt: &EventFd,
reset_evt: &EventFd, #[cfg_attr(target_arch = "aarch64", allow(unused_variables))] reset_evt: &EventFd,
vmm_path: PathBuf, vmm_path: PathBuf,
) -> DeviceManagerResult<Arc<Mutex<Self>>> { ) -> DeviceManagerResult<Arc<Mutex<Self>>> {
let device_tree = Arc::new(Mutex::new(DeviceTree::new())); let device_tree = Arc::new(Mutex::new(DeviceTree::new()));
let address_manager = Arc::new(AddressManager { let address_manager = Arc::new(AddressManager {
allocator: memory_manager.lock().unwrap().allocator(), allocator: memory_manager.lock().unwrap().allocator(),
#[cfg(target_arch = "x86_64")]
io_bus: Arc::new(devices::Bus::new()), io_bus: Arc::new(devices::Bus::new()),
mmio_bus: Arc::new(devices::Bus::new()), mmio_bus: Arc::new(devices::Bus::new()),
vm_fd: vm_fd.clone(), vm_fd: vm_fd.clone(),
@ -789,7 +822,10 @@ impl DeviceManager {
device_tree, device_tree,
#[cfg(feature = "acpi")] #[cfg(feature = "acpi")]
exit_evt: _exit_evt.try_clone().map_err(DeviceManagerError::EventFd)?, exit_evt: _exit_evt.try_clone().map_err(DeviceManagerError::EventFd)?,
#[cfg(target_arch = "x86_64")]
reset_evt: reset_evt.try_clone().map_err(DeviceManagerError::EventFd)?, reset_evt: reset_evt.try_clone().map_err(DeviceManagerError::EventFd)?,
#[cfg(target_arch = "aarch64")]
id_to_dev_info: HashMap::new(),
}; };
#[cfg(feature = "acpi")] #[cfg(feature = "acpi")]
@ -846,12 +882,16 @@ impl DeviceManager {
) )
.map_err(DeviceManagerError::BusError)?; .map_err(DeviceManagerError::BusError)?;
#[cfg(target_arch = "x86_64")]
self.add_legacy_devices( self.add_legacy_devices(
self.reset_evt self.reset_evt
.try_clone() .try_clone()
.map_err(DeviceManagerError::EventFd)?, .map_err(DeviceManagerError::EventFd)?,
)?; )?;
#[cfg(target_arch = "aarch64")]
self.add_legacy_devices(&legacy_interrupt_manager)?;
#[cfg(feature = "acpi")] #[cfg(feature = "acpi")]
{ {
self.ged_notification_device = self.add_acpi_devices( self.ged_notification_device = self.add_acpi_devices(
@ -895,6 +935,12 @@ impl DeviceManager {
Ok(()) Ok(())
} }
#[cfg(target_arch = "aarch64")]
/// Gets the information of the devices registered up to some point in time.
pub fn get_device_info(&self) -> &HashMap<(DeviceType, String), MMIODeviceInfo> {
&self.id_to_dev_info
}
#[allow(unused_variables)] #[allow(unused_variables)]
fn add_pci_devices( fn add_pci_devices(
&mut self, &mut self,
@ -980,6 +1026,7 @@ impl DeviceManager {
let pci_config_io = Arc::new(Mutex::new(PciConfigIo::new(Arc::clone(&pci_bus)))); let pci_config_io = Arc::new(Mutex::new(PciConfigIo::new(Arc::clone(&pci_bus))));
self.bus_devices self.bus_devices
.push(Arc::clone(&pci_config_io) as Arc<Mutex<dyn BusDevice>>); .push(Arc::clone(&pci_config_io) as Arc<Mutex<dyn BusDevice>>);
#[cfg(target_arch = "x86_64")]
self.address_manager self.address_manager
.io_bus .io_bus
.insert(pci_config_io, 0xcf8, 0x8) .insert(pci_config_io, 0xcf8, 0x8)
@ -1144,6 +1191,7 @@ impl DeviceManager {
Ok(Some(ged_device)) Ok(Some(ged_device))
} }
#[cfg(target_arch = "x86_64")]
fn add_legacy_devices(&mut self, reset_evt: EventFd) -> DeviceManagerResult<()> { fn add_legacy_devices(&mut self, reset_evt: EventFd) -> DeviceManagerResult<()> {
// Add a shutdown device (i8042) // Add a shutdown device (i8042)
let i8042 = Arc::new(Mutex::new(devices::legacy::I8042Device::new(reset_evt))); let i8042 = Arc::new(Mutex::new(devices::legacy::I8042Device::new(reset_evt)));
@ -1200,21 +1248,56 @@ impl DeviceManager {
Ok(()) Ok(())
} }
fn add_console_device( #[cfg(target_arch = "aarch64")]
fn add_legacy_devices(
&mut self, &mut self,
interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = LegacyIrqGroupConfig>>, interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = LegacyIrqGroupConfig>>,
virtio_devices: &mut Vec<(VirtioDeviceArc, bool, String)>, ) -> DeviceManagerResult<()> {
) -> DeviceManagerResult<Arc<Console>> { // Add a RTC device
let serial_config = self.config.lock().unwrap().serial.clone(); let rtc_irq = self
let serial_writer: Option<Box<dyn io::Write + Send>> = match serial_config.mode { .address_manager
ConsoleOutputMode::File => Some(Box::new( .allocator
File::create(serial_config.file.as_ref().unwrap()) .lock()
.map_err(DeviceManagerError::SerialOutputFileOpen)?, .unwrap()
)), .allocate_irq()
ConsoleOutputMode::Tty => Some(Box::new(stdout())), .unwrap();
ConsoleOutputMode::Off | ConsoleOutputMode::Null => None,
}; let interrupt_group = interrupt_manager
let serial = if serial_config.mode != ConsoleOutputMode::Off { .create_group(LegacyIrqGroupConfig {
irq: rtc_irq as InterruptIndex,
})
.map_err(DeviceManagerError::CreateInterruptGroup)?;
let rtc_device = Arc::new(Mutex::new(devices::legacy::RTC::new(interrupt_group)));
self.bus_devices
.push(Arc::clone(&rtc_device) as Arc<Mutex<dyn BusDevice>>);
let addr = GuestAddress(arch::layout::LEGACY_RTC_MAPPED_IO_START);
self.address_manager
.mmio_bus
.insert(rtc_device.clone(), addr.0, MMIO_LEN)
.map_err(DeviceManagerError::BusError)?;
self.id_to_dev_info.insert(
(DeviceType::RTC, "rtc".to_string()),
MMIODeviceInfo {
addr: addr.0,
len: MMIO_LEN,
irq: rtc_irq,
},
);
Ok(())
}
#[cfg(target_arch = "x86_64")]
fn add_serial_device(
&mut self,
interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = LegacyIrqGroupConfig>>,
serial_writer: Option<Box<dyn io::Write + Send>>,
) -> DeviceManagerResult<Arc<Mutex<Serial>>> {
// Serial is tied to IRQ #4 // Serial is tied to IRQ #4
let serial_irq = 4; let serial_irq = 4;
@ -1226,7 +1309,7 @@ impl DeviceManager {
}) })
.map_err(DeviceManagerError::CreateInterruptGroup)?; .map_err(DeviceManagerError::CreateInterruptGroup)?;
let serial = Arc::new(Mutex::new(devices::legacy::Serial::new( let serial = Arc::new(Mutex::new(Serial::new(
id.clone(), id.clone(),
interrupt_group, interrupt_group,
serial_writer, serial_writer,
@ -1255,7 +1338,86 @@ impl DeviceManager {
.unwrap() .unwrap()
.insert(id.clone(), device_node!(id, serial)); .insert(id.clone(), device_node!(id, serial));
Some(serial) Ok(serial)
}
#[cfg(target_arch = "aarch64")]
fn add_serial_device(
&mut self,
interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = LegacyIrqGroupConfig>>,
serial_writer: Option<Box<dyn io::Write + Send>>,
) -> DeviceManagerResult<Arc<Mutex<Serial>>> {
let id = String::from(SERIAL_DEVICE_NAME_PREFIX);
let serial_irq = self
.address_manager
.allocator
.lock()
.unwrap()
.allocate_irq()
.unwrap();
let interrupt_group = interrupt_manager
.create_group(LegacyIrqGroupConfig {
irq: serial_irq as InterruptIndex,
})
.map_err(DeviceManagerError::CreateInterruptGroup)?;
let serial = Arc::new(Mutex::new(Serial::new(
id.clone(),
interrupt_group,
serial_writer,
)));
self.bus_devices
.push(Arc::clone(&serial) as Arc<Mutex<dyn BusDevice>>);
let addr = GuestAddress(arch::layout::LEGACY_SERIAL_MAPPED_IO_START);
self.address_manager
.mmio_bus
.insert(serial.clone(), addr.0, MMIO_LEN)
.map_err(DeviceManagerError::BusError)?;
self.id_to_dev_info.insert(
(DeviceType::Serial, DeviceType::Serial.to_string()),
MMIODeviceInfo {
addr: addr.0,
len: MMIO_LEN,
irq: serial_irq,
},
);
self.cmdline_additions
.push(format!("earlycon=uart,mmio,0x{:08x}", addr.0));
// Fill the device tree with a new node. In case of restore, we
// know there is nothing to do, so we can simply override the
// existing entry.
self.device_tree
.lock()
.unwrap()
.insert(id.clone(), device_node!(id, serial));
Ok(serial)
}
fn add_console_device(
&mut self,
interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = LegacyIrqGroupConfig>>,
virtio_devices: &mut Vec<(VirtioDeviceArc, bool, String)>,
) -> DeviceManagerResult<Arc<Console>> {
let serial_config = self.config.lock().unwrap().serial.clone();
let serial_writer: Option<Box<dyn io::Write + Send>> = match serial_config.mode {
ConsoleOutputMode::File => Some(Box::new(
File::create(serial_config.file.as_ref().unwrap())
.map_err(DeviceManagerError::SerialOutputFileOpen)?,
)),
ConsoleOutputMode::Tty => Some(Box::new(stdout())),
ConsoleOutputMode::Off | ConsoleOutputMode::Null => None,
};
let serial = if serial_config.mode != ConsoleOutputMode::Off {
Some(self.add_serial_device(interrupt_manager, serial_writer)?)
} else { } else {
None None
}; };
@ -2432,6 +2594,7 @@ impl DeviceManager {
pci.register_mapping( pci.register_mapping(
virtio_pci_device.clone(), virtio_pci_device.clone(),
#[cfg(target_arch = "x86_64")]
self.address_manager.io_bus.as_ref(), self.address_manager.io_bus.as_ref(),
self.address_manager.mmio_bus.as_ref(), self.address_manager.mmio_bus.as_ref(),
bars.clone(), bars.clone(),
@ -2533,6 +2696,30 @@ impl DeviceManager {
(base.raw_value(), size) (base.raw_value(), size)
}; };
let irq_num = if let Some(irq) = mmio_irq {
irq
} else {
self.address_manager
.allocator
.lock()
.unwrap()
.allocate_irq()
.ok_or(DeviceManagerError::AllocateIrq)?
};
#[cfg(target_arch = "aarch64")]
{
let device_type = virtio_device.lock().unwrap().device_type();
self.id_to_dev_info.insert(
(DeviceType::Virtio(device_type), virtio_device_id),
MMIODeviceInfo {
addr: mmio_base,
len: mmio_size,
irq: irq_num,
},
);
}
let memory = self.memory_manager.lock().unwrap().guest_memory(); let memory = self.memory_manager.lock().unwrap().guest_memory();
let mut mmio_device = let mut mmio_device =
vm_virtio::transport::MmioDevice::new(id.clone(), memory, virtio_device) vm_virtio::transport::MmioDevice::new(id.clone(), memory, virtio_device)
@ -2546,17 +2733,6 @@ impl DeviceManager {
.map_err(DeviceManagerError::RegisterIoevent)?; .map_err(DeviceManagerError::RegisterIoevent)?;
} }
let irq_num = if let Some(irq) = mmio_irq {
irq
} else {
self.address_manager
.allocator
.lock()
.unwrap()
.allocate_irq()
.ok_or(DeviceManagerError::AllocateIrq)?
};
let interrupt_group = interrupt_manager let interrupt_group = interrupt_manager
.create_group(LegacyIrqGroupConfig { .create_group(LegacyIrqGroupConfig {
irq: irq_num as InterruptIndex, irq: irq_num as InterruptIndex,
@ -2573,6 +2749,7 @@ impl DeviceManager {
.insert(mmio_device_arc.clone(), mmio_base, MMIO_LEN) .insert(mmio_device_arc.clone(), mmio_base, MMIO_LEN)
.map_err(DeviceManagerError::BusError)?; .map_err(DeviceManagerError::BusError)?;
#[cfg(target_arch = "x86_64")]
self.cmdline_additions.push(format!( self.cmdline_additions.push(format!(
"virtio_mmio.device={}K@0x{:08x}:{}", "virtio_mmio.device={}K@0x{:08x}:{}",
mmio_size / 1024, mmio_size / 1024,
@ -2592,6 +2769,7 @@ impl DeviceManager {
Ok(()) Ok(())
} }
#[cfg(target_arch = "x86_64")]
pub fn io_bus(&self) -> &Arc<devices::Bus> { pub fn io_bus(&self) -> &Arc<devices::Bus> {
&self.address_manager.io_bus &self.address_manager.io_bus
} }
@ -2810,6 +2988,7 @@ impl DeviceManager {
.remove_by_device(&pci_device) .remove_by_device(&pci_device)
.map_err(DeviceManagerError::RemoveDeviceFromPciBus)?; .map_err(DeviceManagerError::RemoveDeviceFromPciBus)?;
#[cfg(target_arch = "x86_64")]
// Remove the device from the IO bus // Remove the device from the IO bus
self.io_bus() self.io_bus()
.remove_by_device(&bus_device) .remove_by_device(&bus_device)