Files
kvmtool/main.c
T
Cyrill Gorcunov 6fd7308d51 kvm: The command line storage should not be too short
At moment boot protocol may hold up to 2048 symbols
from command line we should provide the same compatibility
for --params option.

Reported-by: Pekka Enberg <penberg@cs.helsinki.fi>
Signed-off-by: Cyrill Gorcunov <gorcunov@gmail.com>
2015-06-01 16:39:39 +01:00

143 lines
2.8 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>
static void usage(char *argv[])
{
fprintf(stderr, " usage: %s [--single-step] [--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];
int main(int argc, char *argv[])
{
const char *kernel_filename = NULL;
const char *kernel_cmdline = NULL;
bool single_step = false;
int i;
signal(SIGQUIT, handle_sigquit);
for (i = 1; i < argc; i++) {
if (!strncmp("--kernel=", argv[i], 9)) {
kernel_filename = &argv[i][9];
continue;
} else if (!strncmp("--params=", argv[i], 9)) {
kernel_cmdline = &argv[i][9];
continue;
} else if (!strncmp("--single-step", argv[i], 13)) {
single_step = 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__setup_cpuid(kvm);
strcpy(real_cmdline, "notsc nolapic nosmp noacpi 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);
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;
}