#include #include #include #include #include #include #include #include #include #include #define BUFFER_SIZE 100 static bool all; static int instance; static int nmi = -1; static bool dump; static const char *instance_name; 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_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 cmd = {KVM_IPC_DEBUG, 2 * sizeof(u32)}; int r; if (dump) cmd.params.dbg_type |= KVM_DEBUG_CMD_TYPE_DUMP; if (nmi != -1) { cmd.params.dbg_type |= KVM_DEBUG_CMD_TYPE_NMI; cmd.params.cpu = nmi; } r = xwrite(sock, &cmd, sizeof(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); if (all) return kvm__enumerate_instances(do_debug); if (instance_name == NULL && instance == 0) kvm_debug_help(); if (instance_name) instance = kvm__get_sock_by_instance(instance_name); if (instance <= 0) die("Failed locating instance"); return do_debug(instance_name, instance); }