2020-07-02 13:25:19 +01:00
|
|
|
// Copyright 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
|
|
//
|
|
|
|
// Portions Copyright 2017 The Chromium OS Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE-BSD-3-Clause file.
|
|
|
|
//
|
|
|
|
// Copyright © 2019 Intel Corporation
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 AND BSD-3-Clause
|
|
|
|
|
|
|
|
//! Implements virtio devices, queues, and transport mechanisms.
|
|
|
|
|
|
|
|
#[macro_use]
|
2021-02-18 15:10:51 +00:00
|
|
|
extern crate event_monitor;
|
|
|
|
#[macro_use]
|
2020-07-02 13:25:19 +01:00
|
|
|
extern crate log;
|
|
|
|
|
2022-05-17 14:04:38 -07:00
|
|
|
use serde::{Deserialize, Serialize};
|
2020-07-02 13:25:19 +01:00
|
|
|
use std::io;
|
2022-08-10 15:06:02 -07:00
|
|
|
use thiserror::Error;
|
2020-07-02 13:25:19 +01:00
|
|
|
|
|
|
|
#[macro_use]
|
|
|
|
mod device;
|
2020-03-20 11:43:17 +08:00
|
|
|
pub mod balloon;
|
2021-01-22 11:15:13 +01:00
|
|
|
pub mod block;
|
2020-07-02 13:25:19 +01:00
|
|
|
mod console;
|
2020-07-22 14:34:36 +01:00
|
|
|
pub mod epoll_helper;
|
2020-07-02 13:25:19 +01:00
|
|
|
mod iommu;
|
|
|
|
pub mod mem;
|
|
|
|
pub mod net;
|
|
|
|
mod pmem;
|
|
|
|
mod rng;
|
2020-08-03 19:45:53 -07:00
|
|
|
pub mod seccomp_filters;
|
2021-09-03 11:43:30 +01:00
|
|
|
mod thread_helper;
|
2020-07-02 13:25:19 +01:00
|
|
|
pub mod transport;
|
virtio-devices: Add Vdpa device
vDPA is a kernel framework introduced fairly recently in order to handle
devices complying with virtio specification on their datapath, while the
control path is vendor specific. For the datapath, that means the
virtqueues are handled through DMA directly between the hardware and the
guest, while the control path goes through the vDPA framework,
eventually exposed through a vhost-vdpa device.
vDPA, like VFIO, aims at achieving baremetal performance for devices
that are passed into a VM. But unlike VFIO, it provides a simpler/better
framework for achieving migration. Because the DMA accesses between the
device and the guest are going through virtio queues, migration can be
achieved way more easily, and doesn't require each device driver to
implement the migration support. In the VFIO case, each vendor is
expected to provide an implementation of the VFIO migration framework,
which makes things harder as it must be done for each and every device.
So to summarize the point is to support migration for hardware devices
through which we can achieve baremetal performances.
Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
2022-03-07 15:34:44 +01:00
|
|
|
pub mod vdpa;
|
2020-07-02 13:25:19 +01:00
|
|
|
pub mod vhost_user;
|
|
|
|
pub mod vsock;
|
2020-09-25 10:35:13 +01:00
|
|
|
pub mod watchdog;
|
2020-07-02 13:25:19 +01:00
|
|
|
|
2023-04-22 08:17:55 +01:00
|
|
|
pub use self::balloon::Balloon;
|
|
|
|
pub use self::block::{Block, BlockState};
|
|
|
|
pub use self::console::{Console, ConsoleResizer, Endpoint};
|
|
|
|
pub use self::device::{
|
|
|
|
DmaRemapping, UserspaceMapping, VirtioCommon, VirtioDevice, VirtioInterrupt,
|
|
|
|
VirtioInterruptType, VirtioSharedMemoryList,
|
|
|
|
};
|
|
|
|
pub use self::epoll_helper::{
|
|
|
|
EpollHelper, EpollHelperError, EpollHelperHandler, EPOLL_HELPER_EVENT_LAST,
|
|
|
|
};
|
|
|
|
pub use self::iommu::{AccessPlatformMapping, Iommu, IommuMapping};
|
|
|
|
pub use self::mem::{BlocksState, Mem, VirtioMemMappingSource, VIRTIO_MEM_ALIGN_SIZE};
|
|
|
|
pub use self::net::{Net, NetCtrlEpollHandler};
|
|
|
|
pub use self::pmem::Pmem;
|
|
|
|
pub use self::rng::Rng;
|
|
|
|
pub use self::vdpa::{Vdpa, VdpaDmaMapping};
|
|
|
|
pub use self::vsock::Vsock;
|
|
|
|
pub use self::watchdog::Watchdog;
|
2021-06-02 12:08:04 -07:00
|
|
|
use vm_memory::{bitmap::AtomicBitmap, GuestAddress, GuestMemory};
|
2021-10-21 12:41:16 +02:00
|
|
|
use vm_virtio::VirtioDeviceType;
|
2020-07-02 13:25:19 +01:00
|
|
|
|
2021-06-02 12:08:04 -07:00
|
|
|
type GuestMemoryMmap = vm_memory::GuestMemoryMmap<AtomicBitmap>;
|
|
|
|
type GuestRegionMmap = vm_memory::GuestRegionMmap<AtomicBitmap>;
|
|
|
|
type MmapRegion = vm_memory::MmapRegion<AtomicBitmap>;
|
|
|
|
|
2020-07-02 13:25:19 +01:00
|
|
|
const DEVICE_INIT: u32 = 0x00;
|
|
|
|
const DEVICE_ACKNOWLEDGE: u32 = 0x01;
|
|
|
|
const DEVICE_DRIVER: u32 = 0x02;
|
|
|
|
const DEVICE_DRIVER_OK: u32 = 0x04;
|
|
|
|
const DEVICE_FEATURES_OK: u32 = 0x08;
|
|
|
|
const DEVICE_FAILED: u32 = 0x80;
|
|
|
|
|
2021-05-21 18:26:00 +02:00
|
|
|
const VIRTIO_F_RING_INDIRECT_DESC: u32 = 28;
|
|
|
|
const VIRTIO_F_RING_EVENT_IDX: u32 = 29;
|
2020-07-02 13:25:19 +01:00
|
|
|
const VIRTIO_F_VERSION_1: u32 = 32;
|
|
|
|
const VIRTIO_F_IOMMU_PLATFORM: u32 = 33;
|
|
|
|
const VIRTIO_F_IN_ORDER: u32 = 35;
|
2021-05-21 18:26:00 +02:00
|
|
|
const VIRTIO_F_ORDER_PLATFORM: u32 = 36;
|
|
|
|
#[allow(dead_code)]
|
|
|
|
const VIRTIO_F_SR_IOV: u32 = 37;
|
|
|
|
const VIRTIO_F_NOTIFICATION_DATA: u32 = 38;
|
2020-07-02 13:25:19 +01:00
|
|
|
|
2022-08-10 15:06:02 -07:00
|
|
|
#[derive(Error, Debug)]
|
2020-07-02 13:25:19 +01:00
|
|
|
pub enum ActivateError {
|
2022-10-20 11:45:16 -07:00
|
|
|
#[error("Failed to activate virtio device")]
|
2020-07-02 13:25:19 +01:00
|
|
|
BadActivate,
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to clone exit event fd: {0}")]
|
2021-09-07 16:10:48 +01:00
|
|
|
CloneExitEventFd(std::io::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to spawn thread: {0}")]
|
2021-09-07 16:10:48 +01:00
|
|
|
ThreadSpawn(std::io::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to setup vhost-user-fs daemon: {0}")]
|
2021-03-11 20:53:05 +08:00
|
|
|
VhostUserFsSetup(vhost_user::Error),
|
2023-04-12 16:58:59 +00:00
|
|
|
#[error("Failed to setup vhost-user daemon: {0}")]
|
|
|
|
VhostUserSetup(vhost_user::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to create seccomp filter: {0}")]
|
2021-08-16 20:40:11 -07:00
|
|
|
CreateSeccompFilter(seccompiler::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to create rate limiter: {0}")]
|
2021-02-26 12:06:10 -08:00
|
|
|
CreateRateLimiter(std::io::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to activate the vDPA device: {0}")]
|
virtio-devices: Add Vdpa device
vDPA is a kernel framework introduced fairly recently in order to handle
devices complying with virtio specification on their datapath, while the
control path is vendor specific. For the datapath, that means the
virtqueues are handled through DMA directly between the hardware and the
guest, while the control path goes through the vDPA framework,
eventually exposed through a vhost-vdpa device.
vDPA, like VFIO, aims at achieving baremetal performance for devices
that are passed into a VM. But unlike VFIO, it provides a simpler/better
framework for achieving migration. Because the DMA accesses between the
device and the guest are going through virtio queues, migration can be
achieved way more easily, and doesn't require each device driver to
implement the migration support. In the VFIO case, each vendor is
expected to provide an implementation of the VFIO migration framework,
which makes things harder as it must be done for each and every device.
So to summarize the point is to support migration for hardware devices
through which we can achieve baremetal performances.
Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
2022-03-07 15:34:44 +01:00
|
|
|
ActivateVdpa(vdpa::Error),
|
2020-07-02 13:25:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub type ActivateResult = std::result::Result<(), ActivateError>;
|
|
|
|
|
|
|
|
pub type DeviceEventT = u16;
|
|
|
|
|
2022-08-10 15:06:02 -07:00
|
|
|
#[derive(Error, Debug)]
|
2020-07-02 13:25:19 +01:00
|
|
|
pub enum Error {
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to single used queue: {0}")]
|
2020-07-02 13:25:19 +01:00
|
|
|
FailedSignalingUsedQueue(io::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("I/O Error: {0}")]
|
2020-07-02 13:25:19 +01:00
|
|
|
IoError(io::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to update memory vhost-user: {0}")]
|
2020-07-02 13:25:19 +01:00
|
|
|
VhostUserUpdateMemory(vhost_user::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to add memory region vhost-user: {0}")]
|
2021-03-05 14:32:57 +01:00
|
|
|
VhostUserAddMemoryRegion(vhost_user::Error),
|
2022-10-20 11:45:16 -07:00
|
|
|
#[error("Failed to set shared memory region")]
|
2020-07-02 13:25:19 +01:00
|
|
|
SetShmRegionsNotSupported,
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to process net queue: {0}")]
|
2020-07-07 16:50:13 +01:00
|
|
|
NetQueuePair(::net_util::NetQueuePairError),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to : {0}")]
|
2021-10-21 12:41:16 +02:00
|
|
|
QueueAddUsed(virtio_queue::Error),
|
2022-08-10 15:06:02 -07:00
|
|
|
#[error("Failed to : {0}")]
|
2021-10-21 12:41:16 +02:00
|
|
|
QueueIterator(virtio_queue::Error),
|
2020-07-02 13:25:19 +01:00
|
|
|
}
|
2021-02-23 14:57:10 +01:00
|
|
|
|
2022-06-30 16:41:46 +01:00
|
|
|
#[derive(Clone, Copy, Debug, Default, Deserialize, Serialize, PartialEq, Eq)]
|
2021-02-26 12:06:10 -08:00
|
|
|
pub struct TokenBucketConfig {
|
|
|
|
pub size: u64,
|
|
|
|
pub one_time_burst: Option<u64>,
|
|
|
|
pub refill_time: u64,
|
|
|
|
}
|
|
|
|
|
2022-06-30 16:41:46 +01:00
|
|
|
#[derive(Clone, Copy, Debug, Default, Deserialize, Serialize, PartialEq, Eq)]
|
2021-02-26 12:06:10 -08:00
|
|
|
#[serde(deny_unknown_fields)]
|
|
|
|
pub struct RateLimiterConfig {
|
|
|
|
pub bandwidth: Option<TokenBucketConfig>,
|
|
|
|
pub ops: Option<TokenBucketConfig>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryInto<rate_limiter::RateLimiter> for RateLimiterConfig {
|
|
|
|
type Error = io::Error;
|
|
|
|
|
|
|
|
fn try_into(self) -> std::result::Result<rate_limiter::RateLimiter, Self::Error> {
|
|
|
|
let bw = self.bandwidth.unwrap_or_default();
|
|
|
|
let ops = self.ops.unwrap_or_default();
|
|
|
|
rate_limiter::RateLimiter::new(
|
|
|
|
bw.size,
|
|
|
|
bw.one_time_burst.unwrap_or(0),
|
|
|
|
bw.refill_time,
|
|
|
|
ops.size,
|
|
|
|
ops.one_time_burst.unwrap_or(0),
|
|
|
|
ops.refill_time,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-06 11:02:16 -07:00
|
|
|
/// Return the host virtual address corresponding to the given guest address range
|
|
|
|
///
|
2021-02-23 14:57:10 +01:00
|
|
|
/// Convert an absolute address into an address space (GuestMemory)
|
|
|
|
/// to a host pointer and verify that the provided size define a valid
|
|
|
|
/// range within a single memory region.
|
|
|
|
/// Return None if it is out of bounds or if addr+size overlaps a single region.
|
2022-07-06 16:08:08 +02:00
|
|
|
pub fn get_host_address_range<M: GuestMemory + ?Sized>(
|
2021-02-23 14:57:10 +01:00
|
|
|
mem: &M,
|
|
|
|
addr: GuestAddress,
|
|
|
|
size: usize,
|
|
|
|
) -> Option<*mut u8> {
|
|
|
|
if mem.check_range(addr, size) {
|
|
|
|
Some(mem.get_host_address(addr).unwrap())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|