// SPDX-License-Identifier: GPL-2.0-or-later /* * trace-event-scripting. Scripting engine common and initialization code. * * Copyright (C) 2009-2010 Tom Zanussi */ #include #include #include #include #ifdef HAVE_LIBTRACEEVENT #include #endif #include "archinsn.h" #include "debug.h" #include "event.h" #include "trace-event.h" #include "evsel.h" #include #include #include "util/sample.h" unsigned int scripting_max_stack = PERF_MAX_STACK_DEPTH; struct scripting_context *scripting_context; struct script_spec { struct list_head node; struct scripting_ops *ops; char spec[]; }; static LIST_HEAD(script_specs); static struct script_spec *script_spec__new(const char *spec, struct scripting_ops *ops) { struct script_spec *s = malloc(sizeof(*s) + strlen(spec) + 1); if (s != NULL) { strcpy(s->spec, spec); s->ops = ops; } return s; } static void script_spec__add(struct script_spec *s) { list_add_tail(&s->node, &script_specs); } static struct script_spec *script_spec__find(const char *spec) { struct script_spec *s; list_for_each_entry(s, &script_specs, node) if (strcasecmp(s->spec, spec) == 0) return s; return NULL; } static int script_spec_register(const char *spec, struct scripting_ops *ops) { struct script_spec *s; s = script_spec__find(spec); if (s) return -1; s = script_spec__new(spec, ops); if (!s) return -1; script_spec__add(s); return 0; } struct scripting_ops *script_spec__lookup(const char *spec) { struct script_spec *s = script_spec__find(spec); if (!s) return NULL; return s->ops; } int script_spec__for_each(int (*cb)(struct scripting_ops *ops, const char *spec)) { struct script_spec *s; int ret = 0; list_for_each_entry(s, &script_specs, node) { ret = cb(s->ops, s->spec); if (ret) break; } return ret; } void scripting_context__update(struct scripting_context *c, union perf_event *event, struct perf_sample *sample, struct evsel *evsel, struct addr_location *al, struct addr_location *addr_al) { #ifdef HAVE_LIBTRACEEVENT const struct tep_event *tp_format = evsel__tp_format(evsel); c->pevent = tp_format ? tp_format->tep : NULL; #else c->pevent = NULL; #endif c->event_data = sample->raw_data; c->event = event; c->sample = sample; c->evsel = evsel; c->al = al; c->addr_al = addr_al; } static int flush_script_unsupported(void) { return 0; } static int stop_script_unsupported(void) { return 0; } static void process_event_unsupported(union perf_event *event __maybe_unused, struct perf_sample *sample __maybe_unused, struct evsel *evsel __maybe_unused, struct addr_location *al __maybe_unused, struct addr_location *addr_al __maybe_unused) { } static void print_python_unsupported_msg(void) { fprintf(stderr, "Python scripting not supported." " Install libpython and rebuild perf to enable it.\n" "For example:\n # apt-get install python-dev (ubuntu)" "\n # yum install python-devel (Fedora)" "\n etc.\n"); } static int python_start_script_unsupported(const char *script __maybe_unused, int argc __maybe_unused, const char **argv __maybe_unused, struct perf_session *session __maybe_unused) { print_python_unsupported_msg(); return -1; } static int python_generate_script_unsupported(struct tep_handle *pevent __maybe_unused, const char *outfile __maybe_unused) { print_python_unsupported_msg(); return -1; } struct scripting_ops python_scripting_unsupported_ops = { .name = "Python", .dirname = "python", .start_script = python_start_script_unsupported, .flush_script = flush_script_unsupported, .stop_script = stop_script_unsupported, .process_event = process_event_unsupported, .generate_script = python_generate_script_unsupported, }; static void register_python_scripting(struct scripting_ops *scripting_ops) { if (scripting_context == NULL) scripting_context = malloc(sizeof(*scripting_context)); if (scripting_context == NULL || script_spec_register("Python", scripting_ops) || script_spec_register("py", scripting_ops)) { pr_err("Error registering Python script extension: disabling it\n"); zfree(&scripting_context); } } #ifndef HAVE_LIBPYTHON_SUPPORT void setup_python_scripting(void) { register_python_scripting(&python_scripting_unsupported_ops); } #else extern struct scripting_ops python_scripting_ops; void setup_python_scripting(void) { register_python_scripting(&python_scripting_ops); } #endif #ifdef HAVE_LIBTRACEEVENT static void print_perl_unsupported_msg(void) { fprintf(stderr, "Perl scripting not supported." " Install libperl and rebuild perf to enable it.\n" "For example:\n # apt-get install libperl-dev (ubuntu)" "\n # yum install 'perl(ExtUtils::Embed)' (Fedora)" "\n etc.\n"); } static int perl_start_script_unsupported(const char *script __maybe_unused, int argc __maybe_unused, const char **argv __maybe_unused, struct perf_session *session __maybe_unused) { print_perl_unsupported_msg(); return -1; } static int perl_generate_script_unsupported(struct tep_handle *pevent __maybe_unused, const char *outfile __maybe_unused) { print_perl_unsupported_msg(); return -1; } struct scripting_ops perl_scripting_unsupported_ops = { .name = "Perl", .dirname = "perl", .start_script = perl_start_script_unsupported, .flush_script = flush_script_unsupported, .stop_script = stop_script_unsupported, .process_event = process_event_unsupported, .generate_script = perl_generate_script_unsupported, }; static void register_perl_scripting(struct scripting_ops *scripting_ops) { if (scripting_context == NULL) scripting_context = malloc(sizeof(*scripting_context)); if (scripting_context == NULL || script_spec_register("Perl", scripting_ops) || script_spec_register("pl", scripting_ops)) { pr_err("Error registering Perl script extension: disabling it\n"); zfree(&scripting_context); } } #ifndef HAVE_LIBPERL_SUPPORT void setup_perl_scripting(void) { register_perl_scripting(&perl_scripting_unsupported_ops); } #else extern struct scripting_ops perl_scripting_ops; void setup_perl_scripting(void) { register_perl_scripting(&perl_scripting_ops); } #endif #endif #if !defined(__i386__) && !defined(__x86_64__) void arch_fetch_insn(struct perf_sample *sample __maybe_unused, struct thread *thread __maybe_unused, struct machine *machine __maybe_unused) { } #endif void script_fetch_insn(struct perf_sample *sample, struct thread *thread, struct machine *machine, bool native_arch) { if (sample->insn_len == 0 && native_arch) arch_fetch_insn(sample, thread, machine); } static const struct { u32 flags; const char *name; } sample_flags[] = { {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL, "call"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_RETURN, "return"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CONDITIONAL, "jcc"}, {PERF_IP_FLAG_BRANCH, "jmp"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_INTERRUPT, "int"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_RETURN | PERF_IP_FLAG_INTERRUPT, "iret"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_SYSCALLRET, "syscall"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_RETURN | PERF_IP_FLAG_SYSCALLRET, "sysret"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_ASYNC, "async"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_ASYNC | PERF_IP_FLAG_INTERRUPT, "hw int"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_TX_ABORT, "tx abrt"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_TRACE_BEGIN, "tr strt"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_TRACE_END, "tr end"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_VMENTRY, "vmentry"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_VMEXIT, "vmexit"}, {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_BRANCH_MISS, "br miss"}, {0, NULL} }; static const char *sample_flags_to_name(u32 flags) { int i; for (i = 0; sample_flags[i].name ; i++) { if (sample_flags[i].flags == flags) return sample_flags[i].name; } return NULL; } int perf_sample__sprintf_flags(u32 flags, char *str, size_t sz) { u32 xf = PERF_IP_FLAG_IN_TX | PERF_IP_FLAG_INTR_DISABLE | PERF_IP_FLAG_INTR_TOGGLE; const char *chars = PERF_IP_FLAG_CHARS; const size_t n = strlen(PERF_IP_FLAG_CHARS); const char *name = NULL; size_t i, pos = 0; char xs[16] = {0}; if (flags & xf) snprintf(xs, sizeof(xs), "(%s%s%s)", flags & PERF_IP_FLAG_IN_TX ? "x" : "", flags & PERF_IP_FLAG_INTR_DISABLE ? "D" : "", flags & PERF_IP_FLAG_INTR_TOGGLE ? "t" : ""); name = sample_flags_to_name(flags & ~xf); if (name) return snprintf(str, sz, "%-15s%6s", name, xs); if (flags & PERF_IP_FLAG_TRACE_BEGIN) { name = sample_flags_to_name(flags & ~(xf | PERF_IP_FLAG_TRACE_BEGIN)); if (name) return snprintf(str, sz, "tr strt %-7s%6s", name, xs); } if (flags & PERF_IP_FLAG_TRACE_END) { name = sample_flags_to_name(flags & ~(xf | PERF_IP_FLAG_TRACE_END)); if (name) return snprintf(str, sz, "tr end %-7s%6s", name, xs); } for (i = 0; i < n; i++, flags >>= 1) { if ((flags & 1) && pos < sz) str[pos++] = chars[i]; } for (; i < 32; i++, flags >>= 1) { if ((flags & 1) && pos < sz) str[pos++] = '?'; } if (pos < sz) str[pos] = 0; return pos; }