Commit d392e49a authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'trace-tools-v6.3' of git://git.kernel.org/pub/scm/linux/kernel/git/trace/linux-trace

Pull tracing tools updates from Steven Rostedt:

 - Use total duration to calculate average in rtla osnoise_hist

 - Use 2 digit precision for displaying average

 - Print an intuitive auto analysis of timerlat results

 - Add auto analysis to timerlat top

 - Add hwnoise, which is the same as osnoise but focuses on hardware

 - Small clean ups

* tag 'trace-tools-v6.3' of git://git.kernel.org/pub/scm/linux/kernel/git/trace/linux-trace:
  Documentation/rtla: Add hwnoise man page
  rtla: Add hwnoise tool
  Documentation/rtla: Add timerlat-top auto-analysis options
  rtla/timerlat: Add auto-analysis support to timerlat top
  rtla/timerlat: Add auto-analysis core
  tools/tracing/rtla: osnoise_hist: display average with two-digit precision
  tools/tracing/rtla: osnoise_hist: use total duration for average calculation
  tools/rv: Remove unneeded semicolon
parents 2562af68 5dc3750e
**--dump-tasks**
prints the task running on all CPUs if stop conditions are met (depends on !--no-aa)
**--no-aa**
disable auto-analysis, reducing rtla timerlat cpu usage
...@@ -17,6 +17,7 @@ behavior on specific hardware. ...@@ -17,6 +17,7 @@ behavior on specific hardware.
rtla-timerlat rtla-timerlat
rtla-timerlat-hist rtla-timerlat-hist
rtla-timerlat-top rtla-timerlat-top
rtla-hwnoise
.. only:: subproject and html .. only:: subproject and html
......
.. SPDX-License-Identifier: GPL-2.0
============
rtla-hwnoise
============
------------------------------------------
Detect and quantify hardware-related noise
------------------------------------------
:Manual section: 1
SYNOPSIS
========
**rtla hwnoise** [*OPTIONS*]
DESCRIPTION
===========
**rtla hwnoise** collects the periodic summary from the *osnoise* tracer
running with *interrupts disabled*. By disabling interrupts, and the scheduling
of threads as a consequence, only non-maskable interrupts and hardware-related
noise is allowed.
The tool also allows the configurations of the *osnoise* tracer and the
collection of the tracer output.
OPTIONS
=======
.. include:: common_osnoise_options.rst
.. include:: common_top_options.rst
.. include:: common_options.rst
EXAMPLE
=======
In the example below, the **rtla hwnoise** tool is set to run on CPUs *1-7*
on a system with 8 cores/16 threads with hyper-threading enabled.
The tool is set to detect any noise higher than *one microsecond*,
to run for *ten minutes*, displaying a summary of the report at the
end of the session::
# rtla hwnoise -c 1-7 -T 1 -d 10m -q
Hardware-related Noise
duration: 0 00:10:00 | time is in us
CPU Period Runtime Noise % CPU Aval Max Noise Max Single HW NMI
1 #599 599000000 138 99.99997 3 3 4 74
2 #599 599000000 85 99.99998 3 3 4 75
3 #599 599000000 86 99.99998 4 3 6 75
4 #599 599000000 81 99.99998 4 4 2 75
5 #599 599000000 85 99.99998 2 2 2 75
6 #599 599000000 76 99.99998 2 2 0 75
7 #599 599000000 77 99.99998 3 3 0 75
The first column shows the *CPU*, and the second column shows how many
*Periods* the tool ran during the session. The *Runtime* is the time
the tool effectively runs on the CPU. The *Noise* column is the sum of
all noise that the tool observed, and the *% CPU Aval* is the relation
between the *Runtime* and *Noise*.
The *Max Noise* column is the maximum hardware noise the tool detected in a
single period, and the *Max Single* is the maximum single noise seen.
The *HW* and *NMI* columns show the total number of *hardware* and *NMI* noise
occurrence observed by the tool.
For example, *CPU 3* ran *599* periods of *1 second Runtime*. The CPU received
*86 us* of noise during the entire execution, leaving *99.99997 %* of CPU time
for the application. In the worst single period, the CPU caused *4 us* of
noise to the application, but it was certainly caused by more than one single
noise, as the *Max Single* noise was of *3 us*. The CPU has *HW noise,* at a
rate of *six occurrences*/*ten minutes*. The CPU also has *NMIs*, at a higher
frequency: around *seven per second*.
The tool should report *0* hardware-related noise in the ideal situation.
For example, by disabling hyper-threading to remove the hardware noise,
and disabling the TSC watchdog to remove the NMI (it is possible to identify
this using tracing options of **rtla hwnoise**), it was possible to reach
the ideal situation in the same hardware::
# rtla hwnoise -c 1-7 -T 1 -d 10m -q
Hardware-related Noise
duration: 0 00:10:00 | time is in us
CPU Period Runtime Noise % CPU Aval Max Noise Max Single HW NMI
1 #599 599000000 0 100.00000 0 0 0 0
2 #599 599000000 0 100.00000 0 0 0 0
3 #599 599000000 0 100.00000 0 0 0 0
4 #599 599000000 0 100.00000 0 0 0 0
5 #599 599000000 0 100.00000 0 0 0 0
6 #599 599000000 0 100.00000 0 0 0 0
7 #599 599000000 0 100.00000 0 0 0 0
SEE ALSO
========
**rtla-osnoise**\(1)
Osnoise tracer documentation: <https://www.kernel.org/doc/html/latest/trace/osnoise-tracer.html>
AUTHOR
======
Written by Daniel Bristot de Oliveira <bristot@kernel.org>
.. include:: common_appendix.rst
...@@ -119,6 +119,8 @@ install: doc_install ...@@ -119,6 +119,8 @@ install: doc_install
$(STRIP) $(DESTDIR)$(BINDIR)/rtla $(STRIP) $(DESTDIR)$(BINDIR)/rtla
@test ! -f $(DESTDIR)$(BINDIR)/osnoise || rm $(DESTDIR)$(BINDIR)/osnoise @test ! -f $(DESTDIR)$(BINDIR)/osnoise || rm $(DESTDIR)$(BINDIR)/osnoise
ln -s rtla $(DESTDIR)$(BINDIR)/osnoise ln -s rtla $(DESTDIR)$(BINDIR)/osnoise
@test ! -f $(DESTDIR)$(BINDIR)/hwnoise || rm $(DESTDIR)$(BINDIR)/hwnoise
ln -s rtla $(DESTDIR)$(BINDIR)/hwnoise
@test ! -f $(DESTDIR)$(BINDIR)/timerlat || rm $(DESTDIR)$(BINDIR)/timerlat @test ! -f $(DESTDIR)$(BINDIR)/timerlat || rm $(DESTDIR)$(BINDIR)/timerlat
ln -s rtla $(DESTDIR)$(BINDIR)/timerlat ln -s rtla $(DESTDIR)$(BINDIR)/timerlat
......
...@@ -734,6 +734,113 @@ void osnoise_put_tracing_thresh(struct osnoise_context *context) ...@@ -734,6 +734,113 @@ void osnoise_put_tracing_thresh(struct osnoise_context *context)
context->orig_tracing_thresh = OSNOISE_OPTION_INIT_VAL; context->orig_tracing_thresh = OSNOISE_OPTION_INIT_VAL;
} }
static int osnoise_options_get_option(char *option)
{
char *options = tracefs_instance_file_read(NULL, "osnoise/options", NULL);
char no_option[128];
int retval = 0;
char *opt;
if (!options)
return OSNOISE_OPTION_INIT_VAL;
/*
* Check first if the option is disabled.
*/
snprintf(no_option, sizeof(no_option), "NO_%s", option);
opt = strstr(options, no_option);
if (opt)
goto out_free;
/*
* Now that it is not disabled, if the string is there, it is
* enabled. If the string is not there, the option does not exist.
*/
opt = strstr(options, option);
if (opt)
retval = 1;
else
retval = OSNOISE_OPTION_INIT_VAL;
out_free:
free(options);
return retval;
}
static int osnoise_options_set_option(char *option, bool onoff)
{
char no_option[128];
if (onoff)
return tracefs_instance_file_write(NULL, "osnoise/options", option);
snprintf(no_option, sizeof(no_option), "NO_%s", option);
return tracefs_instance_file_write(NULL, "osnoise/options", no_option);
}
static int osnoise_get_irq_disable(struct osnoise_context *context)
{
if (context->opt_irq_disable != OSNOISE_OPTION_INIT_VAL)
return context->opt_irq_disable;
if (context->orig_opt_irq_disable != OSNOISE_OPTION_INIT_VAL)
return context->orig_opt_irq_disable;
context->orig_opt_irq_disable = osnoise_options_get_option("OSNOISE_IRQ_DISABLE");
return context->orig_opt_irq_disable;
}
int osnoise_set_irq_disable(struct osnoise_context *context, bool onoff)
{
int opt_irq_disable = osnoise_get_irq_disable(context);
int retval;
if (opt_irq_disable == OSNOISE_OPTION_INIT_VAL)
return -1;
if (opt_irq_disable == onoff)
return 0;
retval = osnoise_options_set_option("OSNOISE_IRQ_DISABLE", onoff);
if (retval < 0)
return -1;
context->opt_irq_disable = onoff;
return 0;
}
static void osnoise_restore_irq_disable(struct osnoise_context *context)
{
int retval;
if (context->orig_opt_irq_disable == OSNOISE_OPTION_INIT_VAL)
return;
if (context->orig_opt_irq_disable == context->opt_irq_disable)
goto out_done;
retval = osnoise_options_set_option("OSNOISE_IRQ_DISABLE", context->orig_opt_irq_disable);
if (retval < 0)
err_msg("Could not restore original OSNOISE_IRQ_DISABLE option\n");
out_done:
context->orig_opt_irq_disable = OSNOISE_OPTION_INIT_VAL;
}
static void osnoise_put_irq_disable(struct osnoise_context *context)
{
osnoise_restore_irq_disable(context);
if (context->orig_opt_irq_disable == OSNOISE_OPTION_INIT_VAL)
return;
context->orig_opt_irq_disable = OSNOISE_OPTION_INIT_VAL;
}
/* /*
* enable_osnoise - enable osnoise tracer in the trace_instance * enable_osnoise - enable osnoise tracer in the trace_instance
*/ */
...@@ -798,6 +905,9 @@ struct osnoise_context *osnoise_context_alloc(void) ...@@ -798,6 +905,9 @@ struct osnoise_context *osnoise_context_alloc(void)
context->orig_tracing_thresh = OSNOISE_OPTION_INIT_VAL; context->orig_tracing_thresh = OSNOISE_OPTION_INIT_VAL;
context->tracing_thresh = OSNOISE_OPTION_INIT_VAL; context->tracing_thresh = OSNOISE_OPTION_INIT_VAL;
context->orig_opt_irq_disable = OSNOISE_OPTION_INIT_VAL;
context->opt_irq_disable = OSNOISE_OPTION_INIT_VAL;
osnoise_get_context(context); osnoise_get_context(context);
return context; return context;
...@@ -824,6 +934,7 @@ void osnoise_put_context(struct osnoise_context *context) ...@@ -824,6 +934,7 @@ void osnoise_put_context(struct osnoise_context *context)
osnoise_put_timerlat_period_us(context); osnoise_put_timerlat_period_us(context);
osnoise_put_print_stack(context); osnoise_put_print_stack(context);
osnoise_put_tracing_thresh(context); osnoise_put_tracing_thresh(context);
osnoise_put_irq_disable(context);
free(context); free(context);
} }
...@@ -958,3 +1069,9 @@ int osnoise_main(int argc, char *argv[]) ...@@ -958,3 +1069,9 @@ int osnoise_main(int argc, char *argv[])
osnoise_usage(1); osnoise_usage(1);
exit(1); exit(1);
} }
int hwnoise_main(int argc, char *argv[])
{
osnoise_top_main(argc, argv);
exit(0);
}
...@@ -38,6 +38,10 @@ struct osnoise_context { ...@@ -38,6 +38,10 @@ struct osnoise_context {
/* -1 as init value because 0 is disabled */ /* -1 as init value because 0 is disabled */
long long orig_print_stack; long long orig_print_stack;
long long print_stack; long long print_stack;
/* -1 as init value because 0 is off */
int orig_opt_irq_disable;
int opt_irq_disable;
}; };
/* /*
...@@ -79,6 +83,8 @@ void osnoise_restore_print_stack(struct osnoise_context *context); ...@@ -79,6 +83,8 @@ void osnoise_restore_print_stack(struct osnoise_context *context);
int osnoise_set_print_stack(struct osnoise_context *context, int osnoise_set_print_stack(struct osnoise_context *context,
long long print_stack); long long print_stack);
int osnoise_set_irq_disable(struct osnoise_context *context, bool onoff);
/* /*
* osnoise_tool - osnoise based tool definition. * osnoise_tool - osnoise based tool definition.
*/ */
...@@ -97,3 +103,4 @@ struct osnoise_tool *osnoise_init_trace_tool(char *tracer); ...@@ -97,3 +103,4 @@ struct osnoise_tool *osnoise_init_trace_tool(char *tracer);
int osnoise_hist_main(int argc, char *argv[]); int osnoise_hist_main(int argc, char *argv[]);
int osnoise_top_main(int argc, char **argv); int osnoise_top_main(int argc, char **argv);
int osnoise_main(int argc, char **argv); int osnoise_main(int argc, char **argv);
int hwnoise_main(int argc, char **argv);
...@@ -121,6 +121,7 @@ static void osnoise_hist_update_multiple(struct osnoise_tool *tool, int cpu, ...@@ -121,6 +121,7 @@ static void osnoise_hist_update_multiple(struct osnoise_tool *tool, int cpu,
{ {
struct osnoise_hist_params *params = tool->params; struct osnoise_hist_params *params = tool->params;
struct osnoise_hist_data *data = tool->data; struct osnoise_hist_data *data = tool->data;
unsigned long long total_duration;
int entries = data->entries; int entries = data->entries;
int bucket; int bucket;
int *hist; int *hist;
...@@ -131,10 +132,12 @@ static void osnoise_hist_update_multiple(struct osnoise_tool *tool, int cpu, ...@@ -131,10 +132,12 @@ static void osnoise_hist_update_multiple(struct osnoise_tool *tool, int cpu,
if (data->bucket_size) if (data->bucket_size)
bucket = duration / data->bucket_size; bucket = duration / data->bucket_size;
total_duration = duration * count;
hist = data->hist[cpu].samples; hist = data->hist[cpu].samples;
data->hist[cpu].count += count; data->hist[cpu].count += count;
update_min(&data->hist[cpu].min_sample, &duration); update_min(&data->hist[cpu].min_sample, &duration);
update_sum(&data->hist[cpu].sum_sample, &duration); update_sum(&data->hist[cpu].sum_sample, &total_duration);
update_max(&data->hist[cpu].max_sample, &duration); update_max(&data->hist[cpu].max_sample, &duration);
if (bucket < entries) if (bucket < entries)
...@@ -332,8 +335,8 @@ osnoise_print_summary(struct osnoise_hist_params *params, ...@@ -332,8 +335,8 @@ osnoise_print_summary(struct osnoise_hist_params *params,
continue; continue;
if (data->hist[cpu].count) if (data->hist[cpu].count)
trace_seq_printf(trace->seq, "%9llu ", trace_seq_printf(trace->seq, "%9.2f ",
data->hist[cpu].sum_sample / data->hist[cpu].count); ((double) data->hist[cpu].sum_sample) / data->hist[cpu].count);
else else
trace_seq_printf(trace->seq, " - "); trace_seq_printf(trace->seq, " - ");
} }
......
...@@ -14,6 +14,11 @@ ...@@ -14,6 +14,11 @@
#include "osnoise.h" #include "osnoise.h"
#include "utils.h" #include "utils.h"
enum osnoise_mode {
MODE_OSNOISE = 0,
MODE_HWNOISE
};
/* /*
* osnoise top parameters * osnoise top parameters
*/ */
...@@ -32,6 +37,7 @@ struct osnoise_top_params { ...@@ -32,6 +37,7 @@ struct osnoise_top_params {
int set_sched; int set_sched;
struct sched_attr sched_param; struct sched_attr sched_param;
struct trace_events *events; struct trace_events *events;
enum osnoise_mode mode;
}; };
struct osnoise_top_cpu { struct osnoise_top_cpu {
...@@ -143,15 +149,23 @@ osnoise_top_handler(struct trace_seq *s, struct tep_record *record, ...@@ -143,15 +149,23 @@ osnoise_top_handler(struct trace_seq *s, struct tep_record *record,
*/ */
static void osnoise_top_header(struct osnoise_tool *top) static void osnoise_top_header(struct osnoise_tool *top)
{ {
struct osnoise_top_params *params = top->params;
struct trace_seq *s = top->trace.seq; struct trace_seq *s = top->trace.seq;
char duration[26]; char duration[26];
get_duration(top->start_time, duration, sizeof(duration)); get_duration(top->start_time, duration, sizeof(duration));
trace_seq_printf(s, "\033[2;37;40m"); trace_seq_printf(s, "\033[2;37;40m");
trace_seq_printf(s, " Operating System Noise"); trace_seq_printf(s, " ");
trace_seq_printf(s, " ");
trace_seq_printf(s, " "); if (params->mode == MODE_OSNOISE) {
trace_seq_printf(s, "Operating System Noise");
trace_seq_printf(s, " ");
} else if (params->mode == MODE_HWNOISE) {
trace_seq_printf(s, "Hardware-related Noise");
}
trace_seq_printf(s, " ");
trace_seq_printf(s, "\033[0;0;0m"); trace_seq_printf(s, "\033[0;0;0m");
trace_seq_printf(s, "\n"); trace_seq_printf(s, "\n");
...@@ -162,7 +176,14 @@ static void osnoise_top_header(struct osnoise_tool *top) ...@@ -162,7 +176,14 @@ static void osnoise_top_header(struct osnoise_tool *top)
trace_seq_printf(s, " Noise "); trace_seq_printf(s, " Noise ");
trace_seq_printf(s, " %% CPU Aval "); trace_seq_printf(s, " %% CPU Aval ");
trace_seq_printf(s, " Max Noise Max Single "); trace_seq_printf(s, " Max Noise Max Single ");
trace_seq_printf(s, " HW NMI IRQ Softirq Thread"); trace_seq_printf(s, " HW NMI");
if (params->mode == MODE_HWNOISE)
goto eol;
trace_seq_printf(s, " IRQ Softirq Thread");
eol:
trace_seq_printf(s, "\033[0;0;0m"); trace_seq_printf(s, "\033[0;0;0m");
trace_seq_printf(s, "\n"); trace_seq_printf(s, "\n");
} }
...@@ -181,6 +202,7 @@ static void clear_terminal(struct trace_seq *seq) ...@@ -181,6 +202,7 @@ static void clear_terminal(struct trace_seq *seq)
*/ */
static void osnoise_top_print(struct osnoise_tool *tool, int cpu) static void osnoise_top_print(struct osnoise_tool *tool, int cpu)
{ {
struct osnoise_top_params *params = tool->params;
struct trace_seq *s = tool->trace.seq; struct trace_seq *s = tool->trace.seq;
struct osnoise_top_cpu *cpu_data; struct osnoise_top_cpu *cpu_data;
struct osnoise_top_data *data; struct osnoise_top_data *data;
...@@ -205,6 +227,12 @@ static void osnoise_top_print(struct osnoise_tool *tool, int cpu) ...@@ -205,6 +227,12 @@ static void osnoise_top_print(struct osnoise_tool *tool, int cpu)
trace_seq_printf(s, "%12llu ", cpu_data->hw_count); trace_seq_printf(s, "%12llu ", cpu_data->hw_count);
trace_seq_printf(s, "%12llu ", cpu_data->nmi_count); trace_seq_printf(s, "%12llu ", cpu_data->nmi_count);
if (params->mode == MODE_HWNOISE) {
trace_seq_printf(s, "\n");
return;
}
trace_seq_printf(s, "%12llu ", cpu_data->irq_count); trace_seq_printf(s, "%12llu ", cpu_data->irq_count);
trace_seq_printf(s, "%12llu ", cpu_data->softirq_count); trace_seq_printf(s, "%12llu ", cpu_data->softirq_count);
trace_seq_printf(s, "%12llu\n", cpu_data->thread_count); trace_seq_printf(s, "%12llu\n", cpu_data->thread_count);
...@@ -241,12 +269,12 @@ osnoise_print_stats(struct osnoise_top_params *params, struct osnoise_tool *top) ...@@ -241,12 +269,12 @@ osnoise_print_stats(struct osnoise_top_params *params, struct osnoise_tool *top)
/* /*
* osnoise_top_usage - prints osnoise top usage message * osnoise_top_usage - prints osnoise top usage message
*/ */
void osnoise_top_usage(char *usage) static void osnoise_top_usage(struct osnoise_top_params *params, char *usage)
{ {
int i; int i;
static const char * const msg[] = { static const char * const msg[] = {
" usage: rtla osnoise [top] [-h] [-q] [-D] [-d s] [-a us] [-p us] [-r us] [-s us] [-S us] \\", " [-h] [-q] [-D] [-d s] [-a us] [-p us] [-r us] [-s us] [-S us] \\",
" [-T us] [-t[=file]] [-e sys[:event]] [--filter <filter>] [--trigger <trigger>] \\", " [-T us] [-t[=file]] [-e sys[:event]] [--filter <filter>] [--trigger <trigger>] \\",
" [-c cpu-list] [-P priority]", " [-c cpu-list] [-P priority]",
"", "",
...@@ -277,9 +305,22 @@ void osnoise_top_usage(char *usage) ...@@ -277,9 +305,22 @@ void osnoise_top_usage(char *usage)
if (usage) if (usage)
fprintf(stderr, "%s\n", usage); fprintf(stderr, "%s\n", usage);
fprintf(stderr, "rtla osnoise top: a per-cpu summary of the OS noise (version %s)\n", if (params->mode == MODE_OSNOISE) {
fprintf(stderr,
"rtla osnoise top: a per-cpu summary of the OS noise (version %s)\n",
VERSION); VERSION);
fprintf(stderr, " usage: rtla osnoise [top]");
}
if (params->mode == MODE_HWNOISE) {
fprintf(stderr,
"rtla hwnoise: a summary of hardware-related noise (version %s)\n",
VERSION);
fprintf(stderr, " usage: rtla hwnoise");
}
for (i = 0; msg[i]; i++) for (i = 0; msg[i]; i++)
fprintf(stderr, "%s\n", msg[i]); fprintf(stderr, "%s\n", msg[i]);
exit(1); exit(1);
...@@ -299,6 +340,9 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -299,6 +340,9 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
if (!params) if (!params)
exit(1); exit(1);
if (strcmp(argv[0], "hwnoise") == 0)
params->mode = MODE_HWNOISE;
while (1) { while (1) {
static struct option long_options[] = { static struct option long_options[] = {
{"auto", required_argument, 0, 'a'}, {"auto", required_argument, 0, 'a'},
...@@ -345,7 +389,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -345,7 +389,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
case 'c': case 'c':
retval = parse_cpu_list(optarg, &params->monitored_cpus); retval = parse_cpu_list(optarg, &params->monitored_cpus);
if (retval) if (retval)
osnoise_top_usage("\nInvalid -c cpu list\n"); osnoise_top_usage(params, "\nInvalid -c cpu list\n");
params->cpus = optarg; params->cpus = optarg;
break; break;
case 'D': case 'D':
...@@ -354,7 +398,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -354,7 +398,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
case 'd': case 'd':
params->duration = parse_seconds_duration(optarg); params->duration = parse_seconds_duration(optarg);
if (!params->duration) if (!params->duration)
osnoise_top_usage("Invalid -D duration\n"); osnoise_top_usage(params, "Invalid -D duration\n");
break; break;
case 'e': case 'e':
tevent = trace_event_alloc(optarg); tevent = trace_event_alloc(optarg);
...@@ -370,17 +414,17 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -370,17 +414,17 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
break; break;
case 'h': case 'h':
case '?': case '?':
osnoise_top_usage(NULL); osnoise_top_usage(params, NULL);
break; break;
case 'p': case 'p':
params->period = get_llong_from_str(optarg); params->period = get_llong_from_str(optarg);
if (params->period > 10000000) if (params->period > 10000000)
osnoise_top_usage("Period longer than 10 s\n"); osnoise_top_usage(params, "Period longer than 10 s\n");
break; break;
case 'P': case 'P':
retval = parse_prio(optarg, &params->sched_param); retval = parse_prio(optarg, &params->sched_param);
if (retval == -1) if (retval == -1)
osnoise_top_usage("Invalid -P priority"); osnoise_top_usage(params, "Invalid -P priority");
params->set_sched = 1; params->set_sched = 1;
break; break;
case 'q': case 'q':
...@@ -389,7 +433,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -389,7 +433,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
case 'r': case 'r':
params->runtime = get_llong_from_str(optarg); params->runtime = get_llong_from_str(optarg);
if (params->runtime < 100) if (params->runtime < 100)
osnoise_top_usage("Runtime shorter than 100 us\n"); osnoise_top_usage(params, "Runtime shorter than 100 us\n");
break; break;
case 's': case 's':
params->stop_us = get_llong_from_str(optarg); params->stop_us = get_llong_from_str(optarg);
...@@ -415,7 +459,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -415,7 +459,7 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
} else { } else {
osnoise_top_usage("--trigger requires a previous -e\n"); osnoise_top_usage(params, "--trigger requires a previous -e\n");
} }
break; break;
case '1': /* filter */ case '1': /* filter */
...@@ -426,11 +470,11 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv) ...@@ -426,11 +470,11 @@ struct osnoise_top_params *osnoise_top_parse_args(int argc, char **argv)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
} else { } else {
osnoise_top_usage("--filter requires a previous -e\n"); osnoise_top_usage(params, "--filter requires a previous -e\n");
} }
break; break;
default: default:
osnoise_top_usage("Invalid option"); osnoise_top_usage(params, "Invalid option");
} }
} }
...@@ -495,6 +539,14 @@ osnoise_top_apply_config(struct osnoise_tool *tool, struct osnoise_top_params *p ...@@ -495,6 +539,14 @@ osnoise_top_apply_config(struct osnoise_tool *tool, struct osnoise_top_params *p
} }
} }
if (params->mode == MODE_HWNOISE) {
retval = osnoise_set_irq_disable(tool->context, 1);
if (retval) {
err_msg("Failed to set OSNOISE_IRQ_DISABLE option\n");
goto out_err;
}
}
return 0; return 0;
out_err: out_err:
......
...@@ -26,6 +26,7 @@ static void rtla_usage(int err) ...@@ -26,6 +26,7 @@ static void rtla_usage(int err)
"", "",
" commands:", " commands:",
" osnoise - gives information about the operating system noise (osnoise)", " osnoise - gives information about the operating system noise (osnoise)",
" hwnoise - gives information about hardware-related noise",
" timerlat - measures the timer irq and thread latency", " timerlat - measures the timer irq and thread latency",
"", "",
NULL, NULL,
...@@ -47,6 +48,9 @@ int run_command(int argc, char **argv, int start_position) ...@@ -47,6 +48,9 @@ int run_command(int argc, char **argv, int start_position)
if (strcmp(argv[start_position], "osnoise") == 0) { if (strcmp(argv[start_position], "osnoise") == 0) {
osnoise_main(argc-start_position, &argv[start_position]); osnoise_main(argc-start_position, &argv[start_position]);
goto ran; goto ran;
} else if (strcmp(argv[start_position], "hwnoise") == 0) {
hwnoise_main(argc-start_position, &argv[start_position]);
goto ran;
} else if (strcmp(argv[start_position], "timerlat") == 0) { } else if (strcmp(argv[start_position], "timerlat") == 0) {
timerlat_main(argc-start_position, &argv[start_position]); timerlat_main(argc-start_position, &argv[start_position]);
goto ran; goto ran;
......
This diff is collapsed.
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (C) 2023 Red Hat Inc, Daniel Bristot de Oliveira <bristot@kernel.org>
*/
int timerlat_aa_init(struct osnoise_tool *tool, int nr_cpus, int dump_task);
void timerlat_aa_destroy(void);
int timerlat_aa_handler(struct trace_seq *s, struct tep_record *record,
struct tep_event *event, void *context);
void timerlat_auto_analysis(int irq_thresh, int thread_thresh);
...@@ -10,10 +10,12 @@ ...@@ -10,10 +10,12 @@
#include <unistd.h> #include <unistd.h>
#include <stdio.h> #include <stdio.h>
#include <time.h> #include <time.h>
#include <errno.h>
#include "utils.h" #include "utils.h"
#include "osnoise.h" #include "osnoise.h"
#include "timerlat.h" #include "timerlat.h"
#include "timerlat_aa.h"
struct timerlat_top_params { struct timerlat_top_params {
char *cpus; char *cpus;
...@@ -30,6 +32,8 @@ struct timerlat_top_params { ...@@ -30,6 +32,8 @@ struct timerlat_top_params {
int quiet; int quiet;
int set_sched; int set_sched;
int dma_latency; int dma_latency;
int no_aa;
int dump_tasks;
struct sched_attr sched_param; struct sched_attr sched_param;
struct trace_events *events; struct trace_events *events;
}; };
...@@ -130,17 +134,22 @@ timerlat_top_handler(struct trace_seq *s, struct tep_record *record, ...@@ -130,17 +134,22 @@ timerlat_top_handler(struct trace_seq *s, struct tep_record *record,
struct tep_event *event, void *context) struct tep_event *event, void *context)
{ {
struct trace_instance *trace = context; struct trace_instance *trace = context;
struct timerlat_top_params *params;
unsigned long long latency, thread; unsigned long long latency, thread;
struct osnoise_tool *top; struct osnoise_tool *top;
int cpu = record->cpu; int cpu = record->cpu;
top = container_of(trace, struct osnoise_tool, trace); top = container_of(trace, struct osnoise_tool, trace);
params = top->params;
tep_get_field_val(s, event, "context", record, &thread, 1); tep_get_field_val(s, event, "context", record, &thread, 1);
tep_get_field_val(s, event, "timer_latency", record, &latency, 1); tep_get_field_val(s, event, "timer_latency", record, &latency, 1);
timerlat_top_update(top, cpu, thread, latency); timerlat_top_update(top, cpu, thread, latency);
if (!params->no_aa)
timerlat_aa_handler(s, record, event, context);
return 0; return 0;
} }
...@@ -281,11 +290,13 @@ static void timerlat_top_usage(char *usage) ...@@ -281,11 +290,13 @@ static void timerlat_top_usage(char *usage)
" -c/--cpus cpus: run the tracer only on the given cpus", " -c/--cpus cpus: run the tracer only on the given cpus",
" -d/--duration time[m|h|d]: duration of the session in seconds", " -d/--duration time[m|h|d]: duration of the session in seconds",
" -D/--debug: print debug info", " -D/--debug: print debug info",
" --dump-tasks: prints the task running on all CPUs if stop conditions are met (depends on !--no-aa)",
" -t/--trace[=file]: save the stopped trace to [file|timerlat_trace.txt]", " -t/--trace[=file]: save the stopped trace to [file|timerlat_trace.txt]",
" -e/--event <sys:event>: enable the <sys:event> in the trace instance, multiple -e are allowed", " -e/--event <sys:event>: enable the <sys:event> in the trace instance, multiple -e are allowed",
" --filter <command>: enable a trace event filter to the previous -e event", " --filter <command>: enable a trace event filter to the previous -e event",
" --trigger <command>: enable a trace event trigger to the previous -e event", " --trigger <command>: enable a trace event trigger to the previous -e event",
" -n/--nano: display data in nanoseconds", " -n/--nano: display data in nanoseconds",
" --no-aa: disable auto-analysis, reducing rtla timerlat cpu usage",
" -q/--quiet print only a summary at the end", " -q/--quiet print only a summary at the end",
" --dma-latency us: set /dev/cpu_dma_latency latency <us> to reduce exit from idle latency", " --dma-latency us: set /dev/cpu_dma_latency latency <us> to reduce exit from idle latency",
" -P/--priority o:prio|r:prio|f:prio|d:runtime:period : set scheduling parameters", " -P/--priority o:prio|r:prio|f:prio|d:runtime:period : set scheduling parameters",
...@@ -349,13 +360,15 @@ static struct timerlat_top_params ...@@ -349,13 +360,15 @@ static struct timerlat_top_params
{"trigger", required_argument, 0, '0'}, {"trigger", required_argument, 0, '0'},
{"filter", required_argument, 0, '1'}, {"filter", required_argument, 0, '1'},
{"dma-latency", required_argument, 0, '2'}, {"dma-latency", required_argument, 0, '2'},
{"no-aa", no_argument, 0, '3'},
{"dump-tasks", no_argument, 0, '4'},
{0, 0, 0, 0} {0, 0, 0, 0}
}; };
/* getopt_long stores the option index here. */ /* getopt_long stores the option index here. */
int option_index = 0; int option_index = 0;
c = getopt_long(argc, argv, "a:c:d:De:hi:np:P:qs:t::T:0:1:2:", c = getopt_long(argc, argv, "a:c:d:De:hi:np:P:qs:t::T:0:1:2:34",
long_options, &option_index); long_options, &option_index);
/* detect the end of the options. */ /* detect the end of the options. */
...@@ -368,13 +381,13 @@ static struct timerlat_top_params ...@@ -368,13 +381,13 @@ static struct timerlat_top_params
/* set thread stop to auto_thresh */ /* set thread stop to auto_thresh */
params->stop_total_us = auto_thresh; params->stop_total_us = auto_thresh;
params->stop_us = auto_thresh;
/* get stack trace */ /* get stack trace */
params->print_stack = auto_thresh; params->print_stack = auto_thresh;
/* set trace */ /* set trace */
params->trace_output = "timerlat_trace.txt"; params->trace_output = "timerlat_trace.txt";
break; break;
case 'c': case 'c':
retval = parse_cpu_list(optarg, &params->monitored_cpus); retval = parse_cpu_list(optarg, &params->monitored_cpus);
...@@ -437,6 +450,7 @@ static struct timerlat_top_params ...@@ -437,6 +450,7 @@ static struct timerlat_top_params
params->trace_output = &optarg[1]; params->trace_output = &optarg[1];
else else
params->trace_output = "timerlat_trace.txt"; params->trace_output = "timerlat_trace.txt";
break; break;
case '0': /* trigger */ case '0': /* trigger */
if (params->events) { if (params->events) {
...@@ -467,6 +481,12 @@ static struct timerlat_top_params ...@@ -467,6 +481,12 @@ static struct timerlat_top_params
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
break; break;
case '3': /* no-aa */
params->no_aa = 1;
break;
case '4':
params->dump_tasks = 1;
break;
default: default:
timerlat_top_usage("Invalid option"); timerlat_top_usage("Invalid option");
} }
...@@ -477,6 +497,12 @@ static struct timerlat_top_params ...@@ -477,6 +497,12 @@ static struct timerlat_top_params
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
/*
* Auto analysis only happens if stop tracing, thus:
*/
if (!params->stop_us && !params->stop_total_us)
params->no_aa = 1;
return params; return params;
} }
...@@ -547,6 +573,7 @@ static struct osnoise_tool ...@@ -547,6 +573,7 @@ static struct osnoise_tool
{ {
struct osnoise_tool *top; struct osnoise_tool *top;
int nr_cpus; int nr_cpus;
int retval;
nr_cpus = sysconf(_SC_NPROCESSORS_CONF); nr_cpus = sysconf(_SC_NPROCESSORS_CONF);
...@@ -563,6 +590,16 @@ static struct osnoise_tool ...@@ -563,6 +590,16 @@ static struct osnoise_tool
tep_register_event_handler(top->trace.tep, -1, "ftrace", "timerlat", tep_register_event_handler(top->trace.tep, -1, "ftrace", "timerlat",
timerlat_top_handler, top); timerlat_top_handler, top);
/*
* If no auto analysis, we are ready.
*/
if (params->no_aa)
return top;
retval = timerlat_aa_init(top, nr_cpus, params->dump_tasks);
if (retval)
goto out_err;
return top; return top;
out_err: out_err:
...@@ -688,6 +725,10 @@ int timerlat_top_main(int argc, char *argv[]) ...@@ -688,6 +725,10 @@ int timerlat_top_main(int argc, char *argv[])
if (trace_is_off(&top->trace, &record->trace)) { if (trace_is_off(&top->trace, &record->trace)) {
printf("rtla timerlat hit stop tracing\n"); printf("rtla timerlat hit stop tracing\n");
if (!params->no_aa)
timerlat_auto_analysis(params->stop_us, params->stop_total_us);
if (params->trace_output) { if (params->trace_output) {
printf(" Saving trace to %s\n", params->trace_output); printf(" Saving trace to %s\n", params->trace_output);
save_trace_to_file(record->trace.inst, params->trace_output); save_trace_to_file(record->trace.inst, params->trace_output);
...@@ -701,6 +742,7 @@ int timerlat_top_main(int argc, char *argv[]) ...@@ -701,6 +742,7 @@ int timerlat_top_main(int argc, char *argv[])
params->events = NULL; params->events = NULL;
out_free: out_free:
timerlat_free_top(top->data); timerlat_free_top(top->data);
timerlat_aa_destroy();
osnoise_destroy_tool(record); osnoise_destroy_tool(record);
osnoise_destroy_tool(top); osnoise_destroy_tool(top);
free(params); free(params);
......
...@@ -56,3 +56,6 @@ struct sched_attr { ...@@ -56,3 +56,6 @@ struct sched_attr {
int parse_prio(char *arg, struct sched_attr *sched_param); int parse_prio(char *arg, struct sched_attr *sched_param);
int set_comm_sched_attr(const char *comm_prefix, struct sched_attr *attr); int set_comm_sched_attr(const char *comm_prefix, struct sched_attr *attr);
int set_cpu_dma_latency(int32_t latency); int set_cpu_dma_latency(int32_t latency);
#define ns_to_usf(x) (((double)x/1000))
#define ns_to_per(total, part) ((part * 100) / (double)total)
...@@ -519,7 +519,7 @@ static void ikm_usage_print_reactors(void) ...@@ -519,7 +519,7 @@ static void ikm_usage_print_reactors(void)
start = ++end; start = ++end;
end = strstr(start, "\n"); end = strstr(start, "\n");
}; }
fprintf(stderr, "\n"); fprintf(stderr, "\n");
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment