#include "kvm/builtin-run.h" #include "kvm/builtin-setup.h" #include "kvm/virtio-balloon.h" #include "kvm/virtio-console.h" #include "kvm/parse-options.h" #include "kvm/8250-serial.h" #include "kvm/framebuffer.h" #include "kvm/disk-image.h" #include "kvm/threadpool.h" #include "kvm/virtio-scsi.h" #include "kvm/virtio-blk.h" #include "kvm/virtio-net.h" #include "kvm/virtio-rng.h" #include "kvm/ioeventfd.h" #include "kvm/virtio-9p.h" #include "kvm/barrier.h" #include "kvm/kvm-cpu.h" #include "kvm/ioport.h" #include "kvm/symbol.h" #include "kvm/i8042.h" #include "kvm/mutex.h" #include "kvm/term.h" #include "kvm/util.h" #include "kvm/strbuf.h" #include "kvm/vesa.h" #include "kvm/irq.h" #include "kvm/kvm.h" #include "kvm/pci.h" #include "kvm/rtc.h" #include "kvm/sdl.h" #include "kvm/vnc.h" #include "kvm/guest_compat.h" #include "kvm/pci-shmem.h" #include "kvm/kvm-ipc.h" #include "kvm/builtin-debug.h" #include #include #include #include #include #include #include #include #include #include #include #include #define MB_SHIFT (20) #define KB_SHIFT (10) #define GB_SHIFT (30) struct kvm *kvm; __thread struct kvm_cpu *current_kvm_cpu; static int kvm_run_wrapper; bool do_debug_print = false; extern char _binary_guest_init_start; extern char _binary_guest_init_size; static const char * const run_usage[] = { "lkvm run [] []", NULL }; enum { KVM_RUN_DEFAULT, KVM_RUN_SANDBOX, }; static int img_name_parser(const struct option *opt, const char *arg, int unset) { char path[PATH_MAX]; struct stat st; struct kvm *kvm = opt->ptr; if (stat(arg, &st) == 0 && S_ISDIR(st.st_mode)) { char tmp[PATH_MAX]; if (kvm->cfg.using_rootfs) die("Please use only one rootfs directory atmost"); if (realpath(arg, tmp) == 0 || virtio_9p__register(kvm, tmp, "/dev/root") < 0) die("Unable to initialize virtio 9p"); kvm->cfg.using_rootfs = 1; return 0; } snprintf(path, PATH_MAX, "%s%s", kvm__get_dir(), arg); if (stat(path, &st) == 0 && S_ISDIR(st.st_mode)) { char tmp[PATH_MAX]; if (kvm->cfg.using_rootfs) die("Please use only one rootfs directory atmost"); if (realpath(path, tmp) == 0 || virtio_9p__register(kvm, tmp, "/dev/root") < 0) die("Unable to initialize virtio 9p"); if (virtio_9p__register(kvm, "/", "hostfs") < 0) die("Unable to initialize virtio 9p"); kvm_setup_resolv(arg); kvm->cfg.using_rootfs = kvm->cfg.custom_rootfs = 1; kvm->cfg.custom_rootfs_name = arg; return 0; } return disk_img_name_parser(opt, arg, unset); } void kvm_run_set_wrapper_sandbox(void) { kvm_run_wrapper = KVM_RUN_SANDBOX; } static int virtio_9p_rootdir_parser(const struct option *opt, const char *arg, int unset) { char *tag_name; char tmp[PATH_MAX]; /* * 9p dir can be of the form dirname,tag_name or * just dirname. In the later case we use the * default tag name */ tag_name = strstr(arg, ","); if (tag_name) { *tag_name = '\0'; tag_name++; } if (realpath(arg, tmp)) { if (virtio_9p__register(kvm, tmp, tag_name) < 0) die("Unable to initialize virtio 9p"); } else die("Failed resolving 9p path"); return 0; } #define BUILD_OPTIONS(name, cfg, kvm) \ struct option name[] = { \ OPT_GROUP("Basic options:"), \ OPT_STRING('\0', "name", &(cfg)->guest_name, "guest name", \ "A name for the guest"), \ OPT_INTEGER('c', "cpus", &(cfg)->nrcpus, "Number of CPUs"), \ OPT_U64('m', "mem", &(cfg)->ram_size, "Virtual machine memory size\ in MiB."), \ OPT_CALLBACK('\0', "shmem", NULL, \ "[pci:]:[:handle=][:create]", \ "Share host shmem with guest via pci device", \ shmem_parser, NULL), \ OPT_CALLBACK('d', "disk", kvm, "image or rootfs_dir", "Disk \ image or rootfs directory", img_name_parser, \ kvm), \ OPT_BOOLEAN('\0', "balloon", &(cfg)->balloon, "Enable virtio \ balloon"), \ OPT_BOOLEAN('\0', "vnc", &(cfg)->vnc, "Enable VNC framebuffer"),\ OPT_BOOLEAN('\0', "sdl", &(cfg)->sdl, "Enable SDL framebuffer"),\ OPT_BOOLEAN('\0', "rng", &(cfg)->virtio_rng, "Enable virtio Random\ Number Generator"), \ OPT_CALLBACK('\0', "9p", NULL, "dir_to_share,tag_name", \ "Enable virtio 9p to share files between host and \ guest", virtio_9p_rootdir_parser, NULL), \ OPT_STRING('\0', "console", &(cfg)->console, "serial, virtio or \ hv", "Console to use"), \ OPT_STRING('\0', "dev", &(cfg)->dev, "device_file", \ "KVM device file"), \ OPT_CALLBACK('\0', "tty", NULL, "tty id", \ "Remap guest TTY into a pty on the host", \ tty_parser, NULL), \ OPT_STRING('\0', "sandbox", &(cfg)->sandbox, "script", \ "Run this script when booting into custom \ rootfs"), \ OPT_STRING('\0', "hugetlbfs", &(cfg)->hugetlbfs_path, "path", \ "Hugetlbfs path"), \ \ OPT_GROUP("Kernel options:"), \ OPT_STRING('k', "kernel", &(cfg)->kernel_filename, "kernel", \ "Kernel to boot in virtual machine"), \ OPT_STRING('i', "initrd", &(cfg)->initrd_filename, "initrd", \ "Initial RAM disk image"), \ OPT_STRING('p', "params", &(cfg)->kernel_cmdline, "params", \ "Kernel command line arguments"), \ OPT_STRING('f', "firmware", &(cfg)->firmware_filename, "firmware",\ "Firmware image to boot in virtual machine"), \ \ OPT_GROUP("Networking options:"), \ OPT_CALLBACK_DEFAULT('n', "network", NULL, "network params", \ "Create a new guest NIC", \ netdev_parser, NULL, kvm), \ OPT_BOOLEAN('\0', "no-dhcp", &(cfg)->no_dhcp, "Disable kernel DHCP\ in rootfs mode"), \ \ OPT_GROUP("BIOS options:"), \ OPT_INTEGER('\0', "vidmode", &(cfg)->vidmode, \ "Video mode"), \ \ OPT_GROUP("Debug options:"), \ OPT_BOOLEAN('\0', "debug", &do_debug_print, \ "Enable debug messages"), \ OPT_BOOLEAN('\0', "debug-single-step", &(cfg)->single_step, \ "Enable single stepping"), \ OPT_BOOLEAN('\0', "debug-ioport", &(cfg)->ioport_debug, \ "Enable ioport debugging"), \ OPT_BOOLEAN('\0', "debug-mmio", &(cfg)->mmio_debug, \ "Enable MMIO debugging"), \ OPT_INTEGER('\0', "debug-iodelay", &(cfg)->debug_iodelay, \ "Delay IO by millisecond"), \ OPT_END() \ }; static void handle_sigalrm(int sig) { kvm__arch_periodic_poll(kvm); } static void *kvm_cpu_thread(void *arg) { current_kvm_cpu = arg; if (kvm_cpu__start(current_kvm_cpu)) goto panic_kvm; return (void *) (intptr_t) 0; panic_kvm: fprintf(stderr, "KVM exit reason: %u (\"%s\")\n", current_kvm_cpu->kvm_run->exit_reason, kvm_exit_reasons[current_kvm_cpu->kvm_run->exit_reason]); if (current_kvm_cpu->kvm_run->exit_reason == KVM_EXIT_UNKNOWN) fprintf(stderr, "KVM exit code: 0x%Lu\n", current_kvm_cpu->kvm_run->hw.hardware_exit_reason); kvm_cpu__set_debug_fd(STDOUT_FILENO); kvm_cpu__show_registers(current_kvm_cpu); kvm_cpu__show_code(current_kvm_cpu); kvm_cpu__show_page_tables(current_kvm_cpu); return (void *) (intptr_t) 1; } static char kernel[PATH_MAX]; static const char *host_kernels[] = { "/boot/vmlinuz", "/boot/bzImage", NULL }; static const char *default_kernels[] = { "./bzImage", "arch/" BUILD_ARCH "/boot/bzImage", "../../arch/" BUILD_ARCH "/boot/bzImage", NULL }; static const char *default_vmlinux[] = { "vmlinux", "../../../vmlinux", "../../vmlinux", NULL }; static void kernel_usage_with_options(void) { const char **k; struct utsname uts; fprintf(stderr, "Fatal: could not find default kernel image in:\n"); k = &default_kernels[0]; while (*k) { fprintf(stderr, "\t%s\n", *k); k++; } if (uname(&uts) < 0) return; k = &host_kernels[0]; while (*k) { if (snprintf(kernel, PATH_MAX, "%s-%s", *k, uts.release) < 0) return; fprintf(stderr, "\t%s\n", kernel); k++; } fprintf(stderr, "\nPlease see '%s run --help' for more options.\n\n", KVM_BINARY_NAME); } static u64 host_ram_size(void) { long page_size; long nr_pages; nr_pages = sysconf(_SC_PHYS_PAGES); if (nr_pages < 0) { pr_warning("sysconf(_SC_PHYS_PAGES) failed"); return 0; } page_size = sysconf(_SC_PAGE_SIZE); if (page_size < 0) { pr_warning("sysconf(_SC_PAGE_SIZE) failed"); return 0; } return (nr_pages * page_size) >> MB_SHIFT; } /* * If user didn't specify how much memory it wants to allocate for the guest, * avoid filling the whole host RAM. */ #define RAM_SIZE_RATIO 0.8 static u64 get_ram_size(int nr_cpus) { u64 available; u64 ram_size; ram_size = 64 * (nr_cpus + 3); available = host_ram_size() * RAM_SIZE_RATIO; if (!available) available = MIN_RAM_SIZE_MB; if (ram_size > available) ram_size = available; return ram_size; } static const char *find_kernel(void) { const char **k; struct stat st; struct utsname uts; k = &default_kernels[0]; while (*k) { if (stat(*k, &st) < 0 || !S_ISREG(st.st_mode)) { k++; continue; } strncpy(kernel, *k, PATH_MAX); return kernel; } if (uname(&uts) < 0) return NULL; k = &host_kernels[0]; while (*k) { if (snprintf(kernel, PATH_MAX, "%s-%s", *k, uts.release) < 0) return NULL; if (stat(kernel, &st) < 0 || !S_ISREG(st.st_mode)) { k++; continue; } return kernel; } return NULL; } static const char *find_vmlinux(void) { const char **vmlinux; vmlinux = &default_vmlinux[0]; while (*vmlinux) { struct stat st; if (stat(*vmlinux, &st) < 0 || !S_ISREG(st.st_mode)) { vmlinux++; continue; } return *vmlinux; } return NULL; } void kvm_run_help(void) { BUILD_OPTIONS(options, &kvm->cfg, kvm); usage_with_options(run_usage, options); } static int kvm_setup_guest_init(void) { const char *rootfs = kvm->cfg.custom_rootfs_name; char tmp[PATH_MAX]; size_t size; int fd, ret; char *data; /* Setup /virt/init */ size = (size_t)&_binary_guest_init_size; data = (char *)&_binary_guest_init_start; snprintf(tmp, PATH_MAX, "%s%s/virt/init", kvm__get_dir(), rootfs); remove(tmp); fd = open(tmp, O_CREAT | O_WRONLY, 0755); if (fd < 0) die("Fail to setup %s", tmp); ret = xwrite(fd, data, size); if (ret < 0) die("Fail to setup %s", tmp); close(fd); return 0; } static int kvm_run_set_sandbox(void) { const char *guestfs_name = kvm->cfg.custom_rootfs_name; char path[PATH_MAX], script[PATH_MAX], *tmp; snprintf(path, PATH_MAX, "%s%s/virt/sandbox.sh", kvm__get_dir(), guestfs_name); remove(path); if (kvm->cfg.sandbox == NULL) return 0; tmp = realpath(kvm->cfg.sandbox, NULL); if (tmp == NULL) return -ENOMEM; snprintf(script, PATH_MAX, "/host/%s", tmp); free(tmp); return symlink(script, path); } static void kvm_write_sandbox_cmd_exactly(int fd, const char *arg) { const char *single_quote; if (!*arg) { /* zero length string */ if (write(fd, "''", 2) <= 0) die("Failed writing sandbox script"); return; } while (*arg) { single_quote = strchrnul(arg, '\''); /* write non-single-quote string as #('string') */ if (arg != single_quote) { if (write(fd, "'", 1) <= 0 || write(fd, arg, single_quote - arg) <= 0 || write(fd, "'", 1) <= 0) die("Failed writing sandbox script"); } /* write single quote as #("'") */ if (*single_quote) { if (write(fd, "\"'\"", 3) <= 0) die("Failed writing sandbox script"); } else break; arg = single_quote + 1; } } static void resolve_program(const char *src, char *dst, size_t len) { struct stat st; int err; err = stat(src, &st); if (!err && S_ISREG(st.st_mode)) { char resolved_path[PATH_MAX]; if (!realpath(src, resolved_path)) die("Unable to resolve program %s: %s\n", src, strerror(errno)); snprintf(dst, len, "/host%s", resolved_path); } else strncpy(dst, src, len); } static void kvm_run_write_sandbox_cmd(const char **argv, int argc) { const char script_hdr[] = "#! /bin/bash\n\n"; char program[PATH_MAX]; int fd; remove(kvm->cfg.sandbox); fd = open(kvm->cfg.sandbox, O_RDWR | O_CREAT, 0777); if (fd < 0) die("Failed creating sandbox script"); if (write(fd, script_hdr, sizeof(script_hdr) - 1) <= 0) die("Failed writing sandbox script"); resolve_program(argv[0], program, PATH_MAX); kvm_write_sandbox_cmd_exactly(fd, program); argv++; argc--; while (argc) { if (write(fd, " ", 1) <= 0) die("Failed writing sandbox script"); kvm_write_sandbox_cmd_exactly(fd, argv[0]); argv++; argc--; } if (write(fd, "\n", 1) <= 0) die("Failed writing sandbox script"); close(fd); } static int kvm_cmd_run_init(int argc, const char **argv) { static char real_cmdline[2048], default_name[20]; unsigned int nr_online_cpus; int r; kvm = kvm__new(); if (IS_ERR(kvm)) return PTR_ERR(kvm); signal(SIGALRM, handle_sigalrm); nr_online_cpus = sysconf(_SC_NPROCESSORS_ONLN); kvm->cfg.custom_rootfs_name = "default"; while (argc != 0) { BUILD_OPTIONS(options, &kvm->cfg, kvm); argc = parse_options(argc, argv, options, run_usage, PARSE_OPT_STOP_AT_NON_OPTION | PARSE_OPT_KEEP_DASHDASH); if (argc != 0) { /* Cusrom options, should have been handled elsewhere */ if (strcmp(argv[0], "--") == 0) { if (kvm_run_wrapper == KVM_RUN_SANDBOX) { kvm->cfg.sandbox = DEFAULT_SANDBOX_FILENAME; kvm_run_write_sandbox_cmd(argv+1, argc-1); break; } } if ((kvm_run_wrapper == KVM_RUN_DEFAULT && kvm->cfg.kernel_filename) || (kvm_run_wrapper == KVM_RUN_SANDBOX && kvm->cfg.sandbox)) { fprintf(stderr, "Cannot handle parameter: " "%s\n", argv[0]); usage_with_options(run_usage, options); free(kvm); return -EINVAL; } if (kvm_run_wrapper == KVM_RUN_SANDBOX) { /* * first unhandled parameter is treated as * sandbox command */ kvm->cfg.sandbox = DEFAULT_SANDBOX_FILENAME; kvm_run_write_sandbox_cmd(argv, argc); } else { /* * first unhandled parameter is treated as a kernel * image */ kvm->cfg.kernel_filename = argv[0]; } argv++; argc--; } } kvm->nr_disks = kvm->cfg.image_count; if (!kvm->cfg.kernel_filename) kvm->cfg.kernel_filename = find_kernel(); if (!kvm->cfg.kernel_filename) { kernel_usage_with_options(); return -EINVAL; } kvm->cfg.vmlinux_filename = find_vmlinux(); kvm->vmlinux = kvm->cfg.vmlinux_filename; if (kvm->cfg.nrcpus == 0) kvm->cfg.nrcpus = nr_online_cpus; if (!kvm->cfg.ram_size) kvm->cfg.ram_size = get_ram_size(kvm->cfg.nrcpus); if (kvm->cfg.ram_size < MIN_RAM_SIZE_MB) die("Not enough memory specified: %lluMB (min %lluMB)", kvm->cfg.ram_size, MIN_RAM_SIZE_MB); if (kvm->cfg.ram_size > host_ram_size()) pr_warning("Guest memory size %lluMB exceeds host physical RAM size %lluMB", kvm->cfg.ram_size, host_ram_size()); kvm->cfg.ram_size <<= MB_SHIFT; if (!kvm->cfg.dev) kvm->cfg.dev = DEFAULT_KVM_DEV; if (!kvm->cfg.console) kvm->cfg.console = DEFAULT_CONSOLE; if (!strncmp(kvm->cfg.console, "virtio", 6)) kvm->cfg.active_console = CONSOLE_VIRTIO; else if (!strncmp(kvm->cfg.console, "serial", 6)) kvm->cfg.active_console = CONSOLE_8250; else if (!strncmp(kvm->cfg.console, "hv", 2)) kvm->cfg.active_console = CONSOLE_HV; else pr_warning("No console!"); if (!kvm->cfg.host_ip) kvm->cfg.host_ip = DEFAULT_HOST_ADDR; if (!kvm->cfg.guest_ip) kvm->cfg.guest_ip = DEFAULT_GUEST_ADDR; if (!kvm->cfg.guest_mac) kvm->cfg.guest_mac = DEFAULT_GUEST_MAC; if (!kvm->cfg.host_mac) kvm->cfg.host_mac = DEFAULT_HOST_MAC; if (!kvm->cfg.script) kvm->cfg.script = DEFAULT_SCRIPT; if (!kvm->cfg.vnc && !kvm->cfg.sdl) kvm->cfg.vidmode = -1; memset(real_cmdline, 0, sizeof(real_cmdline)); kvm__arch_set_cmdline(real_cmdline, kvm->cfg.vnc || kvm->cfg.sdl); if (strlen(real_cmdline) > 0) strcat(real_cmdline, " "); if (kvm->cfg.kernel_cmdline) strlcat(real_cmdline, kvm->cfg.kernel_cmdline, sizeof(real_cmdline)); if (!kvm->cfg.guest_name) { if (kvm->cfg.custom_rootfs) { kvm->cfg.guest_name = kvm->cfg.custom_rootfs_name; } else { sprintf(default_name, "guest-%u", getpid()); kvm->cfg.guest_name = default_name; } } if (!kvm->cfg.using_rootfs && !kvm->cfg.disk_image[0].filename && !kvm->cfg.initrd_filename) { char tmp[PATH_MAX]; kvm_setup_create_new(kvm->cfg.custom_rootfs_name); kvm_setup_resolv(kvm->cfg.custom_rootfs_name); snprintf(tmp, PATH_MAX, "%s%s", kvm__get_dir(), "default"); if (virtio_9p__register(kvm, tmp, "/dev/root") < 0) die("Unable to initialize virtio 9p"); if (virtio_9p__register(kvm, "/", "hostfs") < 0) die("Unable to initialize virtio 9p"); kvm->cfg.using_rootfs = kvm->cfg.custom_rootfs = 1; } if (kvm->cfg.using_rootfs) { strcat(real_cmdline, " root=/dev/root rw rootflags=rw,trans=virtio,version=9p2000.L rootfstype=9p"); if (kvm->cfg.custom_rootfs) { kvm_run_set_sandbox(); strcat(real_cmdline, " init=/virt/init"); if (!kvm->cfg.no_dhcp) strcat(real_cmdline, " ip=dhcp"); if (kvm_setup_guest_init()) die("Failed to setup init for guest."); } } else if (!strstr(real_cmdline, "root=")) { strlcat(real_cmdline, " root=/dev/vda rw ", sizeof(real_cmdline)); } kvm->cfg.real_cmdline = real_cmdline; printf(" # %s run -k %s -m %Lu -c %d --name %s\n", KVM_BINARY_NAME, kvm->cfg.kernel_filename, kvm->cfg.ram_size / 1024 / 1024, kvm->cfg.nrcpus, kvm->cfg.guest_name); r = kvm__init(kvm); if (r) goto fail; r = term_init(kvm); if (r < 0) { pr_err("term_init() failed with error %d\n", r); goto fail; } r = ioeventfd__init(kvm); if (r < 0) { pr_err("ioeventfd__init() failed with error %d\n", r); goto fail; } r = kvm_cpu__init(kvm); if (r < 0) { pr_err("kvm_cpu__init() failed with error %d\n", r); goto fail; } r = irq__init(kvm); if (r < 0) { pr_err("irq__init() failed with error %d\n", r); goto fail; } r = pci__init(kvm); if (r < 0) { pr_err("pci__init() failed with error %d\n", r); goto fail; } r = ioport__init(kvm); if (r < 0) { pr_err("ioport__init() failed with error %d\n", r); goto fail; } r = disk_image__init(kvm); if (r < 0) { pr_err("disk_image__init() failed with error %d\n", r); goto fail; } r = symbol_init(kvm); if (r < 0) pr_debug("symbol_init() failed with error %d\n", r); r = rtc__init(kvm); if (r < 0) { pr_err("rtc__init() failed with error %d\n", r); goto fail; } r = serial8250__init(kvm); if (r < 0) { pr_err("serial__init() failed with error %d\n", r); goto fail; } r = virtio_blk__init(kvm); if (r < 0) { pr_err("virtio_blk__init() failed with error %d\n", r); goto fail; } r = virtio_scsi_init(kvm); if (r < 0) { pr_err("virtio_scsi_init() failed with error %d\n", r); goto fail; } r = virtio_console__init(kvm); if (r < 0) { pr_err("virtio_console__init() failed with error %d\n", r); goto fail; } r = virtio_rng__init(kvm); if (r < 0) { pr_err("virtio_rng__init() failed with error %d\n", r); goto fail; } r = virtio_bln__init(kvm); if (r < 0) { pr_err("virtio_rng__init() failed with error %d\n", r); goto fail; } if (!kvm->cfg.network) kvm->cfg.network = DEFAULT_NETWORK; virtio_9p__init(kvm); r = virtio_net__init(kvm); if (r < 0) { pr_err("virtio_net__init() failed with error %d\n", r); goto fail; } kvm__init_ram(kvm); r = kbd__init(kvm); if (r < 0) { pr_err("kbd__init() failed with error %d\n", r); goto fail; } r = pci_shmem__init(kvm); if (r < 0) { pr_err("pci_shmem__init() failed with error %d\n", r); goto fail; } r = vnc__init(kvm); if (r < 0) { pr_err("vnc__init() failed with error %d\n", r); goto fail; } r = sdl__init(kvm); if (r < 0) { pr_err("sdl__init() failed with error %d\n", r); goto fail; } r = fb__init(kvm); if (r < 0) { pr_err("fb__init() failed with error %d\n", r); goto fail; } /* * Device init all done; firmware init must * come after this (it may set up device trees etc.) */ r = kvm_timer__init(kvm); if (r < 0) { pr_err("kvm_timer__init() failed with error %d\n", r); goto fail; } r = thread_pool__init(kvm); if (r < 0) { pr_err("thread_pool__init() failed with error %d\n", r); goto fail; } fail: return r; } static int kvm_cmd_run_work(void) { int i; void *ret = NULL; for (i = 0; i < kvm->nrcpus; i++) { if (pthread_create(&kvm->cpus[i]->thread, NULL, kvm_cpu_thread, kvm->cpus[i]) != 0) die("unable to create KVM VCPU thread"); } /* Only VCPU #0 is going to exit by itself when shutting down */ return pthread_join(kvm->cpus[0]->thread, &ret); } static void kvm_cmd_run_exit(int guest_ret) { int r = 0; compat__print_all_messages(); r = kvm_cpu__exit(kvm); if (r < 0) pr_warning("kvm_cpu__exit() failed with error %d\n", r); r = symbol_exit(kvm); if (r < 0) pr_warning("symbol_exit() failed with error %d\n", r); r = irq__exit(kvm); if (r < 0) pr_warning("irq__exit() failed with error %d\n", r); r = kvm_timer__exit(kvm); if (r < 0) pr_warning("kvm_timer__exit() failed with error %d\n", r); r = fb__exit(kvm); if (r < 0) pr_warning("kvm_timer__exit() failed with error %d\n", r); r = virtio_net__exit(kvm); if (r < 0) pr_warning("virtio_net__exit() failed with error %d\n", r); r = virtio_scsi_exit(kvm); if (r < 0) pr_warning("virtio_scsi_exit() failed with error %d\n", r); r = virtio_blk__exit(kvm); if (r < 0) pr_warning("virtio_blk__exit() failed with error %d\n", r); r = virtio_rng__exit(kvm); if (r < 0) pr_warning("virtio_rng__exit() failed with error %d\n", r); r = virtio_bln__exit(kvm); if (r < 0) pr_warning("virtio_bln__exit() failed with error %d\n", r); r = virtio_console__exit(kvm); if (r < 0) pr_warning("virtio_console__exit() failed with error %d\n", r); r = pci_shmem__exit(kvm); if (r < 0) pr_warning("pci_shmem__exit() failed with error %d\n", r); r = disk_image__exit(kvm); if (r < 0) pr_warning("disk_image__exit() failed with error %d\n", r); r = serial8250__exit(kvm); if (r < 0) pr_warning("serial8250__exit() failed with error %d\n", r); r = rtc__exit(kvm); if (r < 0) pr_warning("rtc__exit() failed with error %d\n", r); r = kvm__arch_free_firmware(kvm); if (r < 0) pr_warning("kvm__arch_free_firmware() failed with error %d\n", r); r = ioport__exit(kvm); if (r < 0) pr_warning("ioport__exit() failed with error %d\n", r); r = ioeventfd__exit(kvm); if (r < 0) pr_warning("ioeventfd__exit() failed with error %d\n", r); r = pci__exit(kvm); if (r < 0) pr_warning("pci__exit() failed with error %d\n", r); r = term_exit(kvm); if (r < 0) pr_warning("pci__exit() failed with error %d\n", r); r = thread_pool__exit(kvm); if (r < 0) pr_warning("thread_pool__exit() failed with error %d\n", r); r = kvm__exit(kvm); if (r < 0) pr_warning("pci__exit() failed with error %d\n", r); if (guest_ret == 0) printf("\n # KVM session ended normally.\n"); } int kvm_cmd_run(int argc, const char **argv, const char *prefix) { int r, ret = -EFAULT; r = kvm_cmd_run_init(argc, argv); if (r < 0) return r; ret = kvm_cmd_run_work(); kvm_cmd_run_exit(ret); return ret; }