use std::path::PathBuf;
use std::str::FromStr;
use anyhow::anyhow;
use anyhow::bail;
use anyhow::Context;
use devices::IommuDevType;
use devices::PciAddress;
use devices::SerialParameters;
use libc::getegid;
use libc::geteuid;
use serde::Deserialize;
use serde::Serialize;
use serde_keyvalue::from_key_values;
use serde_keyvalue::FromKeyValues;
use crate::crosvm::config::Config;
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize, FromKeyValues)]
#[serde(deny_unknown_fields, rename_all = "kebab-case")]
pub enum HypervisorKind {
Kvm {
device: Option<PathBuf>,
},
#[cfg(any(target_arch = "arm", target_arch = "aarch64"))]
#[cfg(feature = "geniezone")]
Geniezone {
device: Option<PathBuf>,
},
#[cfg(all(any(target_arch = "arm", target_arch = "aarch64"), feature = "gunyah"))]
Gunyah {
device: Option<PathBuf>,
},
}
pub fn check_serial_params(_serial_params: &SerialParameters) -> Result<(), String> {
Ok(())
}
pub fn validate_config(_cfg: &mut Config) -> std::result::Result<(), String> {
Ok(())
}
#[derive(Serialize, Deserialize, FromKeyValues)]
#[serde(deny_unknown_fields, rename_all = "kebab-case")]
pub struct VfioOption {
pub path: PathBuf,
#[serde(default)]
pub iommu: IommuDevType,
pub guest_address: Option<PciAddress>,
pub dt_symbol: Option<String>,
}
#[derive(Default, Eq, PartialEq, Serialize, Deserialize)]
pub enum SharedDirKind {
FS,
#[default]
P9,
}
impl FromStr for SharedDirKind {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
use SharedDirKind::*;
match s {
"fs" | "FS" => Ok(FS),
"9p" | "9P" | "p9" | "P9" => Ok(P9),
_ => {
bail!("invalid file system type");
}
}
}
}
pub struct SharedDir {
pub src: PathBuf,
pub tag: String,
pub kind: SharedDirKind,
pub ugid: (Option<u32>, Option<u32>),
pub uid_map: String,
pub gid_map: String,
pub fs_cfg: devices::virtio::fs::Config,
pub p9_cfg: p9::Config,
}
impl Default for SharedDir {
fn default() -> SharedDir {
SharedDir {
src: Default::default(),
tag: Default::default(),
kind: Default::default(),
ugid: (None, None),
uid_map: format!("0 {} 1", unsafe { geteuid() }),
gid_map: format!("0 {} 1", unsafe { getegid() }),
fs_cfg: Default::default(),
p9_cfg: Default::default(),
}
}
}
impl FromStr for SharedDir {
type Err = anyhow::Error;
fn from_str(param: &str) -> Result<Self, Self::Err> {
let mut components = param.split(':');
let src = PathBuf::from(
components
.next()
.context("missing source path for `shared-dir`")?,
);
let tag = components
.next()
.context("missing tag for `shared-dir`")?
.to_owned();
if !src.is_dir() {
bail!("source path for `shared-dir` must be a directory");
}
let mut shared_dir = SharedDir {
src,
tag,
..Default::default()
};
let mut type_opts = vec![];
for opt in components {
let mut o = opt.splitn(2, '=');
let kind = o.next().context("`shared-dir` options must not be empty")?;
let value = o
.next()
.context("`shared-dir` options must be of the form `kind=value`")?;
match kind {
"type" => {
shared_dir.kind = value.parse().with_context(|| {
anyhow!("`type` must be one of `fs` or `9p` but {value}")
})?
}
"uidmap" => shared_dir.uid_map = value.into(),
"gidmap" => shared_dir.gid_map = value.into(),
"uid" => {
shared_dir.ugid.0 = Some(value.parse().context("`uid` must be an integer")?);
}
"gid" => {
shared_dir.ugid.1 = Some(value.parse().context("`gid` must be an integer")?);
}
_ => type_opts.push(opt),
}
}
match shared_dir.kind {
SharedDirKind::FS => {
shared_dir.fs_cfg = from_key_values(&type_opts.join(","))
.map_err(|e| anyhow!("failed to parse fs config '{:?}': {e}", type_opts))?;
if shared_dir.fs_cfg.ascii_casefold && !shared_dir.fs_cfg.negative_timeout.is_zero()
{
bail!("'negative_timeout' cannot be used with 'ascii_casefold'");
}
}
SharedDirKind::P9 => {
shared_dir.p9_cfg = type_opts
.join(":")
.parse()
.map_err(|e| anyhow!("failed to parse 9p config '{:?}': {e}", type_opts))?;
}
}
Ok(shared_dir)
}
}
#[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]
#[serde(deny_unknown_fields)]
pub struct PmemExt2Option {
pub path: PathBuf,
pub blocks_per_group: u32,
pub inodes_per_group: u32,
pub size: u32,
}
impl Default for PmemExt2Option {
fn default() -> Self {
let blocks_per_group = 4096;
let inodes_per_group = 1024;
let size = ext2::BLOCK_SIZE as u32 * blocks_per_group; Self {
path: Default::default(),
blocks_per_group,
inodes_per_group,
size,
}
}
}
pub fn parse_pmem_ext2_option(param: &str) -> Result<PmemExt2Option, String> {
let mut opt = PmemExt2Option::default();
let mut components = param.split(':');
opt.path = PathBuf::from(
components
.next()
.ok_or("missing source path for `pmem-ext2`")?,
);
for c in components {
let mut o = c.splitn(2, '=');
let kind = o.next().ok_or("`pmem-ext2` options must not be empty")?;
let value = o
.next()
.ok_or("`pmem-ext2` options must be of the form `kind=value`")?;
match kind {
"blocks_per_group" => {
opt.blocks_per_group = value
.parse()
.map_err(|e| format!("failed to parse blocks_per_groups '{value}': {:#}", e))?
}
"inodes_per_group" => {
opt.inodes_per_group = value
.parse()
.map_err(|e| format!("failed to parse inodes_per_groups '{value}': {:#}", e))?
}
"size" => {
opt.size = value
.parse()
.map_err(|e| format!("failed to parse memory size '{value}': {:#}", e))?
}
_ => return Err(format!("invalid `pmem-ext2` option: {}", kind)),
}
}
Ok(opt)
}
#[cfg(test)]
mod tests {
use std::path::Path;
use std::path::PathBuf;
use std::time::Duration;
use argh::FromArgs;
use devices::virtio::fs::CachePolicy;
use super::*;
use crate::crosvm::config::from_key_values;
#[test]
fn parse_coiommu_options() {
use std::time::Duration;
use devices::CoIommuParameters;
use devices::CoIommuUnpinPolicy;
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_policy=off").unwrap();
assert_eq!(
coiommu_params,
CoIommuParameters {
unpin_policy: CoIommuUnpinPolicy::Off,
..Default::default()
}
);
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_policy=lru").unwrap();
assert_eq!(
coiommu_params,
CoIommuParameters {
unpin_policy: CoIommuUnpinPolicy::Lru,
..Default::default()
}
);
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_policy=foo");
assert!(coiommu_params.is_err());
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_interval=42").unwrap();
assert_eq!(
coiommu_params,
CoIommuParameters {
unpin_interval: Duration::from_secs(42),
..Default::default()
}
);
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_interval=foo");
assert!(coiommu_params.is_err());
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_limit=256").unwrap();
assert_eq!(
coiommu_params,
CoIommuParameters {
unpin_limit: Some(256),
..Default::default()
}
);
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_limit=0");
assert!(coiommu_params.is_err());
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_limit=foo");
assert!(coiommu_params.is_err());
let coiommu_params =
from_key_values::<CoIommuParameters>("unpin_gen_threshold=32").unwrap();
assert_eq!(
coiommu_params,
CoIommuParameters {
unpin_gen_threshold: 32,
..Default::default()
}
);
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_gen_threshold=foo");
assert!(coiommu_params.is_err());
let coiommu_params = from_key_values::<CoIommuParameters>(
"unpin_policy=lru,unpin_interval=90,unpin_limit=8,unpin_gen_threshold=64",
)
.unwrap();
assert_eq!(
coiommu_params,
CoIommuParameters {
unpin_policy: CoIommuUnpinPolicy::Lru,
unpin_interval: Duration::from_secs(90),
unpin_limit: Some(8),
unpin_gen_threshold: 64,
}
);
let coiommu_params = from_key_values::<CoIommuParameters>("unpin_invalid_param=0");
assert!(coiommu_params.is_err());
}
#[test]
fn vfio_pci_path() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--vfio", "/path/to/dev", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
let vfio = config.vfio.first().unwrap();
assert_eq!(vfio.path, PathBuf::from("/path/to/dev"));
assert_eq!(vfio.iommu, IommuDevType::NoIommu);
assert_eq!(vfio.guest_address, None);
}
#[test]
fn vfio_pci_path_coiommu() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--vfio", "/path/to/dev,iommu=coiommu", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
let vfio = config.vfio.first().unwrap();
assert_eq!(vfio.path, PathBuf::from("/path/to/dev"));
assert_eq!(vfio.iommu, IommuDevType::CoIommu);
assert_eq!(vfio.guest_address, None);
}
#[test]
fn vfio_pci_path_viommu_guest_address() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&[
"--vfio",
"/path/to/dev,iommu=viommu,guest-address=42:15.4",
"/dev/null",
],
)
.unwrap()
.try_into()
.unwrap();
let vfio = config.vfio.first().unwrap();
assert_eq!(vfio.path, PathBuf::from("/path/to/dev"));
assert_eq!(vfio.iommu, IommuDevType::VirtioIommu);
assert_eq!(
vfio.guest_address,
Some(PciAddress::new(0, 0x42, 0x15, 4).unwrap())
);
}
#[test]
fn vfio_platform() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--vfio-platform", "/path/to/dev", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
let vfio = config.vfio.first().unwrap();
assert_eq!(vfio.path, PathBuf::from("/path/to/dev"));
}
#[test]
fn hypervisor_default() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(&[], &["/dev/null"])
.unwrap()
.try_into()
.unwrap();
assert_eq!(config.hypervisor, None);
}
#[test]
fn hypervisor_kvm() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--hypervisor", "kvm", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
assert_eq!(
config.hypervisor,
Some(HypervisorKind::Kvm { device: None })
);
}
#[test]
fn hypervisor_kvm_device() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--hypervisor", "kvm[device=/not/default]", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
assert_eq!(
config.hypervisor,
Some(HypervisorKind::Kvm {
device: Some(PathBuf::from("/not/default"))
})
);
}
#[test]
#[cfg(any(target_arch = "arm", target_arch = "aarch64"))]
#[cfg(feature = "geniezone")]
fn hypervisor_geniezone() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--hypervisor", "geniezone", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
assert_eq!(
config.hypervisor,
Some(HypervisorKind::Geniezone { device: None })
);
}
#[test]
#[cfg(any(target_arch = "arm", target_arch = "aarch64"))]
#[cfg(feature = "geniezone")]
fn hypervisor_geniezone_device() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&[
"--hypervisor",
"geniezone[device=/not/default]",
"/dev/null",
],
)
.unwrap()
.try_into()
.unwrap();
assert_eq!(
config.hypervisor,
Some(HypervisorKind::Geniezone {
device: Some(PathBuf::from("/not/default"))
})
);
}
#[test]
#[cfg(all(any(target_arch = "arm", target_arch = "aarch64"), feature = "gunyah"))]
fn hypervisor_gunyah() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--hypervisor", "gunyah", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
assert_eq!(
config.hypervisor,
Some(HypervisorKind::Gunyah { device: None })
);
}
#[test]
#[cfg(all(any(target_arch = "arm", target_arch = "aarch64"), feature = "gunyah"))]
fn hypervisor_gunyah_device() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--hypervisor", "gunyah[device=/not/default]", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
assert_eq!(
config.hypervisor,
Some(HypervisorKind::Gunyah {
device: Some(PathBuf::from("/not/default"))
})
);
}
#[test]
fn parse_shared_dir() {
let s = "/:usr_local_bin:type=fs:cache=always:uidmap=0 655360 5000,5000 600 50,5050 660410 1994950:gidmap=0 655360 1065,1065 20119 1,1066 656426 3934,5000 600 50,5050 660410 1994950:timeout=3600:rewrite-security-xattrs=true:writeback=true";
let shared_dir: SharedDir = s.parse().unwrap();
assert_eq!(shared_dir.src, Path::new("/").to_path_buf());
assert_eq!(shared_dir.tag, "usr_local_bin");
assert!(shared_dir.kind == SharedDirKind::FS);
assert_eq!(
shared_dir.uid_map,
"0 655360 5000,5000 600 50,5050 660410 1994950"
);
assert_eq!(
shared_dir.gid_map,
"0 655360 1065,1065 20119 1,1066 656426 3934,5000 600 50,5050 660410 1994950"
);
assert_eq!(shared_dir.fs_cfg.ascii_casefold, false);
assert_eq!(shared_dir.fs_cfg.timeout, Duration::from_secs(3600));
assert_eq!(shared_dir.fs_cfg.negative_timeout, Duration::ZERO);
assert_eq!(shared_dir.fs_cfg.writeback, true);
assert_eq!(
shared_dir.fs_cfg.cache_policy,
devices::virtio::fs::CachePolicy::Always
);
assert_eq!(shared_dir.fs_cfg.rewrite_security_xattrs, true);
assert_eq!(shared_dir.fs_cfg.use_dax, false);
assert_eq!(shared_dir.fs_cfg.posix_acl, true);
assert_eq!(shared_dir.ugid, (None, None));
}
#[test]
fn parse_shared_dir_parses_ascii_casefold_and_posix_acl() {
let s = "/:usr_local_bin:type=fs:ascii_casefold=true:posix_acl=false";
let shared_dir: SharedDir = s.parse().unwrap();
assert_eq!(shared_dir.fs_cfg.ascii_casefold, true);
assert_eq!(shared_dir.fs_cfg.posix_acl, false);
}
#[test]
fn parse_shared_dir_negative_timeout() {
let s = "/:usr_local_bin:type=fs:cache=always:timeout=3600:negative_timeout=60";
let shared_dir: SharedDir = s.parse().unwrap();
assert_eq!(shared_dir.src, Path::new("/").to_path_buf());
assert_eq!(shared_dir.tag, "usr_local_bin");
assert!(shared_dir.kind == SharedDirKind::FS);
assert_eq!(
shared_dir.fs_cfg.cache_policy,
devices::virtio::fs::CachePolicy::Always
);
assert_eq!(shared_dir.fs_cfg.timeout, Duration::from_secs(3600));
assert_eq!(shared_dir.fs_cfg.negative_timeout, Duration::from_secs(60));
}
#[test]
fn parse_shared_dir_oem() {
let shared_dir: SharedDir = "/:oem_etc:type=fs:cache=always:uidmap=0 299 1, 5000 600 50:gidmap=0 300 1, 5000 600 50:timeout=3600:rewrite-security-xattrs=true".parse().unwrap();
assert_eq!(shared_dir.src, Path::new("/").to_path_buf());
assert_eq!(shared_dir.tag, "oem_etc");
assert!(shared_dir.kind == SharedDirKind::FS);
assert_eq!(shared_dir.uid_map, "0 299 1, 5000 600 50");
assert_eq!(shared_dir.gid_map, "0 300 1, 5000 600 50");
assert_eq!(shared_dir.fs_cfg.ascii_casefold, false);
assert_eq!(shared_dir.fs_cfg.timeout, Duration::from_secs(3600));
assert_eq!(shared_dir.fs_cfg.negative_timeout, Duration::ZERO);
assert_eq!(shared_dir.fs_cfg.writeback, false);
assert_eq!(
shared_dir.fs_cfg.cache_policy,
devices::virtio::fs::CachePolicy::Always
);
assert_eq!(shared_dir.fs_cfg.rewrite_security_xattrs, true);
assert_eq!(shared_dir.fs_cfg.use_dax, false);
assert_eq!(shared_dir.fs_cfg.posix_acl, true);
assert_eq!(shared_dir.ugid, (None, None));
}
#[test]
#[cfg(feature = "arc_quota")]
fn parse_shared_dir_arcvm_data() {
let arcvm_arg = "/:_data:type=fs:cache=always:uidmap=0 655360 5000,5000 600 50,5050 660410 1994950:gidmap=0 655360 1065,1065 20119 1,1066 656426 3934,5000 600 50,5050 660410 1994950:timeout=3600:rewrite-security-xattrs=true:writeback=true:privileged_quota_uids=0";
assert_eq!(
arcvm_arg.parse::<SharedDir>().unwrap().fs_cfg,
devices::virtio::fs::Config {
cache_policy: CachePolicy::Always,
timeout: Duration::from_secs(3600),
rewrite_security_xattrs: true,
writeback: true,
privileged_quota_uids: vec![0],
..Default::default()
}
);
}
#[test]
fn parse_shared_dir_ugid_set() {
let shared_dir: SharedDir =
"/:hostRoot:type=fs:uidmap=40417 40417 1:gidmap=5000 5000 1:uid=40417:gid=5000"
.parse()
.unwrap();
assert_eq!(shared_dir.src, Path::new("/").to_path_buf());
assert_eq!(shared_dir.tag, "hostRoot");
assert!(shared_dir.kind == SharedDirKind::FS);
assert_eq!(shared_dir.uid_map, "40417 40417 1");
assert_eq!(shared_dir.gid_map, "5000 5000 1");
assert_eq!(shared_dir.ugid, (Some(40417), Some(5000)));
}
#[test]
fn parse_shared_dir_vm_fio() {
let shared_dir: SharedDir =
"/:root:type=fs:cache=always:timeout=5:writeback=false:dax=false:ascii_casefold=false"
.parse()
.unwrap();
assert_eq!(
shared_dir.fs_cfg,
devices::virtio::fs::Config {
cache_policy: CachePolicy::Always,
timeout: Duration::from_secs(5),
writeback: false,
use_dax: false,
ascii_casefold: false,
..Default::default()
}
);
let shared_dir: SharedDir =
"/:shared:type=fs:cache=auto:timeout=1:writeback=true:dax=true:ascii_casefold=false"
.parse()
.unwrap();
assert_eq!(
shared_dir.fs_cfg,
devices::virtio::fs::Config {
cache_policy: CachePolicy::Auto,
timeout: Duration::from_secs(1),
writeback: true,
use_dax: true,
ascii_casefold: false,
..Default::default()
}
);
}
#[test]
fn parse_cache_policy() {
assert_eq!(
"/:_data:type=fs"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.cache_policy,
CachePolicy::Auto
);
assert_eq!(
"/:_data:type=fs:cache=always"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.cache_policy,
CachePolicy::Always
);
assert_eq!(
"/:_data:type=fs:cache=auto"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.cache_policy,
CachePolicy::Auto
);
assert_eq!(
"/:_data:type=fs:cache=never"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.cache_policy,
CachePolicy::Never
);
assert!("/:_data:type=fs:cache=Always".parse::<SharedDir>().is_err());
assert!("/:_data:type=fs:cache=ALWAYS".parse::<SharedDir>().is_err());
assert!("/:_data:type=fs:cache=Auto".parse::<SharedDir>().is_err());
assert!("/:_data:type=fs:cache=AUTO".parse::<SharedDir>().is_err());
assert!("/:_data:type=fs:cache=Never".parse::<SharedDir>().is_err());
assert!("/:_data:type=fs:cache=NEVER".parse::<SharedDir>().is_err());
assert!("/:_data:type=fs:cache=foobar".parse::<SharedDir>().is_err());
}
#[cfg(feature = "arc_quota")]
#[test]
fn parse_privileged_quota_uids() {
assert_eq!(
"/:_data:type=fs:privileged_quota_uids=0"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.privileged_quota_uids,
vec![0]
);
assert_eq!(
"/:_data:type=fs:privileged_quota_uids=0 1 2 3 4"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.privileged_quota_uids,
vec![0, 1, 2, 3, 4]
);
}
#[test]
fn parse_dax() {
assert!(
!"/:_data:type=fs"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.use_dax
);
assert!(
"/:_data:type=fs:dax=true"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.use_dax
);
assert!(
!"/:_data:type=fs:dax=false"
.parse::<SharedDir>()
.unwrap()
.fs_cfg
.use_dax
);
}
#[test]
fn parse_pmem_ext2() {
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&["--pmem-ext2", "/path/to/dir", "/dev/null"],
)
.unwrap()
.try_into()
.unwrap();
let opt = config.pmem_ext2.first().unwrap();
assert_eq!(opt.path, PathBuf::from("/path/to/dir"));
}
#[test]
fn parse_pmem_ext2_size() {
let blocks_per_group = 2048;
let inodes_per_group = 1024;
let size = 4096 * blocks_per_group;
let config: Config = crate::crosvm::cmdline::RunCommand::from_args(
&[],
&[
"--pmem-ext2",
&format!("/path/to/dir:blocks_per_group={blocks_per_group}:inodes_per_group={inodes_per_group}:size={size}"),
"/dev/null",
],
)
.unwrap()
.try_into()
.unwrap();
let opt = config.pmem_ext2.first().unwrap();
assert_eq!(opt.path, PathBuf::from("/path/to/dir"));
assert_eq!(opt.blocks_per_group, blocks_per_group);
assert_eq!(opt.inodes_per_group, inodes_per_group);
assert_eq!(opt.size, size);
}
}