|  | #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; | 
|  | } |