Merge branch 'tip/perf/core' of git://git.kernel.org/pub/scm/linux/kernel/git/rostedt/linux-2.6-trace into perf/core
This commit is contained in:
@@ -1638,8 +1638,8 @@ ftrace_failures_open(struct inode *inode, struct file *file)
|
|||||||
|
|
||||||
ret = ftrace_avail_open(inode, file);
|
ret = ftrace_avail_open(inode, file);
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
m = (struct seq_file *)file->private_data;
|
m = file->private_data;
|
||||||
iter = (struct ftrace_iterator *)m->private;
|
iter = m->private;
|
||||||
iter->flags = FTRACE_ITER_FAILURES;
|
iter->flags = FTRACE_ITER_FAILURES;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -2196,7 +2196,7 @@ int tracing_open_generic(struct inode *inode, struct file *filp)
|
|||||||
|
|
||||||
static int tracing_release(struct inode *inode, struct file *file)
|
static int tracing_release(struct inode *inode, struct file *file)
|
||||||
{
|
{
|
||||||
struct seq_file *m = (struct seq_file *)file->private_data;
|
struct seq_file *m = file->private_data;
|
||||||
struct trace_iterator *iter;
|
struct trace_iterator *iter;
|
||||||
int cpu;
|
int cpu;
|
||||||
|
|
||||||
|
@@ -343,6 +343,10 @@ void trace_function(struct trace_array *tr,
|
|||||||
unsigned long ip,
|
unsigned long ip,
|
||||||
unsigned long parent_ip,
|
unsigned long parent_ip,
|
||||||
unsigned long flags, int pc);
|
unsigned long flags, int pc);
|
||||||
|
void trace_graph_function(struct trace_array *tr,
|
||||||
|
unsigned long ip,
|
||||||
|
unsigned long parent_ip,
|
||||||
|
unsigned long flags, int pc);
|
||||||
void trace_default_header(struct seq_file *m);
|
void trace_default_header(struct seq_file *m);
|
||||||
void print_trace_header(struct seq_file *m, struct trace_iterator *iter);
|
void print_trace_header(struct seq_file *m, struct trace_iterator *iter);
|
||||||
int trace_empty(struct trace_iterator *iter);
|
int trace_empty(struct trace_iterator *iter);
|
||||||
|
@@ -262,6 +262,34 @@ int trace_graph_thresh_entry(struct ftrace_graph_ent *trace)
|
|||||||
return trace_graph_entry(trace);
|
return trace_graph_entry(trace);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
__trace_graph_function(struct trace_array *tr,
|
||||||
|
unsigned long ip, unsigned long flags, int pc)
|
||||||
|
{
|
||||||
|
u64 time = trace_clock_local();
|
||||||
|
struct ftrace_graph_ent ent = {
|
||||||
|
.func = ip,
|
||||||
|
.depth = 0,
|
||||||
|
};
|
||||||
|
struct ftrace_graph_ret ret = {
|
||||||
|
.func = ip,
|
||||||
|
.depth = 0,
|
||||||
|
.calltime = time,
|
||||||
|
.rettime = time,
|
||||||
|
};
|
||||||
|
|
||||||
|
__trace_graph_entry(tr, &ent, flags, pc);
|
||||||
|
__trace_graph_return(tr, &ret, flags, pc);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
trace_graph_function(struct trace_array *tr,
|
||||||
|
unsigned long ip, unsigned long parent_ip,
|
||||||
|
unsigned long flags, int pc)
|
||||||
|
{
|
||||||
|
__trace_graph_function(tr, ip, flags, pc);
|
||||||
|
}
|
||||||
|
|
||||||
void __trace_graph_return(struct trace_array *tr,
|
void __trace_graph_return(struct trace_array *tr,
|
||||||
struct ftrace_graph_ret *trace,
|
struct ftrace_graph_ret *trace,
|
||||||
unsigned long flags,
|
unsigned long flags,
|
||||||
@@ -888,12 +916,20 @@ check_irq_entry(struct trace_iterator *iter, u32 flags,
|
|||||||
unsigned long addr, int depth)
|
unsigned long addr, int depth)
|
||||||
{
|
{
|
||||||
int cpu = iter->cpu;
|
int cpu = iter->cpu;
|
||||||
|
int *depth_irq;
|
||||||
struct fgraph_data *data = iter->private;
|
struct fgraph_data *data = iter->private;
|
||||||
int *depth_irq = &(per_cpu_ptr(data->cpu_data, cpu)->depth_irq);
|
|
||||||
|
|
||||||
if (flags & TRACE_GRAPH_PRINT_IRQS)
|
/*
|
||||||
|
* If we are either displaying irqs, or we got called as
|
||||||
|
* a graph event and private data does not exist,
|
||||||
|
* then we bypass the irq check.
|
||||||
|
*/
|
||||||
|
if ((flags & TRACE_GRAPH_PRINT_IRQS) ||
|
||||||
|
(!data))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
depth_irq = &(per_cpu_ptr(data->cpu_data, cpu)->depth_irq);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We are inside the irq code
|
* We are inside the irq code
|
||||||
*/
|
*/
|
||||||
@@ -926,12 +962,20 @@ static int
|
|||||||
check_irq_return(struct trace_iterator *iter, u32 flags, int depth)
|
check_irq_return(struct trace_iterator *iter, u32 flags, int depth)
|
||||||
{
|
{
|
||||||
int cpu = iter->cpu;
|
int cpu = iter->cpu;
|
||||||
|
int *depth_irq;
|
||||||
struct fgraph_data *data = iter->private;
|
struct fgraph_data *data = iter->private;
|
||||||
int *depth_irq = &(per_cpu_ptr(data->cpu_data, cpu)->depth_irq);
|
|
||||||
|
|
||||||
if (flags & TRACE_GRAPH_PRINT_IRQS)
|
/*
|
||||||
|
* If we are either displaying irqs, or we got called as
|
||||||
|
* a graph event and private data does not exist,
|
||||||
|
* then we bypass the irq check.
|
||||||
|
*/
|
||||||
|
if ((flags & TRACE_GRAPH_PRINT_IRQS) ||
|
||||||
|
(!data))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
depth_irq = &(per_cpu_ptr(data->cpu_data, cpu)->depth_irq);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We are not inside the irq code.
|
* We are not inside the irq code.
|
||||||
*/
|
*/
|
||||||
@@ -1163,7 +1207,7 @@ print_graph_comment(struct trace_seq *s, struct trace_entry *ent,
|
|||||||
|
|
||||||
|
|
||||||
enum print_line_t
|
enum print_line_t
|
||||||
print_graph_function_flags(struct trace_iterator *iter, u32 flags)
|
__print_graph_function_flags(struct trace_iterator *iter, u32 flags)
|
||||||
{
|
{
|
||||||
struct ftrace_graph_ent_entry *field;
|
struct ftrace_graph_ent_entry *field;
|
||||||
struct fgraph_data *data = iter->private;
|
struct fgraph_data *data = iter->private;
|
||||||
@@ -1226,7 +1270,18 @@ print_graph_function_flags(struct trace_iterator *iter, u32 flags)
|
|||||||
static enum print_line_t
|
static enum print_line_t
|
||||||
print_graph_function(struct trace_iterator *iter)
|
print_graph_function(struct trace_iterator *iter)
|
||||||
{
|
{
|
||||||
return print_graph_function_flags(iter, tracer_flags.val);
|
return __print_graph_function_flags(iter, tracer_flags.val);
|
||||||
|
}
|
||||||
|
|
||||||
|
enum print_line_t print_graph_function_flags(struct trace_iterator *iter,
|
||||||
|
u32 flags)
|
||||||
|
{
|
||||||
|
if (trace_flags & TRACE_ITER_LATENCY_FMT)
|
||||||
|
flags |= TRACE_GRAPH_PRINT_DURATION;
|
||||||
|
else
|
||||||
|
flags |= TRACE_GRAPH_PRINT_ABS_TIME;
|
||||||
|
|
||||||
|
return __print_graph_function_flags(iter, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum print_line_t
|
static enum print_line_t
|
||||||
@@ -1258,7 +1313,7 @@ static void print_lat_header(struct seq_file *s, u32 flags)
|
|||||||
seq_printf(s, "#%.*s|||| / \n", size, spaces);
|
seq_printf(s, "#%.*s|||| / \n", size, spaces);
|
||||||
}
|
}
|
||||||
|
|
||||||
void print_graph_headers_flags(struct seq_file *s, u32 flags)
|
static void __print_graph_headers_flags(struct seq_file *s, u32 flags)
|
||||||
{
|
{
|
||||||
int lat = trace_flags & TRACE_ITER_LATENCY_FMT;
|
int lat = trace_flags & TRACE_ITER_LATENCY_FMT;
|
||||||
|
|
||||||
@@ -1299,6 +1354,23 @@ void print_graph_headers(struct seq_file *s)
|
|||||||
print_graph_headers_flags(s, tracer_flags.val);
|
print_graph_headers_flags(s, tracer_flags.val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void print_graph_headers_flags(struct seq_file *s, u32 flags)
|
||||||
|
{
|
||||||
|
struct trace_iterator *iter = s->private;
|
||||||
|
|
||||||
|
if (trace_flags & TRACE_ITER_LATENCY_FMT) {
|
||||||
|
/* print nothing if the buffers are empty */
|
||||||
|
if (trace_empty(iter))
|
||||||
|
return;
|
||||||
|
|
||||||
|
print_trace_header(s, iter);
|
||||||
|
flags |= TRACE_GRAPH_PRINT_DURATION;
|
||||||
|
} else
|
||||||
|
flags |= TRACE_GRAPH_PRINT_ABS_TIME;
|
||||||
|
|
||||||
|
__print_graph_headers_flags(s, flags);
|
||||||
|
}
|
||||||
|
|
||||||
void graph_trace_open(struct trace_iterator *iter)
|
void graph_trace_open(struct trace_iterator *iter)
|
||||||
{
|
{
|
||||||
/* pid and depth on the last trace processed */
|
/* pid and depth on the last trace processed */
|
||||||
|
@@ -87,14 +87,22 @@ static __cacheline_aligned_in_smp unsigned long max_sequence;
|
|||||||
|
|
||||||
#ifdef CONFIG_FUNCTION_TRACER
|
#ifdef CONFIG_FUNCTION_TRACER
|
||||||
/*
|
/*
|
||||||
* irqsoff uses its own tracer function to keep the overhead down:
|
* Prologue for the preempt and irqs off function tracers.
|
||||||
|
*
|
||||||
|
* Returns 1 if it is OK to continue, and data->disabled is
|
||||||
|
* incremented.
|
||||||
|
* 0 if the trace is to be ignored, and data->disabled
|
||||||
|
* is kept the same.
|
||||||
|
*
|
||||||
|
* Note, this function is also used outside this ifdef but
|
||||||
|
* inside the #ifdef of the function graph tracer below.
|
||||||
|
* This is OK, since the function graph tracer is
|
||||||
|
* dependent on the function tracer.
|
||||||
*/
|
*/
|
||||||
static void
|
static int func_prolog_dec(struct trace_array *tr,
|
||||||
irqsoff_tracer_call(unsigned long ip, unsigned long parent_ip)
|
struct trace_array_cpu **data,
|
||||||
|
unsigned long *flags)
|
||||||
{
|
{
|
||||||
struct trace_array *tr = irqsoff_trace;
|
|
||||||
struct trace_array_cpu *data;
|
|
||||||
unsigned long flags;
|
|
||||||
long disabled;
|
long disabled;
|
||||||
int cpu;
|
int cpu;
|
||||||
|
|
||||||
@@ -106,17 +114,37 @@ irqsoff_tracer_call(unsigned long ip, unsigned long parent_ip)
|
|||||||
*/
|
*/
|
||||||
cpu = raw_smp_processor_id();
|
cpu = raw_smp_processor_id();
|
||||||
if (likely(!per_cpu(tracing_cpu, cpu)))
|
if (likely(!per_cpu(tracing_cpu, cpu)))
|
||||||
return;
|
return 0;
|
||||||
|
|
||||||
local_save_flags(flags);
|
local_save_flags(*flags);
|
||||||
/* slight chance to get a false positive on tracing_cpu */
|
/* slight chance to get a false positive on tracing_cpu */
|
||||||
if (!irqs_disabled_flags(flags))
|
if (!irqs_disabled_flags(*flags))
|
||||||
return;
|
return 0;
|
||||||
|
|
||||||
data = tr->data[cpu];
|
*data = tr->data[cpu];
|
||||||
disabled = atomic_inc_return(&data->disabled);
|
disabled = atomic_inc_return(&(*data)->disabled);
|
||||||
|
|
||||||
if (likely(disabled == 1))
|
if (likely(disabled == 1))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
atomic_dec(&(*data)->disabled);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* irqsoff uses its own tracer function to keep the overhead down:
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
irqsoff_tracer_call(unsigned long ip, unsigned long parent_ip)
|
||||||
|
{
|
||||||
|
struct trace_array *tr = irqsoff_trace;
|
||||||
|
struct trace_array_cpu *data;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
if (!func_prolog_dec(tr, &data, &flags))
|
||||||
|
return;
|
||||||
|
|
||||||
trace_function(tr, ip, parent_ip, flags, preempt_count());
|
trace_function(tr, ip, parent_ip, flags, preempt_count());
|
||||||
|
|
||||||
atomic_dec(&data->disabled);
|
atomic_dec(&data->disabled);
|
||||||
@@ -155,30 +183,16 @@ static int irqsoff_graph_entry(struct ftrace_graph_ent *trace)
|
|||||||
struct trace_array *tr = irqsoff_trace;
|
struct trace_array *tr = irqsoff_trace;
|
||||||
struct trace_array_cpu *data;
|
struct trace_array_cpu *data;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
long disabled;
|
|
||||||
int ret;
|
int ret;
|
||||||
int cpu;
|
|
||||||
int pc;
|
int pc;
|
||||||
|
|
||||||
cpu = raw_smp_processor_id();
|
if (!func_prolog_dec(tr, &data, &flags))
|
||||||
if (likely(!per_cpu(tracing_cpu, cpu)))
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
local_save_flags(flags);
|
|
||||||
/* slight chance to get a false positive on tracing_cpu */
|
|
||||||
if (!irqs_disabled_flags(flags))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
data = tr->data[cpu];
|
|
||||||
disabled = atomic_inc_return(&data->disabled);
|
|
||||||
|
|
||||||
if (likely(disabled == 1)) {
|
|
||||||
pc = preempt_count();
|
pc = preempt_count();
|
||||||
ret = __trace_graph_entry(tr, trace, flags, pc);
|
ret = __trace_graph_entry(tr, trace, flags, pc);
|
||||||
} else
|
|
||||||
ret = 0;
|
|
||||||
|
|
||||||
atomic_dec(&data->disabled);
|
atomic_dec(&data->disabled);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -187,27 +201,13 @@ static void irqsoff_graph_return(struct ftrace_graph_ret *trace)
|
|||||||
struct trace_array *tr = irqsoff_trace;
|
struct trace_array *tr = irqsoff_trace;
|
||||||
struct trace_array_cpu *data;
|
struct trace_array_cpu *data;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
long disabled;
|
|
||||||
int cpu;
|
|
||||||
int pc;
|
int pc;
|
||||||
|
|
||||||
cpu = raw_smp_processor_id();
|
if (!func_prolog_dec(tr, &data, &flags))
|
||||||
if (likely(!per_cpu(tracing_cpu, cpu)))
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
local_save_flags(flags);
|
|
||||||
/* slight chance to get a false positive on tracing_cpu */
|
|
||||||
if (!irqs_disabled_flags(flags))
|
|
||||||
return;
|
|
||||||
|
|
||||||
data = tr->data[cpu];
|
|
||||||
disabled = atomic_inc_return(&data->disabled);
|
|
||||||
|
|
||||||
if (likely(disabled == 1)) {
|
|
||||||
pc = preempt_count();
|
pc = preempt_count();
|
||||||
__trace_graph_return(tr, trace, flags, pc);
|
__trace_graph_return(tr, trace, flags, pc);
|
||||||
}
|
|
||||||
|
|
||||||
atomic_dec(&data->disabled);
|
atomic_dec(&data->disabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -229,75 +229,33 @@ static void irqsoff_trace_close(struct trace_iterator *iter)
|
|||||||
|
|
||||||
static enum print_line_t irqsoff_print_line(struct trace_iterator *iter)
|
static enum print_line_t irqsoff_print_line(struct trace_iterator *iter)
|
||||||
{
|
{
|
||||||
u32 flags = GRAPH_TRACER_FLAGS;
|
|
||||||
|
|
||||||
if (trace_flags & TRACE_ITER_LATENCY_FMT)
|
|
||||||
flags |= TRACE_GRAPH_PRINT_DURATION;
|
|
||||||
else
|
|
||||||
flags |= TRACE_GRAPH_PRINT_ABS_TIME;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* In graph mode call the graph tracer output function,
|
* In graph mode call the graph tracer output function,
|
||||||
* otherwise go with the TRACE_FN event handler
|
* otherwise go with the TRACE_FN event handler
|
||||||
*/
|
*/
|
||||||
if (is_graph())
|
if (is_graph())
|
||||||
return print_graph_function_flags(iter, flags);
|
return print_graph_function_flags(iter, GRAPH_TRACER_FLAGS);
|
||||||
|
|
||||||
return TRACE_TYPE_UNHANDLED;
|
return TRACE_TYPE_UNHANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void irqsoff_print_header(struct seq_file *s)
|
static void irqsoff_print_header(struct seq_file *s)
|
||||||
{
|
{
|
||||||
if (is_graph()) {
|
if (is_graph())
|
||||||
struct trace_iterator *iter = s->private;
|
print_graph_headers_flags(s, GRAPH_TRACER_FLAGS);
|
||||||
u32 flags = GRAPH_TRACER_FLAGS;
|
else
|
||||||
|
|
||||||
if (trace_flags & TRACE_ITER_LATENCY_FMT) {
|
|
||||||
/* print nothing if the buffers are empty */
|
|
||||||
if (trace_empty(iter))
|
|
||||||
return;
|
|
||||||
|
|
||||||
print_trace_header(s, iter);
|
|
||||||
flags |= TRACE_GRAPH_PRINT_DURATION;
|
|
||||||
} else
|
|
||||||
flags |= TRACE_GRAPH_PRINT_ABS_TIME;
|
|
||||||
|
|
||||||
print_graph_headers_flags(s, flags);
|
|
||||||
} else
|
|
||||||
trace_default_header(s);
|
trace_default_header(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
trace_graph_function(struct trace_array *tr,
|
|
||||||
unsigned long ip, unsigned long flags, int pc)
|
|
||||||
{
|
|
||||||
u64 time = trace_clock_local();
|
|
||||||
struct ftrace_graph_ent ent = {
|
|
||||||
.func = ip,
|
|
||||||
.depth = 0,
|
|
||||||
};
|
|
||||||
struct ftrace_graph_ret ret = {
|
|
||||||
.func = ip,
|
|
||||||
.depth = 0,
|
|
||||||
.calltime = time,
|
|
||||||
.rettime = time,
|
|
||||||
};
|
|
||||||
|
|
||||||
__trace_graph_entry(tr, &ent, flags, pc);
|
|
||||||
__trace_graph_return(tr, &ret, flags, pc);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
__trace_function(struct trace_array *tr,
|
__trace_function(struct trace_array *tr,
|
||||||
unsigned long ip, unsigned long parent_ip,
|
unsigned long ip, unsigned long parent_ip,
|
||||||
unsigned long flags, int pc)
|
unsigned long flags, int pc)
|
||||||
{
|
{
|
||||||
if (!is_graph())
|
if (is_graph())
|
||||||
|
trace_graph_function(tr, ip, parent_ip, flags, pc);
|
||||||
|
else
|
||||||
trace_function(tr, ip, parent_ip, flags, pc);
|
trace_function(tr, ip, parent_ip, flags, pc);
|
||||||
else {
|
|
||||||
trace_graph_function(tr, parent_ip, flags, pc);
|
|
||||||
trace_graph_function(tr, ip, flags, pc);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
@@ -31,13 +31,81 @@ static int wakeup_rt;
|
|||||||
static arch_spinlock_t wakeup_lock =
|
static arch_spinlock_t wakeup_lock =
|
||||||
(arch_spinlock_t)__ARCH_SPIN_LOCK_UNLOCKED;
|
(arch_spinlock_t)__ARCH_SPIN_LOCK_UNLOCKED;
|
||||||
|
|
||||||
|
static void wakeup_reset(struct trace_array *tr);
|
||||||
static void __wakeup_reset(struct trace_array *tr);
|
static void __wakeup_reset(struct trace_array *tr);
|
||||||
|
static int wakeup_graph_entry(struct ftrace_graph_ent *trace);
|
||||||
|
static void wakeup_graph_return(struct ftrace_graph_ret *trace);
|
||||||
|
|
||||||
static int save_lat_flag;
|
static int save_lat_flag;
|
||||||
|
|
||||||
|
#define TRACE_DISPLAY_GRAPH 1
|
||||||
|
|
||||||
|
static struct tracer_opt trace_opts[] = {
|
||||||
|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
||||||
|
/* display latency trace as call graph */
|
||||||
|
{ TRACER_OPT(display-graph, TRACE_DISPLAY_GRAPH) },
|
||||||
|
#endif
|
||||||
|
{ } /* Empty entry */
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct tracer_flags tracer_flags = {
|
||||||
|
.val = 0,
|
||||||
|
.opts = trace_opts,
|
||||||
|
};
|
||||||
|
|
||||||
|
#define is_graph() (tracer_flags.val & TRACE_DISPLAY_GRAPH)
|
||||||
|
|
||||||
#ifdef CONFIG_FUNCTION_TRACER
|
#ifdef CONFIG_FUNCTION_TRACER
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* irqsoff uses its own tracer function to keep the overhead down:
|
* Prologue for the wakeup function tracers.
|
||||||
|
*
|
||||||
|
* Returns 1 if it is OK to continue, and preemption
|
||||||
|
* is disabled and data->disabled is incremented.
|
||||||
|
* 0 if the trace is to be ignored, and preemption
|
||||||
|
* is not disabled and data->disabled is
|
||||||
|
* kept the same.
|
||||||
|
*
|
||||||
|
* Note, this function is also used outside this ifdef but
|
||||||
|
* inside the #ifdef of the function graph tracer below.
|
||||||
|
* This is OK, since the function graph tracer is
|
||||||
|
* dependent on the function tracer.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
func_prolog_preempt_disable(struct trace_array *tr,
|
||||||
|
struct trace_array_cpu **data,
|
||||||
|
int *pc)
|
||||||
|
{
|
||||||
|
long disabled;
|
||||||
|
int cpu;
|
||||||
|
|
||||||
|
if (likely(!wakeup_task))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
*pc = preempt_count();
|
||||||
|
preempt_disable_notrace();
|
||||||
|
|
||||||
|
cpu = raw_smp_processor_id();
|
||||||
|
if (cpu != wakeup_current_cpu)
|
||||||
|
goto out_enable;
|
||||||
|
|
||||||
|
*data = tr->data[cpu];
|
||||||
|
disabled = atomic_inc_return(&(*data)->disabled);
|
||||||
|
if (unlikely(disabled != 1))
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
out:
|
||||||
|
atomic_dec(&(*data)->disabled);
|
||||||
|
|
||||||
|
out_enable:
|
||||||
|
preempt_enable_notrace();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* wakeup uses its own tracer function to keep the overhead down:
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
wakeup_tracer_call(unsigned long ip, unsigned long parent_ip)
|
wakeup_tracer_call(unsigned long ip, unsigned long parent_ip)
|
||||||
@@ -45,34 +113,16 @@ wakeup_tracer_call(unsigned long ip, unsigned long parent_ip)
|
|||||||
struct trace_array *tr = wakeup_trace;
|
struct trace_array *tr = wakeup_trace;
|
||||||
struct trace_array_cpu *data;
|
struct trace_array_cpu *data;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
long disabled;
|
|
||||||
int cpu;
|
|
||||||
int pc;
|
int pc;
|
||||||
|
|
||||||
if (likely(!wakeup_task))
|
if (!func_prolog_preempt_disable(tr, &data, &pc))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
pc = preempt_count();
|
|
||||||
preempt_disable_notrace();
|
|
||||||
|
|
||||||
cpu = raw_smp_processor_id();
|
|
||||||
if (cpu != wakeup_current_cpu)
|
|
||||||
goto out_enable;
|
|
||||||
|
|
||||||
data = tr->data[cpu];
|
|
||||||
disabled = atomic_inc_return(&data->disabled);
|
|
||||||
if (unlikely(disabled != 1))
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
local_irq_save(flags);
|
local_irq_save(flags);
|
||||||
|
|
||||||
trace_function(tr, ip, parent_ip, flags, pc);
|
trace_function(tr, ip, parent_ip, flags, pc);
|
||||||
|
|
||||||
local_irq_restore(flags);
|
local_irq_restore(flags);
|
||||||
|
|
||||||
out:
|
|
||||||
atomic_dec(&data->disabled);
|
atomic_dec(&data->disabled);
|
||||||
out_enable:
|
|
||||||
preempt_enable_notrace();
|
preempt_enable_notrace();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -82,6 +132,156 @@ static struct ftrace_ops trace_ops __read_mostly =
|
|||||||
};
|
};
|
||||||
#endif /* CONFIG_FUNCTION_TRACER */
|
#endif /* CONFIG_FUNCTION_TRACER */
|
||||||
|
|
||||||
|
static int start_func_tracer(int graph)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (!graph)
|
||||||
|
ret = register_ftrace_function(&trace_ops);
|
||||||
|
else
|
||||||
|
ret = register_ftrace_graph(&wakeup_graph_return,
|
||||||
|
&wakeup_graph_entry);
|
||||||
|
|
||||||
|
if (!ret && tracing_is_enabled())
|
||||||
|
tracer_enabled = 1;
|
||||||
|
else
|
||||||
|
tracer_enabled = 0;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void stop_func_tracer(int graph)
|
||||||
|
{
|
||||||
|
tracer_enabled = 0;
|
||||||
|
|
||||||
|
if (!graph)
|
||||||
|
unregister_ftrace_function(&trace_ops);
|
||||||
|
else
|
||||||
|
unregister_ftrace_graph();
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
||||||
|
static int wakeup_set_flag(u32 old_flags, u32 bit, int set)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (!(bit & TRACE_DISPLAY_GRAPH))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (!(is_graph() ^ set))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
stop_func_tracer(!set);
|
||||||
|
|
||||||
|
wakeup_reset(wakeup_trace);
|
||||||
|
tracing_max_latency = 0;
|
||||||
|
|
||||||
|
return start_func_tracer(set);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int wakeup_graph_entry(struct ftrace_graph_ent *trace)
|
||||||
|
{
|
||||||
|
struct trace_array *tr = wakeup_trace;
|
||||||
|
struct trace_array_cpu *data;
|
||||||
|
unsigned long flags;
|
||||||
|
int pc, ret = 0;
|
||||||
|
|
||||||
|
if (!func_prolog_preempt_disable(tr, &data, &pc))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
local_save_flags(flags);
|
||||||
|
ret = __trace_graph_entry(tr, trace, flags, pc);
|
||||||
|
atomic_dec(&data->disabled);
|
||||||
|
preempt_enable_notrace();
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wakeup_graph_return(struct ftrace_graph_ret *trace)
|
||||||
|
{
|
||||||
|
struct trace_array *tr = wakeup_trace;
|
||||||
|
struct trace_array_cpu *data;
|
||||||
|
unsigned long flags;
|
||||||
|
int pc;
|
||||||
|
|
||||||
|
if (!func_prolog_preempt_disable(tr, &data, &pc))
|
||||||
|
return;
|
||||||
|
|
||||||
|
local_save_flags(flags);
|
||||||
|
__trace_graph_return(tr, trace, flags, pc);
|
||||||
|
atomic_dec(&data->disabled);
|
||||||
|
|
||||||
|
preempt_enable_notrace();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wakeup_trace_open(struct trace_iterator *iter)
|
||||||
|
{
|
||||||
|
if (is_graph())
|
||||||
|
graph_trace_open(iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wakeup_trace_close(struct trace_iterator *iter)
|
||||||
|
{
|
||||||
|
if (iter->private)
|
||||||
|
graph_trace_close(iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define GRAPH_TRACER_FLAGS (TRACE_GRAPH_PRINT_PROC)
|
||||||
|
|
||||||
|
static enum print_line_t wakeup_print_line(struct trace_iterator *iter)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* In graph mode call the graph tracer output function,
|
||||||
|
* otherwise go with the TRACE_FN event handler
|
||||||
|
*/
|
||||||
|
if (is_graph())
|
||||||
|
return print_graph_function_flags(iter, GRAPH_TRACER_FLAGS);
|
||||||
|
|
||||||
|
return TRACE_TYPE_UNHANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wakeup_print_header(struct seq_file *s)
|
||||||
|
{
|
||||||
|
if (is_graph())
|
||||||
|
print_graph_headers_flags(s, GRAPH_TRACER_FLAGS);
|
||||||
|
else
|
||||||
|
trace_default_header(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
__trace_function(struct trace_array *tr,
|
||||||
|
unsigned long ip, unsigned long parent_ip,
|
||||||
|
unsigned long flags, int pc)
|
||||||
|
{
|
||||||
|
if (is_graph())
|
||||||
|
trace_graph_function(tr, ip, parent_ip, flags, pc);
|
||||||
|
else
|
||||||
|
trace_function(tr, ip, parent_ip, flags, pc);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
#define __trace_function trace_function
|
||||||
|
|
||||||
|
static int wakeup_set_flag(u32 old_flags, u32 bit, int set)
|
||||||
|
{
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int wakeup_graph_entry(struct ftrace_graph_ent *trace)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum print_line_t wakeup_print_line(struct trace_iterator *iter)
|
||||||
|
{
|
||||||
|
return TRACE_TYPE_UNHANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wakeup_graph_return(struct ftrace_graph_ret *trace) { }
|
||||||
|
static void wakeup_print_header(struct seq_file *s) { }
|
||||||
|
static void wakeup_trace_open(struct trace_iterator *iter) { }
|
||||||
|
static void wakeup_trace_close(struct trace_iterator *iter) { }
|
||||||
|
#endif /* CONFIG_FUNCTION_GRAPH_TRACER */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Should this new latency be reported/recorded?
|
* Should this new latency be reported/recorded?
|
||||||
*/
|
*/
|
||||||
@@ -152,7 +352,7 @@ probe_wakeup_sched_switch(void *ignore,
|
|||||||
/* The task we are waiting for is waking up */
|
/* The task we are waiting for is waking up */
|
||||||
data = wakeup_trace->data[wakeup_cpu];
|
data = wakeup_trace->data[wakeup_cpu];
|
||||||
|
|
||||||
trace_function(wakeup_trace, CALLER_ADDR0, CALLER_ADDR1, flags, pc);
|
__trace_function(wakeup_trace, CALLER_ADDR0, CALLER_ADDR1, flags, pc);
|
||||||
tracing_sched_switch_trace(wakeup_trace, prev, next, flags, pc);
|
tracing_sched_switch_trace(wakeup_trace, prev, next, flags, pc);
|
||||||
|
|
||||||
T0 = data->preempt_timestamp;
|
T0 = data->preempt_timestamp;
|
||||||
@@ -252,7 +452,7 @@ probe_wakeup(void *ignore, struct task_struct *p, int success)
|
|||||||
* is not called by an assembly function (where as schedule is)
|
* is not called by an assembly function (where as schedule is)
|
||||||
* it should be safe to use it here.
|
* it should be safe to use it here.
|
||||||
*/
|
*/
|
||||||
trace_function(wakeup_trace, CALLER_ADDR1, CALLER_ADDR2, flags, pc);
|
__trace_function(wakeup_trace, CALLER_ADDR1, CALLER_ADDR2, flags, pc);
|
||||||
|
|
||||||
out_locked:
|
out_locked:
|
||||||
arch_spin_unlock(&wakeup_lock);
|
arch_spin_unlock(&wakeup_lock);
|
||||||
@@ -303,12 +503,8 @@ static void start_wakeup_tracer(struct trace_array *tr)
|
|||||||
*/
|
*/
|
||||||
smp_wmb();
|
smp_wmb();
|
||||||
|
|
||||||
register_ftrace_function(&trace_ops);
|
if (start_func_tracer(is_graph()))
|
||||||
|
printk(KERN_ERR "failed to start wakeup tracer\n");
|
||||||
if (tracing_is_enabled())
|
|
||||||
tracer_enabled = 1;
|
|
||||||
else
|
|
||||||
tracer_enabled = 0;
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
fail_deprobe_wake_new:
|
fail_deprobe_wake_new:
|
||||||
@@ -320,7 +516,7 @@ fail_deprobe:
|
|||||||
static void stop_wakeup_tracer(struct trace_array *tr)
|
static void stop_wakeup_tracer(struct trace_array *tr)
|
||||||
{
|
{
|
||||||
tracer_enabled = 0;
|
tracer_enabled = 0;
|
||||||
unregister_ftrace_function(&trace_ops);
|
stop_func_tracer(is_graph());
|
||||||
unregister_trace_sched_switch(probe_wakeup_sched_switch, NULL);
|
unregister_trace_sched_switch(probe_wakeup_sched_switch, NULL);
|
||||||
unregister_trace_sched_wakeup_new(probe_wakeup, NULL);
|
unregister_trace_sched_wakeup_new(probe_wakeup, NULL);
|
||||||
unregister_trace_sched_wakeup(probe_wakeup, NULL);
|
unregister_trace_sched_wakeup(probe_wakeup, NULL);
|
||||||
@@ -379,9 +575,15 @@ static struct tracer wakeup_tracer __read_mostly =
|
|||||||
.start = wakeup_tracer_start,
|
.start = wakeup_tracer_start,
|
||||||
.stop = wakeup_tracer_stop,
|
.stop = wakeup_tracer_stop,
|
||||||
.print_max = 1,
|
.print_max = 1,
|
||||||
|
.print_header = wakeup_print_header,
|
||||||
|
.print_line = wakeup_print_line,
|
||||||
|
.flags = &tracer_flags,
|
||||||
|
.set_flag = wakeup_set_flag,
|
||||||
#ifdef CONFIG_FTRACE_SELFTEST
|
#ifdef CONFIG_FTRACE_SELFTEST
|
||||||
.selftest = trace_selftest_startup_wakeup,
|
.selftest = trace_selftest_startup_wakeup,
|
||||||
#endif
|
#endif
|
||||||
|
.open = wakeup_trace_open,
|
||||||
|
.close = wakeup_trace_close,
|
||||||
.use_max_tr = 1,
|
.use_max_tr = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -394,9 +596,15 @@ static struct tracer wakeup_rt_tracer __read_mostly =
|
|||||||
.stop = wakeup_tracer_stop,
|
.stop = wakeup_tracer_stop,
|
||||||
.wait_pipe = poll_wait_pipe,
|
.wait_pipe = poll_wait_pipe,
|
||||||
.print_max = 1,
|
.print_max = 1,
|
||||||
|
.print_header = wakeup_print_header,
|
||||||
|
.print_line = wakeup_print_line,
|
||||||
|
.flags = &tracer_flags,
|
||||||
|
.set_flag = wakeup_set_flag,
|
||||||
#ifdef CONFIG_FTRACE_SELFTEST
|
#ifdef CONFIG_FTRACE_SELFTEST
|
||||||
.selftest = trace_selftest_startup_wakeup,
|
.selftest = trace_selftest_startup_wakeup,
|
||||||
#endif
|
#endif
|
||||||
|
.open = wakeup_trace_open,
|
||||||
|
.close = wakeup_trace_close,
|
||||||
.use_max_tr = 1,
|
.use_max_tr = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user