vmm: Call munmap() on regions that have been mmap()ed

For virtio-fs and virtio-pmem regions of memory are manually mapped into
the address space of the VMM. In order to cleanly reboot we need to
unmap those regions.

Fixes: #223

Signed-off-by: Rob Bradford <robert.bradford@intel.com>
This commit is contained in:
Rob Bradford 2019-09-05 10:10:44 +01:00
parent 037807f949
commit 5dd675710b

View File

@ -633,6 +633,9 @@ struct DeviceManager {
// PCI root
pci: Arc<Mutex<PciConfigIo>>,
// mmap()ed region to unmap on drop
mmap_regions: Vec<(*mut libc::c_void, usize)>,
}
impl DeviceManager {
@ -740,6 +743,8 @@ impl DeviceManager {
None
};
let mut mmap_regions = Vec::new();
DeviceManager::add_virtio_devices(
vm_info,
allocator,
@ -747,6 +752,7 @@ impl DeviceManager {
&mut buses,
&interrupt_info,
&mut mem_slots,
&mut mmap_regions,
)?;
DeviceManager::add_vfio_devices(vm_info, allocator, &mut pci, &mut buses, mem_slots)?;
@ -765,6 +771,7 @@ impl DeviceManager {
reset_evt,
ioapic,
pci,
mmap_regions,
})
}
@ -775,6 +782,7 @@ impl DeviceManager {
buses: &mut BusInfo,
interrupt_info: &InterruptInfo,
mut mem_slots: &mut u32,
mmap_regions: &mut Vec<(*mut libc::c_void, usize)>,
) -> DeviceManagerResult<()> {
// Add virtio-blk if required
DeviceManager::add_virtio_block_devices(vm_info, allocator, pci, buses, &interrupt_info)?;
@ -793,6 +801,7 @@ impl DeviceManager {
buses,
&interrupt_info,
&mut mem_slots,
mmap_regions,
)?;
// Add virtio-pmem if required
@ -803,6 +812,7 @@ impl DeviceManager {
buses,
&interrupt_info,
&mut mem_slots,
mmap_regions,
)?;
// Add virtio-vhost-user-net if required
@ -938,6 +948,7 @@ impl DeviceManager {
buses: &mut BusInfo,
interrupt_info: &InterruptInfo,
mem_slots: &mut u32,
mmap_regions: &mut Vec<(*mut libc::c_void, usize)>,
) -> DeviceManagerResult<()> {
// Add virtio-fs if required
if let Some(fs_list_cfg) = &vm_info.vm_cfg.fs {
@ -968,6 +979,7 @@ impl DeviceManager {
if addr == libc::MAP_FAILED {
return Err(DeviceManagerError::Mmap(io::Error::last_os_error()));
}
mmap_regions.push((addr, fs_cache as usize));
let mem_region = kvm_userspace_memory_region {
slot: *mem_slots as u32,
@ -1029,6 +1041,7 @@ impl DeviceManager {
buses: &mut BusInfo,
interrupt_info: &InterruptInfo,
mem_slots: &mut u32,
mmap_regions: &mut Vec<(*mut libc::c_void, usize)>,
) -> DeviceManagerResult<()> {
// Add virtio-pmem if required
if let Some(pmem_list_cfg) = &vm_info.vm_cfg.pmem {
@ -1067,9 +1080,11 @@ impl DeviceManager {
libc::MAP_NORESERVE | libc::MAP_SHARED,
file.as_raw_fd(),
0 as libc::off_t,
) as *mut u8
)
};
mmap_regions.push((addr, size as usize));
let mem_region = kvm_userspace_memory_region {
slot: *mem_slots as u32,
guest_phys_addr: pmem_guest_addr.raw_value(),
@ -1329,6 +1344,16 @@ impl DeviceManager {
}
}
impl Drop for DeviceManager {
fn drop(&mut self) {
for (addr, size) in self.mmap_regions.drain(..) {
unsafe {
libc::munmap(addr, size);
}
}
}
}
#[derive(Debug, Clone, Copy, PartialEq)]
enum EpollDispatch {
Exit,