mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-11 15:17:42 +00:00
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:
parent
e9488846f1
commit
eeeb45bbb9
@ -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,
|
||||||
|
@ -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)
|
||||||
|
Loading…
Reference in New Issue
Block a user