blob: 4ae51d200374b46ffeb96da786cc0ff9271894d7 [file] [log] [blame] [edit]
#include <kvm/util.h>
#include <kvm/kvm-cmd.h>
#include <kvm/builtin-debug.h>
#include <kvm/kvm.h>
#include <kvm/parse-options.h>
#include <kvm/kvm-ipc.h>
#include <kvm/read-write.h>
#include <stdio.h>
#include <string.h>
#include <signal.h>
#define BUFFER_SIZE 100
static bool all;
static int nmi = -1;
static bool dump;
static const char *instance_name;
static const char *sysrq;
static const char * const debug_usage[] = {
"lkvm debug [--all] [-n name] [-d] [-m vcpu]",
NULL
};
static const struct option debug_options[] = {
OPT_GROUP("General options:"),
OPT_BOOLEAN('d', "dump", &dump, "Generate a debug dump from guest"),
OPT_INTEGER('m', "nmi", &nmi, "Generate NMI on VCPU"),
OPT_STRING('s', "sysrq", &sysrq, "sysrq", "Inject a sysrq"),
OPT_GROUP("Instance options:"),
OPT_BOOLEAN('a', "all", &all, "Debug all instances"),
OPT_STRING('n', "name", &instance_name, "name", "Instance name"),
OPT_END()
};
static void parse_debug_options(int argc, const char **argv)
{
while (argc != 0) {
argc = parse_options(argc, argv, debug_options, debug_usage,
PARSE_OPT_STOP_AT_NON_OPTION);
if (argc != 0)
kvm_debug_help();
}
}
void kvm_debug_help(void)
{
usage_with_options(debug_usage, debug_options);
}
static int do_debug(const char *name, int sock)
{
char buff[BUFFER_SIZE];
struct debug_cmd_params cmd = {.dbg_type = 0};
int r;
if (dump)
cmd.dbg_type |= KVM_DEBUG_CMD_TYPE_DUMP;
if (nmi != -1) {
cmd.dbg_type |= KVM_DEBUG_CMD_TYPE_NMI;
cmd.cpu = nmi;
}
if (sysrq) {
cmd.dbg_type |= KVM_DEBUG_CMD_TYPE_SYSRQ;
cmd.sysrq = sysrq[0];
}
r = kvm_ipc__send_msg(sock, KVM_IPC_DEBUG, sizeof(cmd), (u8 *)&cmd);
if (r < 0)
return r;
if (!dump)
return 0;
do {
r = xread(sock, buff, BUFFER_SIZE);
if (r < 0)
return 0;
printf("%.*s", r, buff);
} while (r > 0);
return 0;
}
int kvm_cmd_debug(int argc, const char **argv, const char *prefix)
{
parse_debug_options(argc, argv);
int instance;
int r;
if (all)
return kvm__enumerate_instances(do_debug);
if (instance_name == NULL)
kvm_debug_help();
instance = kvm__get_sock_by_instance(instance_name);
if (instance <= 0)
die("Failed locating instance");
r = do_debug(instance_name, instance);
close(instance);
return r;
}