xref: /kvmtool/builtin-debug.c (revision b7d2f0130b0f8c99a273d2e28186c1d0f15268ff)
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