mirror of
https://github.com/torvalds/linux.git
synced 2024-12-12 14:12:51 +00:00
316769f757
libtraceevent has added more levels of debug printout and with changes like: https://lore.kernel.org/linux-trace-devel/20210507095022.1079364-3-tz.stoyanov@gmail.com previously generated output like "registering plugin" is no longer displayed. This change makes it so that if perf's verbose debug output is enabled then the debug and info libtraceevent messages can be displayed. This change was previously posted: https://lore.kernel.org/linux-perf-users/20210923001024.550263-4-irogers@google.com/ and reverted: https://lore.kernel.org/linux-perf-users/20220109153446.160593-1-acme@kernel.org/ The previous failure was due to -Itools/lib being on the include path and libtraceevent in tools/lib being version 1.1.0. This meant that when LIBTRACEEVENT_VERSION was 1.3.0 the #if succeeded, but the header file for libtraceevent (taken from tools/lib rather than the intended /usr/include) was for version 1.1.0 and function definitions were missing. Since the previous issue the -Itools/lib include path has been removed: https://lore.kernel.org/lkml/20221109184914.1357295-1-irogers@google.com/ As well as libtraceevent 1.1.0 has been removed from tools/lib: https://lore.kernel.org/lkml/20221130062935.2219247-1-irogers@google.com/ Signed-off-by: Ian Rogers <irogers@google.com> Cc: Adrian Hunter <adrian.hunter@intel.com> Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com> Cc: Athira Jajeev <atrajeev@linux.vnet.ibm.com> Cc: Eelco Chaudron <echaudro@redhat.com> Cc: German Gomez <german.gomez@arm.com> Cc: Ingo Molnar <mingo@redhat.com> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Kan Liang <kan.liang@linux.intel.com> Cc: Kim Phillips <kim.phillips@amd.com> Cc: Leo Yan <leo.yan@linaro.org> Cc: Mark Rutland <mark.rutland@arm.com> Cc: Namhyung Kim <namhyung@kernel.org> Cc: Paolo Bonzini <pbonzini@redhat.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Sean Christopherson <seanjc@google.com> Cc: Stephane Eranian <eranian@google.com> Cc: Yang Jihong <yangjihong1@huawei.com> Link: https://lore.kernel.org/r/20230111070641.1728726-3-irogers@google.com Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
314 lines
6.3 KiB
C
314 lines
6.3 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* For general debugging purposes */
|
|
|
|
#include <inttypes.h>
|
|
#include <string.h>
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/wait.h>
|
|
#include <api/debug.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/time64.h>
|
|
#include <sys/time.h>
|
|
#ifdef HAVE_BACKTRACE_SUPPORT
|
|
#include <execinfo.h>
|
|
#endif
|
|
#include "color.h"
|
|
#include "event.h"
|
|
#include "debug.h"
|
|
#include "print_binary.h"
|
|
#include "target.h"
|
|
#include "trace-event.h"
|
|
#include "ui/helpline.h"
|
|
#include "ui/ui.h"
|
|
#include "util/parse-sublevel-options.h"
|
|
|
|
#include <linux/ctype.h>
|
|
|
|
#ifdef HAVE_LIBTRACEEVENT
|
|
#include <traceevent/event-parse.h>
|
|
#else
|
|
#define LIBTRACEEVENT_VERSION 0
|
|
#endif
|
|
|
|
int verbose;
|
|
int debug_peo_args;
|
|
bool dump_trace = false, quiet = false;
|
|
int debug_ordered_events;
|
|
static int redirect_to_stderr;
|
|
int debug_data_convert;
|
|
static FILE *debug_file;
|
|
bool debug_display_time;
|
|
|
|
void debug_set_file(FILE *file)
|
|
{
|
|
debug_file = file;
|
|
}
|
|
|
|
void debug_set_display_time(bool set)
|
|
{
|
|
debug_display_time = set;
|
|
}
|
|
|
|
static int fprintf_time(FILE *file)
|
|
{
|
|
struct timeval tod;
|
|
struct tm ltime;
|
|
char date[64];
|
|
|
|
if (!debug_display_time)
|
|
return 0;
|
|
|
|
if (gettimeofday(&tod, NULL) != 0)
|
|
return 0;
|
|
|
|
if (localtime_r(&tod.tv_sec, <ime) == NULL)
|
|
return 0;
|
|
|
|
strftime(date, sizeof(date), "%F %H:%M:%S", <ime);
|
|
return fprintf(file, "[%s.%06lu] ", date, (long)tod.tv_usec);
|
|
}
|
|
|
|
int veprintf(int level, int var, const char *fmt, va_list args)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (var >= level) {
|
|
if (use_browser >= 1 && !redirect_to_stderr) {
|
|
ui_helpline__vshow(fmt, args);
|
|
} else {
|
|
ret = fprintf_time(debug_file);
|
|
ret += vfprintf(debug_file, fmt, args);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int eprintf(int level, int var, const char *fmt, ...)
|
|
{
|
|
va_list args;
|
|
int ret;
|
|
|
|
va_start(args, fmt);
|
|
ret = veprintf(level, var, fmt, args);
|
|
va_end(args);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int veprintf_time(u64 t, const char *fmt, va_list args)
|
|
{
|
|
int ret = 0;
|
|
u64 secs, usecs, nsecs = t;
|
|
|
|
secs = nsecs / NSEC_PER_SEC;
|
|
nsecs -= secs * NSEC_PER_SEC;
|
|
usecs = nsecs / NSEC_PER_USEC;
|
|
|
|
ret = fprintf(stderr, "[%13" PRIu64 ".%06" PRIu64 "] ",
|
|
secs, usecs);
|
|
ret += vfprintf(stderr, fmt, args);
|
|
return ret;
|
|
}
|
|
|
|
int eprintf_time(int level, int var, u64 t, const char *fmt, ...)
|
|
{
|
|
int ret = 0;
|
|
va_list args;
|
|
|
|
if (var >= level) {
|
|
va_start(args, fmt);
|
|
ret = veprintf_time(t, fmt, args);
|
|
va_end(args);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Overloading libtraceevent standard info print
|
|
* function, display with -v in perf.
|
|
*/
|
|
void pr_stat(const char *fmt, ...)
|
|
{
|
|
va_list args;
|
|
|
|
va_start(args, fmt);
|
|
veprintf(1, verbose, fmt, args);
|
|
va_end(args);
|
|
eprintf(1, verbose, "\n");
|
|
}
|
|
|
|
int dump_printf(const char *fmt, ...)
|
|
{
|
|
va_list args;
|
|
int ret = 0;
|
|
|
|
if (dump_trace) {
|
|
va_start(args, fmt);
|
|
ret = vprintf(fmt, args);
|
|
va_end(args);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int trace_event_printer(enum binary_printer_ops op,
|
|
unsigned int val, void *extra, FILE *fp)
|
|
{
|
|
const char *color = PERF_COLOR_BLUE;
|
|
union perf_event *event = (union perf_event *)extra;
|
|
unsigned char ch = (unsigned char)val;
|
|
int printed = 0;
|
|
|
|
switch (op) {
|
|
case BINARY_PRINT_DATA_BEGIN:
|
|
printed += fprintf(fp, ".");
|
|
printed += color_fprintf(fp, color, "\n. ... raw event: size %d bytes\n",
|
|
event->header.size);
|
|
break;
|
|
case BINARY_PRINT_LINE_BEGIN:
|
|
printed += fprintf(fp, ".");
|
|
break;
|
|
case BINARY_PRINT_ADDR:
|
|
printed += color_fprintf(fp, color, " %04x: ", val);
|
|
break;
|
|
case BINARY_PRINT_NUM_DATA:
|
|
printed += color_fprintf(fp, color, " %02x", val);
|
|
break;
|
|
case BINARY_PRINT_NUM_PAD:
|
|
printed += color_fprintf(fp, color, " ");
|
|
break;
|
|
case BINARY_PRINT_SEP:
|
|
printed += color_fprintf(fp, color, " ");
|
|
break;
|
|
case BINARY_PRINT_CHAR_DATA:
|
|
printed += color_fprintf(fp, color, "%c",
|
|
isprint(ch) && isascii(ch) ? ch : '.');
|
|
break;
|
|
case BINARY_PRINT_CHAR_PAD:
|
|
printed += color_fprintf(fp, color, " ");
|
|
break;
|
|
case BINARY_PRINT_LINE_END:
|
|
printed += color_fprintf(fp, color, "\n");
|
|
break;
|
|
case BINARY_PRINT_DATA_END:
|
|
printed += fprintf(fp, "\n");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return printed;
|
|
}
|
|
|
|
void trace_event(union perf_event *event)
|
|
{
|
|
unsigned char *raw_event = (void *)event;
|
|
|
|
if (!dump_trace)
|
|
return;
|
|
|
|
print_binary(raw_event, event->header.size, 16,
|
|
trace_event_printer, event);
|
|
}
|
|
|
|
static struct sublevel_option debug_opts[] = {
|
|
{ .name = "verbose", .value_ptr = &verbose },
|
|
{ .name = "ordered-events", .value_ptr = &debug_ordered_events},
|
|
{ .name = "stderr", .value_ptr = &redirect_to_stderr},
|
|
{ .name = "data-convert", .value_ptr = &debug_data_convert },
|
|
{ .name = "perf-event-open", .value_ptr = &debug_peo_args },
|
|
{ .name = NULL, }
|
|
};
|
|
|
|
int perf_debug_option(const char *str)
|
|
{
|
|
int ret;
|
|
|
|
ret = perf_parse_sublevel_options(str, debug_opts);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Allow only verbose value in range (0, 10), otherwise set 0. */
|
|
verbose = (verbose < 0) || (verbose > 10) ? 0 : verbose;
|
|
|
|
#if LIBTRACEEVENT_VERSION >= MAKE_LIBTRACEEVENT_VERSION(1, 3, 0)
|
|
if (verbose == 1)
|
|
tep_set_loglevel(TEP_LOG_INFO);
|
|
else if (verbose == 2)
|
|
tep_set_loglevel(TEP_LOG_DEBUG);
|
|
else if (verbose >= 3)
|
|
tep_set_loglevel(TEP_LOG_ALL);
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
int perf_quiet_option(void)
|
|
{
|
|
struct sublevel_option *opt = &debug_opts[0];
|
|
|
|
/* disable all debug messages */
|
|
while (opt->name) {
|
|
*opt->value_ptr = -1;
|
|
opt++;
|
|
}
|
|
|
|
/* For debug variables that are used as bool types, set to 0. */
|
|
redirect_to_stderr = 0;
|
|
debug_peo_args = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define DEBUG_WRAPPER(__n, __l) \
|
|
static int pr_ ## __n ## _wrapper(const char *fmt, ...) \
|
|
{ \
|
|
va_list args; \
|
|
int ret; \
|
|
\
|
|
va_start(args, fmt); \
|
|
ret = veprintf(__l, verbose, fmt, args); \
|
|
va_end(args); \
|
|
return ret; \
|
|
}
|
|
|
|
DEBUG_WRAPPER(warning, 0);
|
|
DEBUG_WRAPPER(debug, 1);
|
|
|
|
void perf_debug_setup(void)
|
|
{
|
|
debug_set_file(stderr);
|
|
libapi_set_print(pr_warning_wrapper, pr_warning_wrapper, pr_debug_wrapper);
|
|
}
|
|
|
|
/* Obtain a backtrace and print it to stdout. */
|
|
#ifdef HAVE_BACKTRACE_SUPPORT
|
|
void dump_stack(void)
|
|
{
|
|
void *array[16];
|
|
size_t size = backtrace(array, ARRAY_SIZE(array));
|
|
char **strings = backtrace_symbols(array, size);
|
|
size_t i;
|
|
|
|
printf("Obtained %zd stack frames.\n", size);
|
|
|
|
for (i = 0; i < size; i++)
|
|
printf("%s\n", strings[i]);
|
|
|
|
free(strings);
|
|
}
|
|
#else
|
|
void dump_stack(void) {}
|
|
#endif
|
|
|
|
void sighandler_dump_stack(int sig)
|
|
{
|
|
psignal(sig, "perf");
|
|
dump_stack();
|
|
signal(sig, SIG_DFL);
|
|
raise(sig);
|
|
}
|