mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-06 05:06:29 +00:00
function_graph: Have the instances use their own ftrace_ops for filtering
Allow for instances to have their own ftrace_ops part of the fgraph_ops that makes the funtion_graph tracer filter on the set_ftrace_filter file of the instance and not the top instance. This uses the new ftrace_startup_subops(), by using graph_ops as the "manager ops" that defines the callback function and adds the functions defined by the filters of the ops for each trace instance. The callback defined by the manager ops will call the registered fgraph ops that were added to the fgraph_array. Co-developed with Masami Hiramatsu: Link: https://lore.kernel.org/linux-trace-kernel/171509102088.162236.15758883237657317789.stgit@devnote2 Link: https://lore.kernel.org/linux-trace-kernel/20240603190822.832946261@goodmis.org Cc: Mark Rutland <mark.rutland@arm.com> Cc: Mathieu Desnoyers <mathieu.desnoyers@efficios.com> Cc: Andrew Morton <akpm@linux-foundation.org> Cc: Alexei Starovoitov <alexei.starovoitov@gmail.com> Cc: Florent Revest <revest@chromium.org> Cc: Martin KaFai Lau <martin.lau@linux.dev> Cc: bpf <bpf@vger.kernel.org> Cc: Sven Schnelle <svens@linux.ibm.com> Cc: Alexei Starovoitov <ast@kernel.org> Cc: Jiri Olsa <jolsa@kernel.org> Cc: Arnaldo Carvalho de Melo <acme@kernel.org> Cc: Daniel Borkmann <daniel@iogearbox.net> Cc: Alan Maguire <alan.maguire@oracle.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Guo Ren <guoren@kernel.org> Reviewed-by: Masami Hiramatsu (Google) <mhiramat@kernel.org> Signed-off-by: Steven Rostedt (VMware) <rostedt@goodmis.org> Signed-off-by: Masami Hiramatsu (Google) <mhiramat@kernel.org> Signed-off-by: Steven Rostedt (Google) <rostedt@goodmis.org>
This commit is contained in:
parent
d9bbfbd14f
commit
c132be2c4f
@ -1046,6 +1046,7 @@ extern int ftrace_graph_entry_stub(struct ftrace_graph_ent *trace, struct fgraph
|
||||
struct fgraph_ops {
|
||||
trace_func_graph_ent_t entryfunc;
|
||||
trace_func_graph_ret_t retfunc;
|
||||
struct ftrace_ops ops; /* for the hash lists */
|
||||
void *private;
|
||||
int idx;
|
||||
};
|
||||
|
@ -18,15 +18,6 @@
|
||||
#include "ftrace_internal.h"
|
||||
#include "trace.h"
|
||||
|
||||
#ifdef CONFIG_DYNAMIC_FTRACE
|
||||
#define ASSIGN_OPS_HASH(opsname, val) \
|
||||
.func_hash = val, \
|
||||
.local_hash.regex_lock = __MUTEX_INITIALIZER(opsname.local_hash.regex_lock), \
|
||||
.subop_list = LIST_HEAD_INIT(opsname.subop_list),
|
||||
#else
|
||||
#define ASSIGN_OPS_HASH(opsname, val)
|
||||
#endif
|
||||
|
||||
/*
|
||||
* FGRAPH_FRAME_SIZE: Size in bytes of the meta data on the shadow stack
|
||||
* FGRAPH_FRAME_OFFSET: Size in long words of the meta data frame
|
||||
@ -156,6 +147,13 @@ get_bitmap_bits(struct task_struct *t, int offset)
|
||||
return (t->ret_stack[offset] >> FGRAPH_INDEX_SHIFT) & FGRAPH_INDEX_MASK;
|
||||
}
|
||||
|
||||
/* For BITMAP type: set the bits in the bitmap bitmask at @offset on ret_stack */
|
||||
static inline void
|
||||
set_bitmap_bits(struct task_struct *t, int offset, unsigned long bitmap)
|
||||
{
|
||||
t->ret_stack[offset] |= (bitmap << FGRAPH_INDEX_SHIFT);
|
||||
}
|
||||
|
||||
/* Write the bitmap to the ret_stack at @offset (does index, offset and bitmask) */
|
||||
static inline void
|
||||
set_bitmap(struct task_struct *t, int offset, unsigned long bitmap)
|
||||
@ -382,7 +380,8 @@ int function_graph_enter(unsigned long ret, unsigned long func,
|
||||
if (gops == &fgraph_stub)
|
||||
continue;
|
||||
|
||||
if (gops->entryfunc(&trace, gops))
|
||||
if (ftrace_ops_test(&gops->ops, func, NULL) &&
|
||||
gops->entryfunc(&trace, gops))
|
||||
bitmap |= BIT(i);
|
||||
}
|
||||
|
||||
@ -665,16 +664,28 @@ unsigned long ftrace_graph_ret_addr(struct task_struct *task, int *idx,
|
||||
|
||||
static struct ftrace_ops graph_ops = {
|
||||
.func = ftrace_graph_func,
|
||||
.flags = FTRACE_OPS_FL_INITIALIZED |
|
||||
FTRACE_OPS_FL_PID |
|
||||
FTRACE_OPS_GRAPH_STUB,
|
||||
.flags = FTRACE_OPS_GRAPH_STUB,
|
||||
#ifdef FTRACE_GRAPH_TRAMP_ADDR
|
||||
.trampoline = FTRACE_GRAPH_TRAMP_ADDR,
|
||||
/* trampoline_size is only needed for dynamically allocated tramps */
|
||||
#endif
|
||||
ASSIGN_OPS_HASH(graph_ops, &global_ops.local_hash)
|
||||
};
|
||||
|
||||
void fgraph_init_ops(struct ftrace_ops *dst_ops,
|
||||
struct ftrace_ops *src_ops)
|
||||
{
|
||||
dst_ops->flags = FTRACE_OPS_FL_PID | FTRACE_OPS_GRAPH_STUB;
|
||||
|
||||
#ifdef CONFIG_DYNAMIC_FTRACE
|
||||
if (src_ops) {
|
||||
dst_ops->func_hash = &src_ops->local_hash;
|
||||
mutex_init(&dst_ops->local_hash.regex_lock);
|
||||
INIT_LIST_HEAD(&dst_ops->subop_list);
|
||||
dst_ops->flags |= FTRACE_OPS_FL_INITIALIZED;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
void ftrace_graph_sleep_time_control(bool enable)
|
||||
{
|
||||
fgraph_sleep_time = enable;
|
||||
@ -877,6 +888,7 @@ static int start_graph_tracing(void)
|
||||
|
||||
int register_ftrace_graph(struct fgraph_ops *gops)
|
||||
{
|
||||
int command = 0;
|
||||
int ret = 0;
|
||||
int i;
|
||||
|
||||
@ -894,7 +906,7 @@ int register_ftrace_graph(struct fgraph_ops *gops)
|
||||
break;
|
||||
}
|
||||
if (i >= FGRAPH_ARRAY_SIZE) {
|
||||
ret = -EBUSY;
|
||||
ret = -ENOSPC;
|
||||
goto out;
|
||||
}
|
||||
|
||||
@ -908,18 +920,22 @@ int register_ftrace_graph(struct fgraph_ops *gops)
|
||||
if (ftrace_graph_active == 1) {
|
||||
register_pm_notifier(&ftrace_suspend_notifier);
|
||||
ret = start_graph_tracing();
|
||||
if (ret) {
|
||||
ftrace_graph_active--;
|
||||
goto out;
|
||||
}
|
||||
if (ret)
|
||||
goto error;
|
||||
/*
|
||||
* Some archs just test to see if these are not
|
||||
* the default function
|
||||
*/
|
||||
ftrace_graph_return = return_run;
|
||||
ftrace_graph_entry = entry_run;
|
||||
command = FTRACE_START_FUNC_RET;
|
||||
}
|
||||
|
||||
ret = ftrace_startup(&graph_ops, FTRACE_START_FUNC_RET);
|
||||
ret = ftrace_startup_subops(&graph_ops, &gops->ops, command);
|
||||
error:
|
||||
if (ret) {
|
||||
fgraph_array[i] = &fgraph_stub;
|
||||
ftrace_graph_active--;
|
||||
}
|
||||
out:
|
||||
mutex_unlock(&ftrace_lock);
|
||||
@ -928,6 +944,7 @@ int register_ftrace_graph(struct fgraph_ops *gops)
|
||||
|
||||
void unregister_ftrace_graph(struct fgraph_ops *gops)
|
||||
{
|
||||
int command = 0;
|
||||
int i;
|
||||
|
||||
mutex_lock(&ftrace_lock);
|
||||
@ -935,25 +952,29 @@ void unregister_ftrace_graph(struct fgraph_ops *gops)
|
||||
if (unlikely(!ftrace_graph_active))
|
||||
goto out;
|
||||
|
||||
for (i = 0; i < fgraph_array_cnt; i++)
|
||||
if (gops == fgraph_array[i])
|
||||
break;
|
||||
if (i >= fgraph_array_cnt)
|
||||
if (unlikely(gops->idx < 0 || gops->idx >= fgraph_array_cnt))
|
||||
goto out;
|
||||
|
||||
fgraph_array[i] = &fgraph_stub;
|
||||
if (i + 1 == fgraph_array_cnt) {
|
||||
for (; i >= 0; i--)
|
||||
if (fgraph_array[i] != &fgraph_stub)
|
||||
break;
|
||||
WARN_ON_ONCE(fgraph_array[gops->idx] != gops);
|
||||
|
||||
fgraph_array[gops->idx] = &fgraph_stub;
|
||||
if (gops->idx + 1 == fgraph_array_cnt) {
|
||||
i = gops->idx;
|
||||
while (i >= 0 && fgraph_array[i] == &fgraph_stub)
|
||||
i--;
|
||||
fgraph_array_cnt = i + 1;
|
||||
}
|
||||
|
||||
ftrace_graph_active--;
|
||||
|
||||
if (!ftrace_graph_active)
|
||||
command = FTRACE_STOP_FUNC_RET;
|
||||
|
||||
ftrace_shutdown_subops(&graph_ops, &gops->ops, command);
|
||||
|
||||
if (!ftrace_graph_active) {
|
||||
ftrace_graph_return = ftrace_stub_graph;
|
||||
ftrace_graph_entry = ftrace_graph_entry_stub;
|
||||
ftrace_shutdown(&graph_ops, FTRACE_STOP_FUNC_RET);
|
||||
unregister_pm_notifier(&ftrace_suspend_notifier);
|
||||
unregister_trace_sched_switch(ftrace_graph_probe_sched_switch, NULL);
|
||||
}
|
||||
|
@ -7811,7 +7811,7 @@ __init void ftrace_init_global_array_ops(struct trace_array *tr)
|
||||
tr->ops = &global_ops;
|
||||
tr->ops->private = tr;
|
||||
ftrace_init_trace_array(tr);
|
||||
init_array_fgraph_ops(tr);
|
||||
init_array_fgraph_ops(tr, tr->ops);
|
||||
}
|
||||
|
||||
void ftrace_init_array_ops(struct trace_array *tr, ftrace_func_t func)
|
||||
|
@ -894,8 +894,8 @@ extern int __trace_graph_entry(struct trace_array *tr,
|
||||
extern void __trace_graph_return(struct trace_array *tr,
|
||||
struct ftrace_graph_ret *trace,
|
||||
unsigned int trace_ctx);
|
||||
extern void init_array_fgraph_ops(struct trace_array *tr);
|
||||
extern int allocate_fgraph_ops(struct trace_array *tr);
|
||||
extern void init_array_fgraph_ops(struct trace_array *tr, struct ftrace_ops *ops);
|
||||
extern int allocate_fgraph_ops(struct trace_array *tr, struct ftrace_ops *ops);
|
||||
extern void free_fgraph_ops(struct trace_array *tr);
|
||||
|
||||
#ifdef CONFIG_DYNAMIC_FTRACE
|
||||
@ -1003,18 +1003,19 @@ static inline bool ftrace_graph_ignore_func(struct ftrace_graph_ent *trace)
|
||||
(fgraph_max_depth && trace->depth >= fgraph_max_depth);
|
||||
}
|
||||
|
||||
void fgraph_init_ops(struct ftrace_ops *dst_ops,
|
||||
struct ftrace_ops *src_ops);
|
||||
|
||||
#else /* CONFIG_FUNCTION_GRAPH_TRACER */
|
||||
static inline enum print_line_t
|
||||
print_graph_function_flags(struct trace_iterator *iter, u32 flags)
|
||||
{
|
||||
return TRACE_TYPE_UNHANDLED;
|
||||
}
|
||||
static inline void init_array_fgraph_ops(struct trace_array *tr) { }
|
||||
static inline int allocate_fgraph_ops(struct trace_array *tr)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
static inline void free_fgraph_ops(struct trace_array *tr) { }
|
||||
/* ftrace_ops may not be defined */
|
||||
#define init_array_fgraph_ops(tr, ops) do { } while (0)
|
||||
#define allocate_fgraph_ops(tr, ops) ({ 0; })
|
||||
#endif /* CONFIG_FUNCTION_GRAPH_TRACER */
|
||||
|
||||
extern struct list_head ftrace_pids;
|
||||
|
@ -91,7 +91,7 @@ int ftrace_create_function_files(struct trace_array *tr,
|
||||
if (!tr->ops)
|
||||
return -EINVAL;
|
||||
|
||||
ret = allocate_fgraph_ops(tr);
|
||||
ret = allocate_fgraph_ops(tr, tr->ops);
|
||||
if (ret) {
|
||||
kfree(tr->ops);
|
||||
return ret;
|
||||
|
@ -288,7 +288,7 @@ static struct fgraph_ops funcgraph_ops = {
|
||||
.retfunc = &trace_graph_return,
|
||||
};
|
||||
|
||||
int allocate_fgraph_ops(struct trace_array *tr)
|
||||
int allocate_fgraph_ops(struct trace_array *tr, struct ftrace_ops *ops)
|
||||
{
|
||||
struct fgraph_ops *gops;
|
||||
|
||||
@ -301,6 +301,9 @@ int allocate_fgraph_ops(struct trace_array *tr)
|
||||
|
||||
tr->gops = gops;
|
||||
gops->private = tr;
|
||||
|
||||
fgraph_init_ops(&gops->ops, ops);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -309,10 +312,11 @@ void free_fgraph_ops(struct trace_array *tr)
|
||||
kfree(tr->gops);
|
||||
}
|
||||
|
||||
__init void init_array_fgraph_ops(struct trace_array *tr)
|
||||
__init void init_array_fgraph_ops(struct trace_array *tr, struct ftrace_ops *ops)
|
||||
{
|
||||
tr->gops = &funcgraph_ops;
|
||||
funcgraph_ops.private = tr;
|
||||
fgraph_init_ops(&tr->gops->ops, ops);
|
||||
}
|
||||
|
||||
static int graph_trace_init(struct trace_array *tr)
|
||||
|
Loading…
Reference in New Issue
Block a user