1ca379b83SPekka Enberg #include <kvm/util.h> 2ca379b83SPekka Enberg #include <kvm/kvm-cmd.h> 32a24f96dSSasha Levin #include <kvm/builtin-debug.h> 46c757e71SSasha Levin #include <kvm/kvm.h> 5c9cba791SPekka Enberg #include <kvm/parse-options.h> 64b1addaeSSasha Levin #include <kvm/kvm-ipc.h> 7*b7d2f013SSasha Levin #include <kvm/read-write.h> 86c757e71SSasha Levin 96c757e71SSasha Levin #include <stdio.h> 106c757e71SSasha Levin #include <string.h> 116c757e71SSasha Levin #include <signal.h> 12ca379b83SPekka Enberg 13*b7d2f013SSasha Levin #define BUFFER_SIZE 100 14*b7d2f013SSasha Levin 158d7b77f6SSasha Levin static bool all; 164b1addaeSSasha Levin static int instance; 178d7b77f6SSasha Levin static const char *instance_name; 188d7b77f6SSasha Levin 194b1addaeSSasha Levin struct debug_cmd { 204b1addaeSSasha Levin u32 type; 214b1addaeSSasha Levin u32 len; 224b1addaeSSasha Levin }; 234b1addaeSSasha Levin 24c9cba791SPekka Enberg static const char * const debug_usage[] = { 254b1addaeSSasha Levin "kvm debug [--all] [-n name]", 26c9cba791SPekka Enberg NULL 27c9cba791SPekka Enberg }; 28c9cba791SPekka Enberg 29c9cba791SPekka Enberg static const struct option debug_options[] = { 308d7b77f6SSasha Levin OPT_GROUP("General options:"), 318d7b77f6SSasha Levin OPT_BOOLEAN('a', "all", &all, "Debug all instances"), 328d7b77f6SSasha Levin OPT_STRING('n', "name", &instance_name, "name", "Instance name"), 33c9cba791SPekka Enberg OPT_END() 34c9cba791SPekka Enberg }; 35c9cba791SPekka Enberg 368d7b77f6SSasha Levin static void parse_debug_options(int argc, const char **argv) 378d7b77f6SSasha Levin { 388d7b77f6SSasha Levin while (argc != 0) { 398d7b77f6SSasha Levin argc = parse_options(argc, argv, debug_options, debug_usage, 408d7b77f6SSasha Levin PARSE_OPT_STOP_AT_NON_OPTION); 418d7b77f6SSasha Levin if (argc != 0) 428d7b77f6SSasha Levin kvm_debug_help(); 438d7b77f6SSasha Levin } 448d7b77f6SSasha Levin } 458d7b77f6SSasha Levin 460725673aSSasha Levin void kvm_debug_help(void) 470725673aSSasha Levin { 480725673aSSasha Levin usage_with_options(debug_usage, debug_options); 490725673aSSasha Levin } 500725673aSSasha Levin 514b1addaeSSasha Levin static int do_debug(const char *name, int sock) 5224e7dbbaSSasha Levin { 53*b7d2f013SSasha Levin char buff[BUFFER_SIZE]; 544b1addaeSSasha Levin struct debug_cmd cmd = {KVM_IPC_DEBUG, 0}; 554b1addaeSSasha Levin int r; 564b1addaeSSasha Levin 57*b7d2f013SSasha Levin r = xwrite(sock, &cmd, sizeof(cmd)); 584b1addaeSSasha Levin if (r < 0) 594b1addaeSSasha Levin return r; 604b1addaeSSasha Levin 61*b7d2f013SSasha Levin do { 62*b7d2f013SSasha Levin r = xread(sock, buff, BUFFER_SIZE); 63*b7d2f013SSasha Levin if (r < 0) 64*b7d2f013SSasha Levin return 0; 65*b7d2f013SSasha Levin printf("%.*s", r, buff); 66*b7d2f013SSasha Levin } while (r > 0); 67*b7d2f013SSasha Levin 684b1addaeSSasha Levin return 0; 6924e7dbbaSSasha Levin } 7024e7dbbaSSasha Levin 71ca379b83SPekka Enberg int kvm_cmd_debug(int argc, const char **argv, const char *prefix) 72ca379b83SPekka Enberg { 738d7b77f6SSasha Levin parse_debug_options(argc, argv); 746c757e71SSasha Levin 758d7b77f6SSasha Levin if (all) 768d7b77f6SSasha Levin return kvm__enumerate_instances(do_debug); 778d7b77f6SSasha Levin 788d7b77f6SSasha Levin if (instance_name == NULL && 794b1addaeSSasha Levin instance == 0) 800725673aSSasha Levin kvm_debug_help(); 816c757e71SSasha Levin 828d7b77f6SSasha Levin if (instance_name) 834b1addaeSSasha Levin instance = kvm__get_sock_by_instance(instance_name); 8424e7dbbaSSasha Levin 854b1addaeSSasha Levin if (instance <= 0) 868d7b77f6SSasha Levin die("Failed locating instance"); 876c757e71SSasha Levin 884b1addaeSSasha Levin return do_debug(instance_name, instance); 89ca379b83SPekka Enberg } 90