mirror of
https://github.com/clearlinux/kvmtool.git
synced 2026-06-15 18:05:49 +00:00
2f3976eeee
This patch adds BIOS E820 memory map emulation to KVM. Signed-off-by: Pekka Enberg <penberg@cs.helsinki.fi>
159 lines
3.3 KiB
C
159 lines
3.3 KiB
C
#include "kvm/kvm.h"
|
|
|
|
#include "kvm/early_printk.h"
|
|
#include "kvm/util.h"
|
|
|
|
#include <inttypes.h>
|
|
#include <signal.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
|
|
extern bool ioport_debug;
|
|
|
|
static void usage(char *argv[])
|
|
{
|
|
fprintf(stderr, " usage: %s [--single-step] [--ioport-debug] [--params=<kernel-params>] [--kernel=]<kernel-image>\n",
|
|
argv[0]);
|
|
exit(1);
|
|
}
|
|
|
|
static struct kvm *kvm;
|
|
|
|
static void handle_sigquit(int sig)
|
|
{
|
|
kvm__show_registers(kvm);
|
|
kvm__show_code(kvm);
|
|
kvm__show_page_tables(kvm);
|
|
|
|
kvm__delete(kvm);
|
|
|
|
exit(1);
|
|
}
|
|
|
|
static char real_cmdline[2048];
|
|
|
|
static bool option_matches(char *arg, const char *option)
|
|
{
|
|
return !strncmp(arg, option, strlen(option));
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
const char *kernel_filename = NULL;
|
|
const char *kernel_cmdline = NULL;
|
|
const char *kvm_dev = "/dev/kvm";
|
|
bool single_step = false;
|
|
int i;
|
|
|
|
signal(SIGQUIT, handle_sigquit);
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
if (option_matches(argv[i], "--kernel=")) {
|
|
kernel_filename = &argv[i][9];
|
|
continue;
|
|
} else if (option_matches(argv[i], "--params=")) {
|
|
kernel_cmdline = &argv[i][9];
|
|
continue;
|
|
} else if (option_matches(argv[i], "--kvm-dev=")) {
|
|
kvm_dev = &argv[i][10];
|
|
continue;
|
|
} else if (option_matches(argv[i], "--single-step")) {
|
|
single_step = true;
|
|
continue;
|
|
} else if (option_matches(argv[i], "--ioport-debug")) {
|
|
ioport_debug = true;
|
|
continue;
|
|
} else {
|
|
/* any unspecified arg is kernel image */
|
|
if (argv[i][0] != '-')
|
|
kernel_filename = argv[i];
|
|
else
|
|
warning("Unknown option: %s", argv[i]);
|
|
}
|
|
}
|
|
|
|
/* at least we should have kernel image passed */
|
|
if (!kernel_filename)
|
|
usage(argv);
|
|
|
|
kvm = kvm__init(kvm_dev);
|
|
|
|
kvm__setup_cpuid(kvm);
|
|
|
|
strcpy(real_cmdline, "notsc nolapic nosmp noacpi nopci pci=off earlyprintk=serial,keep ");
|
|
if (kernel_cmdline) {
|
|
strlcat(real_cmdline, kernel_cmdline, sizeof(real_cmdline));
|
|
real_cmdline[sizeof(real_cmdline)-1] = '\0';
|
|
}
|
|
|
|
if (!kvm__load_kernel(kvm, kernel_filename, real_cmdline))
|
|
die("unable to load kernel %s", kernel_filename);
|
|
|
|
kvm__reset_vcpu(kvm);
|
|
|
|
kvm__setup_mem(kvm);
|
|
|
|
if (single_step)
|
|
kvm__enable_singlestep(kvm);
|
|
|
|
early_printk__init();
|
|
|
|
for (;;) {
|
|
kvm__run(kvm);
|
|
|
|
switch (kvm->kvm_run->exit_reason) {
|
|
case KVM_EXIT_DEBUG:
|
|
kvm__show_registers(kvm);
|
|
kvm__show_code(kvm);
|
|
break;
|
|
case KVM_EXIT_IO: {
|
|
bool ret;
|
|
|
|
ret = kvm__emulate_io(kvm,
|
|
kvm->kvm_run->io.port,
|
|
(uint8_t *)kvm->kvm_run + kvm->kvm_run->io.data_offset,
|
|
kvm->kvm_run->io.direction,
|
|
kvm->kvm_run->io.size,
|
|
kvm->kvm_run->io.count);
|
|
|
|
if (!ret)
|
|
goto exit_kvm;
|
|
break;
|
|
}
|
|
case KVM_EXIT_MMIO: {
|
|
bool ret;
|
|
|
|
ret = kvm__emulate_mmio(kvm,
|
|
kvm->kvm_run->mmio.phys_addr,
|
|
kvm->kvm_run->mmio.data,
|
|
kvm->kvm_run->mmio.len,
|
|
kvm->kvm_run->mmio.is_write);
|
|
|
|
if (!ret)
|
|
goto exit_kvm;
|
|
break;
|
|
|
|
}
|
|
default:
|
|
goto exit_kvm;
|
|
}
|
|
}
|
|
|
|
exit_kvm:
|
|
|
|
fprintf(stderr, "KVM exit reason: %" PRIu32 " (\"%s\")\n",
|
|
kvm->kvm_run->exit_reason, kvm_exit_reasons[kvm->kvm_run->exit_reason]);
|
|
if (kvm->kvm_run->exit_reason == KVM_EXIT_UNKNOWN)
|
|
fprintf(stderr, "KVM exit code: 0x%016llx\n",
|
|
kvm->kvm_run->hw.hardware_exit_reason);
|
|
|
|
kvm__show_registers(kvm);
|
|
kvm__show_code(kvm);
|
|
kvm__show_page_tables(kvm);
|
|
kvm__delete(kvm);
|
|
|
|
return 0;
|
|
}
|