// Copyright © 2019 Intel Corporation // // SPDX-License-Identifier: Apache-2.0 // extern crate vmm; #[macro_use(crate_version, crate_authors)] extern crate clap; use clap::{App, Arg}; use std::process; use vmm::config; fn main() { let cmd_arguments = App::new("cloud-hypervisor") .version(crate_version!()) .author(crate_authors!()) .about("Launch a cloud-hypervisor VMM.") .arg( Arg::with_name("cpus") .long("cpus") .help("Number of virtual CPUs") .default_value(config::DEFAULT_VCPUS), ) .arg( Arg::with_name("memory") .long("memory") .help( "Memory parameters \"size=,\ file=\"", ) .default_value(config::DEFAULT_MEMORY), ) .arg( Arg::with_name("kernel") .long("kernel") .help("Path to kernel image (vmlinux)") .takes_value(true), ) .arg( Arg::with_name("cmdline") .long("cmdline") .help("Kernel command line") .takes_value(true), ) .arg( Arg::with_name("disk") .long("disk") .help("Path to VM disk image") .takes_value(true) .min_values(1), ) .arg( Arg::with_name("net") .long("net") .help( "Network parameters \"tap=,\ ip=,mask=,mac=\"", ) .takes_value(true) .min_values(1), ) .arg( Arg::with_name("rng") .long("rng") .help("Path to entropy source") .default_value(config::DEFAULT_RNG_SOURCE), ) .arg( Arg::with_name("fs") .long("fs") .help( "virtio-fs parameters \"tag=,\ sock=,num_queues=,\ queue_size=\"", ) .takes_value(true) .min_values(1), ) .arg( Arg::with_name("pmem") .long("pmem") .help( "Persistent memory parameters \"file=,\ size=\"", ) .takes_value(true) .min_values(1), ) .arg( Arg::with_name("serial") .long("serial") .help("Control serial port: off|tty|file=/path/to/a/file") .default_value("tty"), ) .get_matches(); // These .unwrap()s cannot fail as there is a default value defined let cpus = cmd_arguments.value_of("cpus").unwrap(); let memory = cmd_arguments.value_of("memory").unwrap(); let rng = cmd_arguments.value_of("rng").unwrap(); let serial = cmd_arguments.value_of("serial").unwrap(); let kernel = cmd_arguments .value_of("kernel") .expect("Missing argument: kernel"); let cmdline = cmd_arguments.value_of("cmdline"); let disks: Option> = cmd_arguments.values_of("disk").map(|x| x.collect()); let net: Option> = cmd_arguments.values_of("net").map(|x| x.collect()); let fs: Option> = cmd_arguments.values_of("fs").map(|x| x.collect()); let pmem: Option> = cmd_arguments.values_of("pmem").map(|x| x.collect()); let vm_config = match config::VmConfig::parse(config::VmParams { cpus, memory, kernel, cmdline, disks, net, rng, fs, pmem, serial, }) { Ok(config) => config, Err(e) => { println!("Failed parsing parameters {:?}", e); process::exit(1); } }; println!( "Cloud Hypervisor Guest\n\tvCPUs: {}\n\tMemory: {} MB\ \n\tKernel: {:?}\n\tKernel cmdline: {}\n\tDisk(s): {:?}", u8::from(&vm_config.cpus), vm_config.memory.size >> 20, vm_config.kernel.path, vm_config.cmdline.args.as_str(), vm_config.disks, ); if let Err(e) = vmm::boot_kernel(vm_config) { println!("Guest boot failed: {}", e); process::exit(1); } } #[cfg(test)] #[cfg(feature = "integration_tests")] #[macro_use] extern crate credibility; #[cfg(test)] #[cfg(feature = "integration_tests")] mod tests { use ssh2::Session; use std::fs::{self, read, OpenOptions}; use std::io::{Read, Write}; use std::net::TcpStream; use std::process::Command; use std::string::String; use std::thread; use tempdir::TempDir; fn ssh_command(command: &str) -> String { let mut s = String::new(); #[derive(Debug)] enum Error { Connection, Authentication, Command, }; let mut counter = 0; loop { match (|| -> Result<(), Error> { let tcp = TcpStream::connect("192.168.2.2:22").map_err(|_| Error::Connection)?; let mut sess = Session::new().unwrap(); sess.handshake(&tcp).map_err(|_| Error::Connection)?; sess.userauth_password("admin", "cloud123") .map_err(|_| Error::Authentication)?; assert!(sess.authenticated()); let mut channel = sess.channel_session().map_err(|_| Error::Command)?; channel.exec(command).map_err(|_| Error::Command)?; // Intentionally ignore these results here as their failure // does not precipitate a repeat let _ = channel.read_to_string(&mut s); let _ = channel.close(); let _ = channel.wait_close(); Ok(()) })() { Ok(_) => break, Err(e) => { counter += 1; if counter >= 6 { panic!("Took too many attempts to run command. Last error: {:?}", e); } } }; thread::sleep(std::time::Duration::new(10, 0)); } s } fn prepare_files(tmp_dir: &TempDir) -> (Vec, String) { let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut fw_path = workload_path.clone(); fw_path.push("hypervisor-fw"); let mut osdisk_base_path = workload_path.clone(); osdisk_base_path.push("clear-29810-cloud.img"); let mut osdisk_raw_base_path = workload_path.clone(); osdisk_raw_base_path.push("clear-29810-cloud-raw.img"); let cloudinit_path = String::from("/tmp/cloudinit.img"); let osdisk_path = String::from(tmp_dir.path().join("osdisk.img").to_str().unwrap()); let osdisk_raw_path = String::from(tmp_dir.path().join("osdisk_raw.img").to_str().unwrap()); fs::copy(osdisk_base_path, &osdisk_path).expect("copying of OS source disk image failed"); fs::copy(osdisk_raw_base_path, &osdisk_raw_path) .expect("copying of OS source disk raw image failed"); let disks = vec![osdisk_path, cloudinit_path, osdisk_raw_path]; (disks, String::from(fw_path.to_str().unwrap())) } fn prepare_virtiofsd(tmp_dir: &TempDir) -> (std::process::Child, String) { let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut virtiofsd_path = workload_path.clone(); virtiofsd_path.push("virtiofsd"); let virtiofsd_path = String::from(virtiofsd_path.to_str().unwrap()); let mut shared_dir_path = workload_path.clone(); shared_dir_path.push("shared_dir"); let shared_dir_path = String::from(shared_dir_path.to_str().unwrap()); let virtiofsd_socket_path = String::from(tmp_dir.path().join("virtiofs.sock").to_str().unwrap()); // Start the daemon let child = Command::new(virtiofsd_path.as_str()) .args(&[ "-o", format!("vhost_user_socket={}", virtiofsd_socket_path).as_str(), ]) .args(&["-o", format!("source={}", shared_dir_path).as_str()]) .args(&["-o", "cache=none"]) .spawn() .unwrap(); (child, virtiofsd_socket_path) } fn get_cpu_count() -> u32 { ssh_command("grep -c processor /proc/cpuinfo") .trim() .parse() .unwrap() } fn get_initial_apicid() -> u32 { ssh_command("grep \"initial apicid\" /proc/cpuinfo | grep -o \"[0-9]*\"") .trim() .parse() .unwrap() } fn get_total_memory() -> u32 { ssh_command("grep MemTotal /proc/meminfo | grep -o \"[0-9]*\"") .trim() .parse::() .unwrap() } fn get_entropy() -> u32 { ssh_command("cat /proc/sys/kernel/random/entropy_avail") .trim() .parse::() .unwrap() } fn get_pci_bridge_class() -> String { ssh_command("cat /sys/bus/pci/devices/0000:00:00.0/class") .trim() .to_string() } #[test] fn test_simple_launch() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver_eq!(tb, get_cpu_count(), 1); aver_eq!(tb, get_initial_apicid(), 0); aver!(tb, get_total_memory() > 496_000); aver!(tb, get_entropy() >= 1000); aver_eq!(tb, get_pci_bridge_class(), "0x060000"); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_multi_cpu() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "2"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver_eq!(tb, get_cpu_count(), 2); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_large_memory() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=5120M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver!(tb, get_total_memory() > 5_063_000); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_pci_msi() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver_eq!( tb, ssh_command("grep -c PCI-MSI /proc/interrupts") .trim() .parse::() .unwrap(), 8 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_vmlinux_boot() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, _) = prepare_files(&tmp_dir); let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut kernel_path = workload_path.clone(); kernel_path.push("vmlinux"); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", kernel_path.to_str().unwrap()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&["--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0"]) .args(&["--cmdline", "root=PARTUUID=3cb0e0a5-925d-405e-bc55-edf0cec8f10a console=tty0 console=ttyS0,115200n8 console=hvc0 quiet init=/usr/lib/systemd/systemd-bootchart initcall_debug tsc=reliable no_timer_check noreplace-smp cryptomgr.notests rootfstype=ext4,btrfs,xfs kvm-intel.nested=1 rw"]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver_eq!(tb, get_cpu_count(), 1); aver!(tb, get_total_memory() > 496_000); aver!(tb, get_entropy() >= 1000); aver_eq!( tb, ssh_command("grep -c PCI-MSI /proc/interrupts") .trim() .parse::() .unwrap(), 8 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_bzimage_boot() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, _) = prepare_files(&tmp_dir); let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut kernel_path = workload_path.clone(); kernel_path.push("bzImage"); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", kernel_path.to_str().unwrap()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&["--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0"]) .args(&["--cmdline", "root=PARTUUID=3cb0e0a5-925d-405e-bc55-edf0cec8f10a console=tty0 console=ttyS0,115200n8 console=hvc0 quiet init=/usr/lib/systemd/systemd-bootchart initcall_debug tsc=reliable no_timer_check noreplace-smp cryptomgr.notests rootfstype=ext4,btrfs,xfs kvm-intel.nested=1 rw"]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver_eq!(tb, get_cpu_count(), 1); aver!(tb, get_total_memory() > 496_000); aver!(tb, get_entropy() >= 1000); aver_eq!( tb, ssh_command("grep -c PCI-MSI /proc/interrupts") .trim() .parse::() .unwrap(), 8 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_split_irqchip() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); aver_eq!( tb, ssh_command("cat /proc/interrupts | grep 'IO-APIC' | grep -c 'timer'") .trim() .parse::() .unwrap(), 0 ); aver_eq!( tb, ssh_command("cat /proc/interrupts | grep 'IO-APIC' | grep -c 'cascade'") .trim() .parse::() .unwrap(), 0 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_virtio_fs() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, _) = prepare_files(&tmp_dir); let (mut daemon_child, virtiofsd_socket_path) = prepare_virtiofsd(&tmp_dir); let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut kernel_path = workload_path.clone(); kernel_path.push("vmlinux-custom"); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M,file=/dev/shm"]) .args(&["--kernel", kernel_path.to_str().unwrap()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&["--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0"]) .args(&[ "--fs", format!( "tag=virtiofs,sock={},num_queues=1,queue_size=1024", virtiofsd_socket_path ) .as_str(), ]) .args(&["--cmdline", "root=PARTUUID=3cb0e0a5-925d-405e-bc55-edf0cec8f10a console=tty0 console=ttyS0,115200n8 console=hvc0 quiet init=/usr/lib/systemd/systemd-bootchart initcall_debug tsc=reliable no_timer_check noreplace-smp cryptomgr.notests rootfstype=ext4,btrfs,xfs kvm-intel.nested=1 rw"]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); // Mount shared directory through virtio_fs filesystem aver_eq!( tb, ssh_command("mkdir -p mount_dir && sudo mount -t virtio_fs /dev/null mount_dir/ -o tag=virtiofs,rootmode=040000,user_id=1001,group_id=1001 && echo ok") .trim(), "ok" ); // Check file1 exists and its content is "foo" aver_eq!(tb, ssh_command("cat mount_dir/file1").trim(), "foo"); // Check file2 does not exist aver_ne!( tb, ssh_command("ls mount_dir/file2").trim(), "mount_dir/file2" ); // Check file3 exists and its content is "bar" aver_eq!(tb, ssh_command("cat mount_dir/file3").trim(), "bar"); ssh_command("sudo reboot"); let _ = child.wait(); let _ = daemon_child.wait(); Ok(()) }); } #[test] fn test_virtio_pmem() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, _) = prepare_files(&tmp_dir); let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut kernel_path = workload_path.clone(); kernel_path.push("vmlinux-custom"); let pmem_backend_path = tmp_dir.path().join("/tmp/pmem-file"); let mut pmem_backend_file = OpenOptions::new() .read(true) .write(true) .create(true) .open(&pmem_backend_path) .unwrap(); let pmem_backend_content = "foo"; pmem_backend_file .write_all(pmem_backend_content.as_bytes()) .unwrap(); let pmem_backend_file_size = 0x1000; pmem_backend_file.set_len(pmem_backend_file_size).unwrap(); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", kernel_path.to_str().unwrap()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&["--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0"]) .args(&[ "--pmem", format!( "file={},size={}", pmem_backend_path.to_str().unwrap(), pmem_backend_file_size ) .as_str(), ]) .args(&["--cmdline", "root=PARTUUID=3cb0e0a5-925d-405e-bc55-edf0cec8f10a console=tty0 console=ttyS0,115200n8 console=hvc0 quiet init=/usr/lib/systemd/systemd-bootchart initcall_debug tsc=reliable no_timer_check noreplace-smp cryptomgr.notests rootfstype=ext4,btrfs,xfs kvm-intel.nested=1 rw"]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); // Check for the presence of /dev/pmem0 aver_eq!(tb, ssh_command("ls /dev/pmem0").trim(), "/dev/pmem0"); // Check content aver_eq!( tb, &ssh_command("sudo cat /dev/pmem0").trim()[..pmem_backend_content.len()], pmem_backend_content ); // Modify content let new_content = "bar"; ssh_command( format!( "sudo bash -c 'echo {} > /dev/pmem0' && sudo sync /dev/pmem0", new_content ) .as_str(), ); // Check content from host aver_eq!( tb, &String::from_utf8(read(pmem_backend_path).unwrap()) .unwrap() .as_str()[..new_content.len()], new_content ); ssh_command("sudo reboot"); let _ = child.wait(); Ok(()) }); } #[test] fn test_boot_from_virtio_pmem() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, _) = prepare_files(&tmp_dir); let mut workload_path = dirs::home_dir().unwrap(); workload_path.push("workloads"); let mut kernel_path = workload_path.clone(); kernel_path.push("vmlinux-custom"); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", kernel_path.to_str().unwrap()]) .args(&["--disk", disks[1].as_str()]) .args(&["--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0"]) .args(&[ "--pmem", format!( "file={},size={}", disks[2], fs::metadata(&disks[2]).unwrap().len() ) .as_str(), ]) .args(&["--cmdline", "root=PARTUUID=3cb0e0a5-925d-405e-bc55-edf0cec8f10a console=tty0 console=ttyS0,115200n8 console=hvc0 quiet init=/usr/lib/systemd/systemd-bootchart initcall_debug tsc=reliable no_timer_check noreplace-smp cryptomgr.notests rootfstype=ext4,btrfs,xfs kvm-intel.nested=1 rw"]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); // Simple checks to validate the VM booted properly aver_eq!(tb, get_cpu_count(), 1); aver!(tb, get_total_memory() > 496_000); ssh_command("sudo reboot"); let _ = child.wait(); Ok(()) }); } #[test] fn test_multiple_network_interfaces() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", "tap=,mac=8a:6b:6f:5a:de:ac,ip=192.168.3.1,mask=255.255.255.0", "tap=,mac=fe:1f:9e:e1:60:f2,ip=192.168.4.1,mask=255.255.255.0", ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); // 3 network interfaces + default localhost ==> 4 interfaces aver_eq!( tb, ssh_command("ip -o link | wc -l") .trim() .parse::() .unwrap(), 4 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_serial_disable() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .args(&["--serial", "off"]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); // Test that there is no ttyS0 aver_eq!( tb, ssh_command("cat /proc/interrupts | grep 'IO-APIC' | grep -c 'ttyS0'") .trim() .parse::() .unwrap(), 0 ); // Further test that we're MSI only now aver_eq!( tb, ssh_command("cat /proc/interrupts | grep -c 'IO-APIC'") .trim() .parse::() .unwrap(), 0 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } #[test] fn test_serial_file() { test_block!(tb, "", { let tmp_dir = TempDir::new("ch").unwrap(); let (disks, fw_path) = prepare_files(&tmp_dir); let serial_path = tmp_dir.path().join("/tmp/serial-output"); let mut child = Command::new("target/debug/cloud-hypervisor") .args(&["--cpus", "1"]) .args(&["--memory", "size=512M"]) .args(&["--kernel", fw_path.as_str()]) .args(&["--disk", disks[0].as_str(), disks[1].as_str()]) .args(&[ "--net", "tap=,mac=12:34:56:78:90:ab,ip=192.168.2.1,mask=255.255.255.0", ]) .args(&[ "--serial", format!("file={}", serial_path.to_str().unwrap()).as_str(), ]) .spawn() .unwrap(); thread::sleep(std::time::Duration::new(10, 0)); // Test that there is a ttyS0 aver_eq!( tb, ssh_command("cat /proc/interrupts | grep 'IO-APIC' | grep -c 'ttyS0'") .trim() .parse::() .unwrap(), 1 ); ssh_command("sudo reboot"); thread::sleep(std::time::Duration::new(10, 0)); // Do this check after shutdown of the VM as an easy way to ensure // all writes are flushed to disk let mut f = std::fs::File::open(serial_path).unwrap(); let mut buf = String::new(); f.read_to_string(&mut buf).unwrap(); aver!(tb, buf.contains("cloud login:")); let _ = child.kill(); let _ = child.wait(); Ok(()) }); } }