mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-09 23:39:18 +00:00
tracing: Enable creating new instance early boot
Enable creating new trace_array instance in early boot stage. If the instances directory is not created, postpone it until the tracefs is initialized. Link: https://lkml.kernel.org/r/159974154763.478751.6289753509587233103.stgit@devnote2 Signed-off-by: Masami Hiramatsu <mhiramat@kernel.org> Signed-off-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
This commit is contained in:
parent
a838deab4e
commit
4114fbfd02
@ -8636,6 +8636,24 @@ struct trace_array *trace_array_find_get(const char *instance)
|
|||||||
return tr;
|
return tr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int trace_array_create_dir(struct trace_array *tr)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
tr->dir = tracefs_create_dir(tr->name, trace_instance_dir);
|
||||||
|
if (!tr->dir)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
ret = event_trace_add_tracer(tr->dir, tr);
|
||||||
|
if (ret)
|
||||||
|
tracefs_remove(tr->dir);
|
||||||
|
|
||||||
|
init_tracer_tracefs(tr, tr->dir);
|
||||||
|
__update_tracer_options(tr);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static struct trace_array *trace_array_create(const char *name)
|
static struct trace_array *trace_array_create(const char *name)
|
||||||
{
|
{
|
||||||
struct trace_array *tr;
|
struct trace_array *tr;
|
||||||
@ -8671,30 +8689,27 @@ static struct trace_array *trace_array_create(const char *name)
|
|||||||
if (allocate_trace_buffers(tr, trace_buf_size) < 0)
|
if (allocate_trace_buffers(tr, trace_buf_size) < 0)
|
||||||
goto out_free_tr;
|
goto out_free_tr;
|
||||||
|
|
||||||
tr->dir = tracefs_create_dir(name, trace_instance_dir);
|
if (ftrace_allocate_ftrace_ops(tr) < 0)
|
||||||
if (!tr->dir)
|
|
||||||
goto out_free_tr;
|
goto out_free_tr;
|
||||||
|
|
||||||
ret = event_trace_add_tracer(tr->dir, tr);
|
|
||||||
if (ret) {
|
|
||||||
tracefs_remove(tr->dir);
|
|
||||||
goto out_free_tr;
|
|
||||||
}
|
|
||||||
|
|
||||||
ftrace_init_trace_array(tr);
|
ftrace_init_trace_array(tr);
|
||||||
|
|
||||||
init_tracer_tracefs(tr, tr->dir);
|
|
||||||
init_trace_flags_index(tr);
|
init_trace_flags_index(tr);
|
||||||
__update_tracer_options(tr);
|
|
||||||
|
if (trace_instance_dir) {
|
||||||
|
ret = trace_array_create_dir(tr);
|
||||||
|
if (ret)
|
||||||
|
goto out_free_tr;
|
||||||
|
}
|
||||||
|
|
||||||
list_add(&tr->list, &ftrace_trace_arrays);
|
list_add(&tr->list, &ftrace_trace_arrays);
|
||||||
|
|
||||||
tr->ref++;
|
tr->ref++;
|
||||||
|
|
||||||
|
|
||||||
return tr;
|
return tr;
|
||||||
|
|
||||||
out_free_tr:
|
out_free_tr:
|
||||||
|
ftrace_free_ftrace_ops(tr);
|
||||||
free_trace_buffers(tr);
|
free_trace_buffers(tr);
|
||||||
free_cpumask_var(tr->tracing_cpumask);
|
free_cpumask_var(tr->tracing_cpumask);
|
||||||
kfree(tr->name);
|
kfree(tr->name);
|
||||||
@ -8852,11 +8867,27 @@ static int instance_rmdir(const char *name)
|
|||||||
|
|
||||||
static __init void create_trace_instances(struct dentry *d_tracer)
|
static __init void create_trace_instances(struct dentry *d_tracer)
|
||||||
{
|
{
|
||||||
|
struct trace_array *tr;
|
||||||
|
|
||||||
trace_instance_dir = tracefs_create_instance_dir("instances", d_tracer,
|
trace_instance_dir = tracefs_create_instance_dir("instances", d_tracer,
|
||||||
instance_mkdir,
|
instance_mkdir,
|
||||||
instance_rmdir);
|
instance_rmdir);
|
||||||
if (MEM_FAIL(!trace_instance_dir, "Failed to create instances directory\n"))
|
if (MEM_FAIL(!trace_instance_dir, "Failed to create instances directory\n"))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
mutex_lock(&event_mutex);
|
||||||
|
mutex_lock(&trace_types_lock);
|
||||||
|
|
||||||
|
list_for_each_entry(tr, &ftrace_trace_arrays, list) {
|
||||||
|
if (!tr->name)
|
||||||
|
continue;
|
||||||
|
if (MEM_FAIL(trace_array_create_dir(tr) < 0,
|
||||||
|
"Failed to create instance directory\n"))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
mutex_unlock(&trace_types_lock);
|
||||||
|
mutex_unlock(&event_mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -1125,6 +1125,8 @@ extern int ftrace_is_dead(void);
|
|||||||
int ftrace_create_function_files(struct trace_array *tr,
|
int ftrace_create_function_files(struct trace_array *tr,
|
||||||
struct dentry *parent);
|
struct dentry *parent);
|
||||||
void ftrace_destroy_function_files(struct trace_array *tr);
|
void ftrace_destroy_function_files(struct trace_array *tr);
|
||||||
|
int ftrace_allocate_ftrace_ops(struct trace_array *tr);
|
||||||
|
void ftrace_free_ftrace_ops(struct trace_array *tr);
|
||||||
void ftrace_init_global_array_ops(struct trace_array *tr);
|
void ftrace_init_global_array_ops(struct trace_array *tr);
|
||||||
void ftrace_init_array_ops(struct trace_array *tr, ftrace_func_t func);
|
void ftrace_init_array_ops(struct trace_array *tr, ftrace_func_t func);
|
||||||
void ftrace_reset_array_ops(struct trace_array *tr);
|
void ftrace_reset_array_ops(struct trace_array *tr);
|
||||||
@ -1146,6 +1148,11 @@ ftrace_create_function_files(struct trace_array *tr,
|
|||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
static inline int ftrace_allocate_ftrace_ops(struct trace_array *tr)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
static inline void ftrace_free_ftrace_ops(struct trace_array *tr) { }
|
||||||
static inline void ftrace_destroy_function_files(struct trace_array *tr) { }
|
static inline void ftrace_destroy_function_files(struct trace_array *tr) { }
|
||||||
static inline __init void
|
static inline __init void
|
||||||
ftrace_init_global_array_ops(struct trace_array *tr) { }
|
ftrace_init_global_array_ops(struct trace_array *tr) { }
|
||||||
|
@ -34,10 +34,14 @@ enum {
|
|||||||
TRACE_FUNC_OPT_STACK = 0x1,
|
TRACE_FUNC_OPT_STACK = 0x1,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int allocate_ftrace_ops(struct trace_array *tr)
|
int ftrace_allocate_ftrace_ops(struct trace_array *tr)
|
||||||
{
|
{
|
||||||
struct ftrace_ops *ops;
|
struct ftrace_ops *ops;
|
||||||
|
|
||||||
|
/* The top level array uses the "global_ops" */
|
||||||
|
if (tr->flags & TRACE_ARRAY_FL_GLOBAL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
ops = kzalloc(sizeof(*ops), GFP_KERNEL);
|
ops = kzalloc(sizeof(*ops), GFP_KERNEL);
|
||||||
if (!ops)
|
if (!ops)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
@ -48,15 +52,19 @@ static int allocate_ftrace_ops(struct trace_array *tr)
|
|||||||
|
|
||||||
tr->ops = ops;
|
tr->ops = ops;
|
||||||
ops->private = tr;
|
ops->private = tr;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ftrace_free_ftrace_ops(struct trace_array *tr)
|
||||||
|
{
|
||||||
|
kfree(tr->ops);
|
||||||
|
tr->ops = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
int ftrace_create_function_files(struct trace_array *tr,
|
int ftrace_create_function_files(struct trace_array *tr,
|
||||||
struct dentry *parent)
|
struct dentry *parent)
|
||||||
{
|
{
|
||||||
int ret;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The top level array uses the "global_ops", and the files are
|
* The top level array uses the "global_ops", and the files are
|
||||||
* created on boot up.
|
* created on boot up.
|
||||||
@ -64,9 +72,8 @@ int ftrace_create_function_files(struct trace_array *tr,
|
|||||||
if (tr->flags & TRACE_ARRAY_FL_GLOBAL)
|
if (tr->flags & TRACE_ARRAY_FL_GLOBAL)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
ret = allocate_ftrace_ops(tr);
|
if (!tr->ops)
|
||||||
if (ret)
|
return -EINVAL;
|
||||||
return ret;
|
|
||||||
|
|
||||||
ftrace_create_filter_files(tr->ops, parent);
|
ftrace_create_filter_files(tr->ops, parent);
|
||||||
|
|
||||||
@ -76,8 +83,7 @@ int ftrace_create_function_files(struct trace_array *tr,
|
|||||||
void ftrace_destroy_function_files(struct trace_array *tr)
|
void ftrace_destroy_function_files(struct trace_array *tr)
|
||||||
{
|
{
|
||||||
ftrace_destroy_filter_files(tr->ops);
|
ftrace_destroy_filter_files(tr->ops);
|
||||||
kfree(tr->ops);
|
ftrace_free_ftrace_ops(tr);
|
||||||
tr->ops = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int function_trace_init(struct trace_array *tr)
|
static int function_trace_init(struct trace_array *tr)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user