Re: [PATCH trace-cmd] plugin_kvm: disassemble instructions for kvm_emulate_insn

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Sun, 2010-09-19 at 16:33 +0200, Avi Kivity wrote:
> Override kvm_emulate_insn formatting to use a disassembler to format
> the emulated instruction.  If a disassembler (udis86) is not available,
> fall back to showing the instruction bytes in hex.
> 
> Signed-off-by: Avi Kivity <avi@xxxxxxxxxx>
> ---
> 
> Note 1: on top of 'master' with 'trace-cmd-kvm' cherry-picked on top.

Thanks, I merged trace-cmd-kvm into master and applied your patch
(haven't pushed yet).


> 
> Note 2: I get output of the form
> 
> ... kvm_emulate_insn:     0:fffff800010527b5: mov $0x0, 0xfffe00b0CAN'T FIND FIELD "guest_rip"
> 
> which leads me to believe there is a bug in trace_seq_printf when the input
> to %s is "".

I ran this under gdb (nice to do that, where I don't in kernel :-)  And
it takes me to kvm_emulate_insn_handler() which does the
trace_seq_printf() fine, but then calls pevent_print_num_field() and
that passes in "guest_rip" where we get the "CAN'T FIND FIELD" error.

In pevent_print_num_field() it searches for "guest_rip" at the top of
the function (pevent_find_field()), but the event kvm_emulate_insn does
not have a "guest_rip" field, then it jumps to the error message.

-- Steve

> 
>  Makefile     |   11 +++++-
>  plugin_kvm.c |  111 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
>  2 files changed, 121 insertions(+), 1 deletions(-)
> 
> diff --git a/Makefile b/Makefile
> index 5282f94..fe34d1c 100644
> --- a/Makefile
> +++ b/Makefile
> @@ -74,6 +74,14 @@ ifeq ($(shell sh -c "python-config --includes > /dev/null 2>&1 && echo y"), y)
>  	PYTHON_PY_INSTALL := event-viewer.install tracecmd.install tracecmdgui.install
>  endif
>  
> +# $(call test-build, snippet, ret) -> ret if snippet compiles
> +#                                  -> empty otherwise
> +test-build = $(if $(shell $(CC) -o /dev/null -c -x c - > /dev/null 2>&1 \
> +	                  <<<'$1' && echo y), $2)
> +
> +# have udis86 disassembler library?
> +udis86-flags := $(call test-build,\#include <udis86.h>,-DHAVE_UDIS86 -ludis86)
> +
>  ifeq ("$(origin O)", "command line")
>    BUILD_OUTPUT := $(O)
>  endif
> @@ -188,6 +196,7 @@ CFLAGS ?= -g -Wall
>  
>  # Append required CFLAGS
>  override CFLAGS += $(CONFIG_FLAGS) $(INCLUDES) $(PLUGIN_DIR_SQ)
> +override CFLAGS += $(udis86-flags)
>  
>  ifeq ($(VERBOSE),1)
>    Q =
> @@ -228,7 +237,7 @@ do_compile_plugin_obj =				\
>  
>  do_plugin_build =				\
>  	($(print_plugin_build)			\
> -	$(CC) -shared -nostartfiles -o $@ $<)
> +	$(CC) $(CFLAGS) -shared -nostartfiles -o $@ $<)
>  
>  do_build_static_lib =				\
>  	($(print_static_lib_build)		\
> diff --git a/plugin_kvm.c b/plugin_kvm.c
> index 7217e85..00cac5a 100644
> --- a/plugin_kvm.c
> +++ b/plugin_kvm.c
> @@ -21,9 +21,68 @@
>  #include <stdio.h>
>  #include <stdlib.h>
>  #include <string.h>
> +#include <stdint.h>
>  
>  #include "parse-events.h"
>  
> +#ifdef HAVE_UDIS86
> +
> +#include <udis86.h>
> +
> +static ud_t ud;
> +
> +static void init_disassembler(void)
> +{
> +	ud_init(&ud);
> +	ud_set_syntax(&ud, UD_SYN_ATT);
> +}
> +
> +static const char *disassemble(unsigned char *insn, int len, uint64_t rip,
> +			       int cr0_pe, int eflags_vm,
> +			       int cs_d, int cs_l)
> +{
> +	int mode;
> +
> +	if (!cr0_pe)
> +		mode = 16;
> +	else if (eflags_vm)
> +		mode = 16;
> +	else if (cs_l)
> +		mode = 64;
> +	else if (cs_d)
> +		mode = 32;
> +	else
> +		mode = 16;
> +
> +	ud_set_pc(&ud, rip);
> +	ud_set_mode(&ud, mode);
> +	ud_set_input_buffer(&ud, insn, len);
> +	ud_disassemble(&ud);
> +	return ud_insn_asm(&ud);
> +}
> +
> +#else
> +
> +static void init_disassembler(void)
> +{
> +}
> +
> +static const char *disassemble(unsigned char *insn, int len, uint64_t rip,
> +			       int cr0_pe, int eflags_vm,
> +			       int cs_d, int cs_l)
> +{
> +	static char out[15*3+1];
> +	int i;
> +
> +	for (i = 0; i < len; ++i)
> +		sprintf(out + i * 3, "%02x ", insn[i]);
> +	out[len*3-1] = '\0';
> +	return out;
> +}
> +
> +#endif
> +
> +
>  #define VMX_EXIT_REASONS			\
>  	_ER(EXCEPTION_NMI,	0)		\
>  	_ER(EXTERNAL_INTERRUPT,	1)		\
> @@ -99,6 +158,53 @@ static int kvm_exit_handler(struct trace_seq *s, struct record *record,
>  	return 0;
>  }
>  
> +#define KVM_EMUL_INSN_F_CR0_PE (1 << 0)
> +#define KVM_EMUL_INSN_F_EFL_VM (1 << 1)
> +#define KVM_EMUL_INSN_F_CS_D   (1 << 2)
> +#define KVM_EMUL_INSN_F_CS_L   (1 << 3)
> +
> +static int kvm_emulate_insn_handler(struct trace_seq *s, struct record *record,
> +				    struct event_format *event, void *context)
> +{
> +	unsigned long long rip, csbase, len, flags, failed;
> +	int llen;
> +	uint8_t *insn;
> +	const char *disasm;
> +
> +	if (pevent_get_field_val(s, event, "rip", record, &rip, 1) < 0)
> +		return -1;
> +
> +	if (pevent_get_field_val(s, event, "csbase", record, &csbase, 1) < 0)
> +		return -1;
> +
> +	if (pevent_get_field_val(s, event, "len", record, &len, 1) < 0)
> +		return -1;
> +
> +	if (pevent_get_field_val(s, event, "flags", record, &flags, 1) < 0)
> +		return -1;
> +
> +	if (pevent_get_field_val(s, event, "failed", record, &failed, 1) < 0)
> +		return -1;
> +
> +	insn = pevent_get_field_raw(s, event, "insn", record, &llen, 1);
> +	if (!insn)
> +		return -1;
> +
> +	disasm = disassemble(insn, len, rip,
> +			     flags & KVM_EMUL_INSN_F_CR0_PE,
> +			     flags & KVM_EMUL_INSN_F_EFL_VM,
> +			     flags & KVM_EMUL_INSN_F_CS_D,
> +			     flags & KVM_EMUL_INSN_F_CS_L);
> +
> +	trace_seq_printf(s, "%llx:%llx: %s%s", csbase, rip, disasm,
> +			 failed ? " FAIL" : "");
> +
> +	pevent_print_num_field(s, " rip %0xlx", event, "guest_rip", record, 1);
> +
> +	return 0;
> +}
> +
> +
>  static int kvm_nested_vmexit_inject_handler(struct trace_seq *s, struct record *record,
>  					    struct event_format *event, void *context)
>  {
> @@ -199,9 +305,14 @@ static int kvm_mmu_get_page_handler(struct trace_seq *s, struct record *record,
>  
>  int PEVENT_PLUGIN_LOADER(struct pevent *pevent)
>  {
> +	init_disassembler();
> +
>  	pevent_register_event_handler(pevent, -1, "kvm", "kvm_exit",
>  				      kvm_exit_handler, NULL);
>  
> +	pevent_register_event_handler(pevent, -1, "kvm", "kvm_emulate_insn",
> +				      kvm_emulate_insn_handler, NULL);
> +
>  	pevent_register_event_handler(pevent, -1, "kvm", "kvm_nested_vmexit",
>  				      kvm_nested_vmexit_handler, NULL);
>  


--
To unsubscribe from this list: send the line "unsubscribe kvm" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [KVM ARM]     [KVM ia64]     [KVM ppc]     [Virtualization Tools]     [Spice Development]     [Libvirt]     [Libvirt Users]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite Questions]     [Linux Kernel]     [Linux SCSI]     [XFree86]
  Powered by Linux