at master 10 kB view raw
1// SPDX-License-Identifier: GPL-2.0-or-later 2/* 3 * trace-event-scripting. Scripting engine common and initialization code. 4 * 5 * Copyright (C) 2009-2010 Tom Zanussi <tzanussi@gmail.com> 6 */ 7 8#include <stdio.h> 9#include <stdlib.h> 10#include <string.h> 11#include <errno.h> 12#ifdef HAVE_LIBTRACEEVENT 13#include <event-parse.h> 14#endif 15 16#include "archinsn.h" 17#include "debug.h" 18#include "event.h" 19#include "trace-event.h" 20#include "evsel.h" 21#include <linux/perf_event.h> 22#include <linux/zalloc.h> 23#include "util/sample.h" 24 25unsigned int scripting_max_stack = PERF_MAX_STACK_DEPTH; 26 27struct scripting_context *scripting_context; 28 29struct script_spec { 30 struct list_head node; 31 struct scripting_ops *ops; 32 char spec[]; 33}; 34 35static LIST_HEAD(script_specs); 36 37static struct script_spec *script_spec__new(const char *spec, 38 struct scripting_ops *ops) 39{ 40 struct script_spec *s = malloc(sizeof(*s) + strlen(spec) + 1); 41 42 if (s != NULL) { 43 strcpy(s->spec, spec); 44 s->ops = ops; 45 } 46 47 return s; 48} 49 50static void script_spec__add(struct script_spec *s) 51{ 52 list_add_tail(&s->node, &script_specs); 53} 54 55static struct script_spec *script_spec__find(const char *spec) 56{ 57 struct script_spec *s; 58 59 list_for_each_entry(s, &script_specs, node) 60 if (strcasecmp(s->spec, spec) == 0) 61 return s; 62 return NULL; 63} 64 65static int script_spec_register(const char *spec, struct scripting_ops *ops) 66{ 67 struct script_spec *s; 68 69 s = script_spec__find(spec); 70 if (s) 71 return -1; 72 73 s = script_spec__new(spec, ops); 74 if (!s) 75 return -1; 76 77 script_spec__add(s); 78 return 0; 79} 80 81struct scripting_ops *script_spec__lookup(const char *spec) 82{ 83 struct script_spec *s = script_spec__find(spec); 84 85 if (!s) 86 return NULL; 87 88 return s->ops; 89} 90 91int script_spec__for_each(int (*cb)(struct scripting_ops *ops, const char *spec)) 92{ 93 struct script_spec *s; 94 int ret = 0; 95 96 list_for_each_entry(s, &script_specs, node) { 97 ret = cb(s->ops, s->spec); 98 if (ret) 99 break; 100 } 101 return ret; 102} 103 104void scripting_context__update(struct scripting_context *c, 105 union perf_event *event, 106 struct perf_sample *sample, 107 struct evsel *evsel, 108 struct addr_location *al, 109 struct addr_location *addr_al) 110{ 111#ifdef HAVE_LIBTRACEEVENT 112 const struct tep_event *tp_format = evsel__tp_format(evsel); 113 114 c->pevent = tp_format ? tp_format->tep : NULL; 115#else 116 c->pevent = NULL; 117#endif 118 c->event_data = sample->raw_data; 119 c->event = event; 120 c->sample = sample; 121 c->evsel = evsel; 122 c->al = al; 123 c->addr_al = addr_al; 124} 125 126static int flush_script_unsupported(void) 127{ 128 return 0; 129} 130 131static int stop_script_unsupported(void) 132{ 133 return 0; 134} 135 136static void process_event_unsupported(union perf_event *event __maybe_unused, 137 struct perf_sample *sample __maybe_unused, 138 struct evsel *evsel __maybe_unused, 139 struct addr_location *al __maybe_unused, 140 struct addr_location *addr_al __maybe_unused) 141{ 142} 143 144static void print_python_unsupported_msg(void) 145{ 146 fprintf(stderr, "Python scripting not supported." 147 " Install libpython and rebuild perf to enable it.\n" 148 "For example:\n # apt-get install python-dev (ubuntu)" 149 "\n # yum install python-devel (Fedora)" 150 "\n etc.\n"); 151} 152 153static int python_start_script_unsupported(const char *script __maybe_unused, 154 int argc __maybe_unused, 155 const char **argv __maybe_unused, 156 struct perf_session *session __maybe_unused) 157{ 158 print_python_unsupported_msg(); 159 160 return -1; 161} 162 163static int python_generate_script_unsupported(struct tep_handle *pevent 164 __maybe_unused, 165 const char *outfile 166 __maybe_unused) 167{ 168 print_python_unsupported_msg(); 169 170 return -1; 171} 172 173struct scripting_ops python_scripting_unsupported_ops = { 174 .name = "Python", 175 .dirname = "python", 176 .start_script = python_start_script_unsupported, 177 .flush_script = flush_script_unsupported, 178 .stop_script = stop_script_unsupported, 179 .process_event = process_event_unsupported, 180 .generate_script = python_generate_script_unsupported, 181}; 182 183static void register_python_scripting(struct scripting_ops *scripting_ops) 184{ 185 if (scripting_context == NULL) 186 scripting_context = malloc(sizeof(*scripting_context)); 187 188 if (scripting_context == NULL || 189 script_spec_register("Python", scripting_ops) || 190 script_spec_register("py", scripting_ops)) { 191 pr_err("Error registering Python script extension: disabling it\n"); 192 zfree(&scripting_context); 193 } 194} 195 196#ifndef HAVE_LIBPYTHON_SUPPORT 197void setup_python_scripting(void) 198{ 199 register_python_scripting(&python_scripting_unsupported_ops); 200} 201#else 202extern struct scripting_ops python_scripting_ops; 203 204void setup_python_scripting(void) 205{ 206 register_python_scripting(&python_scripting_ops); 207} 208#endif 209 210#ifdef HAVE_LIBTRACEEVENT 211static void print_perl_unsupported_msg(void) 212{ 213 fprintf(stderr, "Perl scripting not supported." 214 " Install libperl and rebuild perf to enable it.\n" 215 "For example:\n # apt-get install libperl-dev (ubuntu)" 216 "\n # yum install 'perl(ExtUtils::Embed)' (Fedora)" 217 "\n etc.\n"); 218} 219 220static int perl_start_script_unsupported(const char *script __maybe_unused, 221 int argc __maybe_unused, 222 const char **argv __maybe_unused, 223 struct perf_session *session __maybe_unused) 224{ 225 print_perl_unsupported_msg(); 226 227 return -1; 228} 229 230static int perl_generate_script_unsupported(struct tep_handle *pevent 231 __maybe_unused, 232 const char *outfile __maybe_unused) 233{ 234 print_perl_unsupported_msg(); 235 236 return -1; 237} 238 239struct scripting_ops perl_scripting_unsupported_ops = { 240 .name = "Perl", 241 .dirname = "perl", 242 .start_script = perl_start_script_unsupported, 243 .flush_script = flush_script_unsupported, 244 .stop_script = stop_script_unsupported, 245 .process_event = process_event_unsupported, 246 .generate_script = perl_generate_script_unsupported, 247}; 248 249static void register_perl_scripting(struct scripting_ops *scripting_ops) 250{ 251 if (scripting_context == NULL) 252 scripting_context = malloc(sizeof(*scripting_context)); 253 254 if (scripting_context == NULL || 255 script_spec_register("Perl", scripting_ops) || 256 script_spec_register("pl", scripting_ops)) { 257 pr_err("Error registering Perl script extension: disabling it\n"); 258 zfree(&scripting_context); 259 } 260} 261 262#ifndef HAVE_LIBPERL_SUPPORT 263void setup_perl_scripting(void) 264{ 265 register_perl_scripting(&perl_scripting_unsupported_ops); 266} 267#else 268extern struct scripting_ops perl_scripting_ops; 269 270void setup_perl_scripting(void) 271{ 272 register_perl_scripting(&perl_scripting_ops); 273} 274#endif 275#endif 276 277#if !defined(__i386__) && !defined(__x86_64__) 278void arch_fetch_insn(struct perf_sample *sample __maybe_unused, 279 struct thread *thread __maybe_unused, 280 struct machine *machine __maybe_unused) 281{ 282} 283#endif 284 285void script_fetch_insn(struct perf_sample *sample, struct thread *thread, 286 struct machine *machine, bool native_arch) 287{ 288 if (sample->insn_len == 0 && native_arch) 289 arch_fetch_insn(sample, thread, machine); 290} 291 292static const struct { 293 u32 flags; 294 const char *name; 295} sample_flags[] = { 296 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL, "call"}, 297 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_RETURN, "return"}, 298 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CONDITIONAL, "jcc"}, 299 {PERF_IP_FLAG_BRANCH, "jmp"}, 300 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_INTERRUPT, "int"}, 301 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_RETURN | PERF_IP_FLAG_INTERRUPT, "iret"}, 302 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_SYSCALLRET, "syscall"}, 303 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_RETURN | PERF_IP_FLAG_SYSCALLRET, "sysret"}, 304 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_ASYNC, "async"}, 305 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_ASYNC | PERF_IP_FLAG_INTERRUPT, 306 "hw int"}, 307 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_TX_ABORT, "tx abrt"}, 308 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_TRACE_BEGIN, "tr strt"}, 309 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_TRACE_END, "tr end"}, 310 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_VMENTRY, "vmentry"}, 311 {PERF_IP_FLAG_BRANCH | PERF_IP_FLAG_CALL | PERF_IP_FLAG_VMEXIT, "vmexit"}, 312 {0, NULL} 313}; 314 315static const struct { 316 u32 flags; 317 const char *name; 318} branch_events[] = { 319 {PERF_IP_FLAG_BRANCH_MISS, "miss"}, 320 {PERF_IP_FLAG_NOT_TAKEN, "not_taken"}, 321 {0, NULL} 322}; 323 324static int sample_flags_to_name(u32 flags, char *str, size_t size) 325{ 326 int i; 327 const char *prefix; 328 int pos = 0, ret, ev_idx = 0; 329 u32 xf = flags & PERF_ADDITIONAL_STATE_MASK; 330 u32 types, events; 331 char xs[16] = { 0 }; 332 333 /* Clear additional state bits */ 334 flags &= ~PERF_ADDITIONAL_STATE_MASK; 335 336 if (flags & PERF_IP_FLAG_TRACE_BEGIN) 337 prefix = "tr strt "; 338 else if (flags & PERF_IP_FLAG_TRACE_END) 339 prefix = "tr end "; 340 else 341 prefix = ""; 342 343 ret = snprintf(str + pos, size - pos, "%s", prefix); 344 if (ret < 0) 345 return ret; 346 pos += ret; 347 348 flags &= ~(PERF_IP_FLAG_TRACE_BEGIN | PERF_IP_FLAG_TRACE_END); 349 350 types = flags & ~PERF_IP_FLAG_BRANCH_EVENT_MASK; 351 for (i = 0; sample_flags[i].name; i++) { 352 if (sample_flags[i].flags != types) 353 continue; 354 355 ret = snprintf(str + pos, size - pos, "%s", sample_flags[i].name); 356 if (ret < 0) 357 return ret; 358 pos += ret; 359 break; 360 } 361 362 events = flags & PERF_IP_FLAG_BRANCH_EVENT_MASK; 363 for (i = 0; branch_events[i].name; i++) { 364 if (!(branch_events[i].flags & events)) 365 continue; 366 367 ret = snprintf(str + pos, size - pos, !ev_idx ? "/%s" : ",%s", 368 branch_events[i].name); 369 if (ret < 0) 370 return ret; 371 pos += ret; 372 ev_idx++; 373 } 374 375 /* Add an end character '/' for events */ 376 if (ev_idx) { 377 ret = snprintf(str + pos, size - pos, "/"); 378 if (ret < 0) 379 return ret; 380 pos += ret; 381 } 382 383 if (!xf) 384 return pos; 385 386 snprintf(xs, sizeof(xs), "(%s%s%s)", 387 flags & PERF_IP_FLAG_IN_TX ? "x" : "", 388 flags & PERF_IP_FLAG_INTR_DISABLE ? "D" : "", 389 flags & PERF_IP_FLAG_INTR_TOGGLE ? "t" : ""); 390 391 /* Right align the string if its length is less than the limit */ 392 if ((pos + strlen(xs)) < SAMPLE_FLAGS_STR_ALIGNED_SIZE) 393 ret = snprintf(str + pos, size - pos, "%*s", 394 (int)(SAMPLE_FLAGS_STR_ALIGNED_SIZE - ret), xs); 395 else 396 ret = snprintf(str + pos, size - pos, " %s", xs); 397 if (ret < 0) 398 return ret; 399 400 return pos + ret; 401} 402 403int perf_sample__sprintf_flags(u32 flags, char *str, size_t sz) 404{ 405 const char *chars = PERF_IP_FLAG_CHARS; 406 const size_t n = strlen(PERF_IP_FLAG_CHARS); 407 size_t i, pos = 0; 408 int ret; 409 410 ret = sample_flags_to_name(flags, str, sz); 411 if (ret > 0) 412 return ret; 413 414 for (i = 0; i < n; i++, flags >>= 1) { 415 if ((flags & 1) && pos < sz) 416 str[pos++] = chars[i]; 417 } 418 for (; i < 32; i++, flags >>= 1) { 419 if ((flags & 1) && pos < sz) 420 str[pos++] = '?'; 421 } 422 if (pos < sz) 423 str[pos] = 0; 424 425 return pos; 426}