forked from luck/tmp_suning_uos_patched
fgraph: Initialize tracing_graph_pause at task creation
commit 7e0a9220467dbcfdc5bc62825724f3e52e50ab31 upstream.
On some archs, the idle task can call into cpu_suspend(). The cpu_suspend()
will disable or pause function graph tracing, as there's some paths in
bringing down the CPU that can have issues with its return address being
modified. The task_struct structure has a "tracing_graph_pause" atomic
counter, that when set to something other than zero, the function graph
tracer will not modify the return address.
The problem is that the tracing_graph_pause counter is initialized when the
function graph tracer is enabled. This can corrupt the counter for the idle
task if it is suspended in these architectures.
CPU 1 CPU 2
----- -----
do_idle()
cpu_suspend()
pause_graph_tracing()
task_struct->tracing_graph_pause++ (0 -> 1)
start_graph_tracing()
for_each_online_cpu(cpu) {
ftrace_graph_init_idle_task(cpu)
task-struct->tracing_graph_pause = 0 (1 -> 0)
unpause_graph_tracing()
task_struct->tracing_graph_pause-- (0 -> -1)
The above should have gone from 1 to zero, and enabled function graph
tracing again. But instead, it is set to -1, which keeps it disabled.
There's no reason that the field tracing_graph_pause on the task_struct can
not be initialized at boot up.
Cc: stable@vger.kernel.org
Fixes: 380c4b1411
("tracing/function-graph-tracer: append the tracing_graph_flag")
Bugzilla: https://bugzilla.kernel.org/show_bug.cgi?id=211339
Reported-by: pierre.gondois@arm.com
Signed-off-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
8847a756e1
commit
43b5bdbf96
|
@ -199,6 +199,7 @@ struct task_struct init_task
|
||||||
#endif
|
#endif
|
||||||
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
||||||
.ret_stack = NULL,
|
.ret_stack = NULL,
|
||||||
|
.tracing_graph_pause = ATOMIC_INIT(0),
|
||||||
#endif
|
#endif
|
||||||
#if defined(CONFIG_TRACING) && defined(CONFIG_PREEMPTION)
|
#if defined(CONFIG_TRACING) && defined(CONFIG_PREEMPTION)
|
||||||
.trace_recursion = 0,
|
.trace_recursion = 0,
|
||||||
|
|
|
@ -395,7 +395,6 @@ static int alloc_retstack_tasklist(struct ftrace_ret_stack **ret_stack_list)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (t->ret_stack == NULL) {
|
if (t->ret_stack == NULL) {
|
||||||
atomic_set(&t->tracing_graph_pause, 0);
|
|
||||||
atomic_set(&t->trace_overrun, 0);
|
atomic_set(&t->trace_overrun, 0);
|
||||||
t->curr_ret_stack = -1;
|
t->curr_ret_stack = -1;
|
||||||
t->curr_ret_depth = -1;
|
t->curr_ret_depth = -1;
|
||||||
|
@ -490,7 +489,6 @@ static DEFINE_PER_CPU(struct ftrace_ret_stack *, idle_ret_stack);
|
||||||
static void
|
static void
|
||||||
graph_init_task(struct task_struct *t, struct ftrace_ret_stack *ret_stack)
|
graph_init_task(struct task_struct *t, struct ftrace_ret_stack *ret_stack)
|
||||||
{
|
{
|
||||||
atomic_set(&t->tracing_graph_pause, 0);
|
|
||||||
atomic_set(&t->trace_overrun, 0);
|
atomic_set(&t->trace_overrun, 0);
|
||||||
t->ftrace_timestamp = 0;
|
t->ftrace_timestamp = 0;
|
||||||
/* make curr_ret_stack visible before we add the ret_stack */
|
/* make curr_ret_stack visible before we add the ret_stack */
|
||||||
|
|
Loading…
Reference in New Issue
Block a user