diff options
Diffstat (limited to 'kernel/trace/ftrace.c')
| -rw-r--r-- | kernel/trace/ftrace.c | 502 | 
1 files changed, 81 insertions, 421 deletions
| diff --git a/kernel/trace/ftrace.c b/kernel/trace/ftrace.c index 0feea145bb29..764668467155 100644 --- a/kernel/trace/ftrace.c +++ b/kernel/trace/ftrace.c @@ -45,6 +45,10 @@  #include "trace_output.h"  #include "trace_stat.h" +/* Flags that do not get reset */ +#define FTRACE_NOCLEAR_FLAGS	(FTRACE_FL_DISABLED | FTRACE_FL_TOUCHED | \ +				 FTRACE_FL_MODIFIED) +  #define FTRACE_INVALID_FUNCTION		"__ftrace_invalid_address__"  #define FTRACE_WARN_ON(cond)			\ @@ -2256,7 +2260,7 @@ static int ftrace_check_record(struct dyn_ftrace *rec, bool enable, bool update)  		flag ^= rec->flags & FTRACE_FL_ENABLED;  		if (update) { -			rec->flags |= FTRACE_FL_ENABLED; +			rec->flags |= FTRACE_FL_ENABLED | FTRACE_FL_TOUCHED;  			if (flag & FTRACE_FL_REGS) {  				if (rec->flags & FTRACE_FL_REGS)  					rec->flags |= FTRACE_FL_REGS_EN; @@ -2270,6 +2274,10 @@ static int ftrace_check_record(struct dyn_ftrace *rec, bool enable, bool update)  					rec->flags &= ~FTRACE_FL_TRAMP_EN;  			} +			/* Keep track of anything that modifies the function */ +			if (rec->flags & (FTRACE_FL_DIRECT | FTRACE_FL_IPMODIFY)) +				rec->flags |= FTRACE_FL_MODIFIED; +  			if (flag & FTRACE_FL_DIRECT) {  				/*  				 * If there's only one user (direct_ops helper) @@ -2326,7 +2334,7 @@ static int ftrace_check_record(struct dyn_ftrace *rec, bool enable, bool update)  	if (update) {  		/* If there's no more users, clear all flags */  		if (!ftrace_rec_count(rec)) -			rec->flags &= FTRACE_FL_DISABLED; +			rec->flags &= FTRACE_NOCLEAR_FLAGS;  		else  			/*  			 * Just disable the record, but keep the ops TRAMP @@ -2583,28 +2591,13 @@ ftrace_add_rec_direct(unsigned long ip, unsigned long addr,  static void call_direct_funcs(unsigned long ip, unsigned long pip,  			      struct ftrace_ops *ops, struct ftrace_regs *fregs)  { -	unsigned long addr; +	unsigned long addr = READ_ONCE(ops->direct_call); -	addr = ftrace_find_rec_direct(ip);  	if (!addr)  		return;  	arch_ftrace_set_direct_caller(fregs, addr);  } - -static struct ftrace_ops direct_ops = { -	.func		= call_direct_funcs, -	.flags		= FTRACE_OPS_FL_DIRECT | FTRACE_OPS_FL_SAVE_REGS -			  | FTRACE_OPS_FL_PERMANENT, -	/* -	 * By declaring the main trampoline as this trampoline -	 * it will never have one allocated for it. Allocated -	 * trampolines should not call direct functions. -	 * The direct_ops should only be called by the builtin -	 * ftrace_regs_caller trampoline. -	 */ -	.trampoline	= FTRACE_REGS_ADDR, -};  #endif /* CONFIG_DYNAMIC_FTRACE_WITH_DIRECT_CALLS */  /** @@ -3162,7 +3155,7 @@ int ftrace_shutdown(struct ftrace_ops *ops, int command)  		struct dyn_ftrace *rec;  		do_for_each_ftrace_rec(pg, rec) { -			if (FTRACE_WARN_ON_ONCE(rec->flags & ~FTRACE_FL_DISABLED)) +			if (FTRACE_WARN_ON_ONCE(rec->flags & ~FTRACE_NOCLEAR_FLAGS))  				pr_warn("  %pS flags:%lx\n",  					(void *)rec->ip, rec->flags);  		} while_for_each_ftrace_rec(); @@ -3613,7 +3606,10 @@ t_func_next(struct seq_file *m, loff_t *pos)  		     !ftrace_lookup_ip(iter->hash, rec->ip)) ||  		    ((iter->flags & FTRACE_ITER_ENABLED) && -		     !(rec->flags & FTRACE_FL_ENABLED))) { +		     !(rec->flags & FTRACE_FL_ENABLED)) || + +		    ((iter->flags & FTRACE_ITER_TOUCHED) && +		     !(rec->flags & FTRACE_FL_TOUCHED))) {  			rec = NULL;  			goto retry; @@ -3872,15 +3868,16 @@ static int t_show(struct seq_file *m, void *v)  		return 0;  	} -	if (iter->flags & FTRACE_ITER_ENABLED) { +	if (iter->flags & (FTRACE_ITER_ENABLED | FTRACE_ITER_TOUCHED)) {  		struct ftrace_ops *ops; -		seq_printf(m, " (%ld)%s%s%s%s", +		seq_printf(m, " (%ld)%s%s%s%s%s",  			   ftrace_rec_count(rec),  			   rec->flags & FTRACE_FL_REGS ? " R" : "  ",  			   rec->flags & FTRACE_FL_IPMODIFY ? " I" : "  ",  			   rec->flags & FTRACE_FL_DIRECT ? " D" : "  ", -			   rec->flags & FTRACE_FL_CALL_OPS ? " O" : "  "); +			   rec->flags & FTRACE_FL_CALL_OPS ? " O" : "  ", +			   rec->flags & FTRACE_FL_MODIFIED ? " M " : "   ");  		if (rec->flags & FTRACE_FL_TRAMP_EN) {  			ops = ftrace_find_tramp_ops_any(rec);  			if (ops) { @@ -3974,6 +3971,31 @@ ftrace_enabled_open(struct inode *inode, struct file *file)  	return 0;  } +static int +ftrace_touched_open(struct inode *inode, struct file *file) +{ +	struct ftrace_iterator *iter; + +	/* +	 * This shows us what functions have ever been enabled +	 * (traced, direct, patched, etc). Not sure if we want lockdown +	 * to hide such critical information for an admin. +	 * Although, perhaps it can show information we don't +	 * want people to see, but if something had traced +	 * something, we probably want to know about it. +	 */ + +	iter = __seq_open_private(file, &show_ftrace_seq_ops, sizeof(*iter)); +	if (!iter) +		return -ENOMEM; + +	iter->pg = ftrace_pages_start; +	iter->flags = FTRACE_ITER_TOUCHED; +	iter->ops = &global_ops; + +	return 0; +} +  /**   * ftrace_regex_open - initialize function tracer filter files   * @ops: The ftrace_ops that hold the hash filters @@ -5301,388 +5323,9 @@ struct ftrace_direct_func {  static LIST_HEAD(ftrace_direct_funcs); -/** - * ftrace_find_direct_func - test an address if it is a registered direct caller - * @addr: The address of a registered direct caller - * - * This searches to see if a ftrace direct caller has been registered - * at a specific address, and if so, it returns a descriptor for it. - * - * This can be used by architecture code to see if an address is - * a direct caller (trampoline) attached to a fentry/mcount location. - * This is useful for the function_graph tracer, as it may need to - * do adjustments if it traced a location that also has a direct - * trampoline attached to it. - */ -struct ftrace_direct_func *ftrace_find_direct_func(unsigned long addr) -{ -	struct ftrace_direct_func *entry; -	bool found = false; - -	/* May be called by fgraph trampoline (protected by rcu tasks) */ -	list_for_each_entry_rcu(entry, &ftrace_direct_funcs, next) { -		if (entry->addr == addr) { -			found = true; -			break; -		} -	} -	if (found) -		return entry; - -	return NULL; -} - -static struct ftrace_direct_func *ftrace_alloc_direct_func(unsigned long addr) -{ -	struct ftrace_direct_func *direct; - -	direct = kmalloc(sizeof(*direct), GFP_KERNEL); -	if (!direct) -		return NULL; -	direct->addr = addr; -	direct->count = 0; -	list_add_rcu(&direct->next, &ftrace_direct_funcs); -	ftrace_direct_func_count++; -	return direct; -} -  static int register_ftrace_function_nolock(struct ftrace_ops *ops); -/** - * register_ftrace_direct - Call a custom trampoline directly - * @ip: The address of the nop at the beginning of a function - * @addr: The address of the trampoline to call at @ip - * - * This is used to connect a direct call from the nop location (@ip) - * at the start of ftrace traced functions. The location that it calls - * (@addr) must be able to handle a direct call, and save the parameters - * of the function being traced, and restore them (or inject new ones - * if needed), before returning. - * - * Returns: - *  0 on success - *  -EBUSY - Another direct function is already attached (there can be only one) - *  -ENODEV - @ip does not point to a ftrace nop location (or not supported) - *  -ENOMEM - There was an allocation failure. - */ -int register_ftrace_direct(unsigned long ip, unsigned long addr) -{ -	struct ftrace_direct_func *direct; -	struct ftrace_func_entry *entry; -	struct ftrace_hash *free_hash = NULL; -	struct dyn_ftrace *rec; -	int ret = -ENODEV; - -	mutex_lock(&direct_mutex); - -	ip = ftrace_location(ip); -	if (!ip) -		goto out_unlock; - -	/* See if there's a direct function at @ip already */ -	ret = -EBUSY; -	if (ftrace_find_rec_direct(ip)) -		goto out_unlock; - -	ret = -ENODEV; -	rec = lookup_rec(ip, ip); -	if (!rec) -		goto out_unlock; - -	/* -	 * Check if the rec says it has a direct call but we didn't -	 * find one earlier? -	 */ -	if (WARN_ON(rec->flags & FTRACE_FL_DIRECT)) -		goto out_unlock; - -	/* Make sure the ip points to the exact record */ -	if (ip != rec->ip) { -		ip = rec->ip; -		/* Need to check this ip for a direct. */ -		if (ftrace_find_rec_direct(ip)) -			goto out_unlock; -	} - -	ret = -ENOMEM; -	direct = ftrace_find_direct_func(addr); -	if (!direct) { -		direct = ftrace_alloc_direct_func(addr); -		if (!direct) -			goto out_unlock; -	} - -	entry = ftrace_add_rec_direct(ip, addr, &free_hash); -	if (!entry) -		goto out_unlock; - -	ret = ftrace_set_filter_ip(&direct_ops, ip, 0, 0); - -	if (!ret && !(direct_ops.flags & FTRACE_OPS_FL_ENABLED)) { -		ret = register_ftrace_function_nolock(&direct_ops); -		if (ret) -			ftrace_set_filter_ip(&direct_ops, ip, 1, 0); -	} - -	if (ret) { -		remove_hash_entry(direct_functions, entry); -		kfree(entry); -		if (!direct->count) { -			list_del_rcu(&direct->next); -			synchronize_rcu_tasks(); -			kfree(direct); -			if (free_hash) -				free_ftrace_hash(free_hash); -			free_hash = NULL; -			ftrace_direct_func_count--; -		} -	} else { -		direct->count++; -	} - out_unlock: -	mutex_unlock(&direct_mutex); - -	if (free_hash) { -		synchronize_rcu_tasks(); -		free_ftrace_hash(free_hash); -	} - -	return ret; -} -EXPORT_SYMBOL_GPL(register_ftrace_direct); - -static struct ftrace_func_entry *find_direct_entry(unsigned long *ip, -						   struct dyn_ftrace **recp) -{ -	struct ftrace_func_entry *entry; -	struct dyn_ftrace *rec; - -	rec = lookup_rec(*ip, *ip); -	if (!rec) -		return NULL; - -	entry = __ftrace_lookup_ip(direct_functions, rec->ip); -	if (!entry) { -		WARN_ON(rec->flags & FTRACE_FL_DIRECT); -		return NULL; -	} - -	WARN_ON(!(rec->flags & FTRACE_FL_DIRECT)); - -	/* Passed in ip just needs to be on the call site */ -	*ip = rec->ip; - -	if (recp) -		*recp = rec; - -	return entry; -} - -int unregister_ftrace_direct(unsigned long ip, unsigned long addr) -{ -	struct ftrace_direct_func *direct; -	struct ftrace_func_entry *entry; -	struct ftrace_hash *hash; -	int ret = -ENODEV; - -	mutex_lock(&direct_mutex); - -	ip = ftrace_location(ip); -	if (!ip) -		goto out_unlock; - -	entry = find_direct_entry(&ip, NULL); -	if (!entry) -		goto out_unlock; - -	hash = direct_ops.func_hash->filter_hash; -	if (hash->count == 1) -		unregister_ftrace_function(&direct_ops); - -	ret = ftrace_set_filter_ip(&direct_ops, ip, 1, 0); - -	WARN_ON(ret); - -	remove_hash_entry(direct_functions, entry); - -	direct = ftrace_find_direct_func(addr); -	if (!WARN_ON(!direct)) { -		/* This is the good path (see the ! before WARN) */ -		direct->count--; -		WARN_ON(direct->count < 0); -		if (!direct->count) { -			list_del_rcu(&direct->next); -			synchronize_rcu_tasks(); -			kfree(direct); -			kfree(entry); -			ftrace_direct_func_count--; -		} -	} - out_unlock: -	mutex_unlock(&direct_mutex); - -	return ret; -} -EXPORT_SYMBOL_GPL(unregister_ftrace_direct); - -static struct ftrace_ops stub_ops = { -	.func		= ftrace_stub, -}; - -/** - * ftrace_modify_direct_caller - modify ftrace nop directly - * @entry: The ftrace hash entry of the direct helper for @rec - * @rec: The record representing the function site to patch - * @old_addr: The location that the site at @rec->ip currently calls - * @new_addr: The location that the site at @rec->ip should call - * - * An architecture may overwrite this function to optimize the - * changing of the direct callback on an ftrace nop location. - * This is called with the ftrace_lock mutex held, and no other - * ftrace callbacks are on the associated record (@rec). Thus, - * it is safe to modify the ftrace record, where it should be - * currently calling @old_addr directly, to call @new_addr. - * - * This is called with direct_mutex locked. - * - * Safety checks should be made to make sure that the code at - * @rec->ip is currently calling @old_addr. And this must - * also update entry->direct to @new_addr. - */ -int __weak ftrace_modify_direct_caller(struct ftrace_func_entry *entry, -				       struct dyn_ftrace *rec, -				       unsigned long old_addr, -				       unsigned long new_addr) -{ -	unsigned long ip = rec->ip; -	int ret; - -	lockdep_assert_held(&direct_mutex); - -	/* -	 * The ftrace_lock was used to determine if the record -	 * had more than one registered user to it. If it did, -	 * we needed to prevent that from changing to do the quick -	 * switch. But if it did not (only a direct caller was attached) -	 * then this function is called. But this function can deal -	 * with attached callers to the rec that we care about, and -	 * since this function uses standard ftrace calls that take -	 * the ftrace_lock mutex, we need to release it. -	 */ -	mutex_unlock(&ftrace_lock); - -	/* -	 * By setting a stub function at the same address, we force -	 * the code to call the iterator and the direct_ops helper. -	 * This means that @ip does not call the direct call, and -	 * we can simply modify it. -	 */ -	ret = ftrace_set_filter_ip(&stub_ops, ip, 0, 0); -	if (ret) -		goto out_lock; - -	ret = register_ftrace_function_nolock(&stub_ops); -	if (ret) { -		ftrace_set_filter_ip(&stub_ops, ip, 1, 0); -		goto out_lock; -	} - -	entry->direct = new_addr; - -	/* -	 * By removing the stub, we put back the direct call, calling -	 * the @new_addr. -	 */ -	unregister_ftrace_function(&stub_ops); -	ftrace_set_filter_ip(&stub_ops, ip, 1, 0); - - out_lock: -	mutex_lock(&ftrace_lock); - -	return ret; -} - -/** - * modify_ftrace_direct - Modify an existing direct call to call something else - * @ip: The instruction pointer to modify - * @old_addr: The address that the current @ip calls directly - * @new_addr: The address that the @ip should call - * - * This modifies a ftrace direct caller at an instruction pointer without - * having to disable it first. The direct call will switch over to the - * @new_addr without missing anything. - * - * Returns: zero on success. Non zero on error, which includes: - *  -ENODEV : the @ip given has no direct caller attached - *  -EINVAL : the @old_addr does not match the current direct caller - */ -int modify_ftrace_direct(unsigned long ip, -			 unsigned long old_addr, unsigned long new_addr) -{ -	struct ftrace_direct_func *direct, *new_direct = NULL; -	struct ftrace_func_entry *entry; -	struct dyn_ftrace *rec; -	int ret = -ENODEV; - -	mutex_lock(&direct_mutex); - -	mutex_lock(&ftrace_lock); - -	ip = ftrace_location(ip); -	if (!ip) -		goto out_unlock; - -	entry = find_direct_entry(&ip, &rec); -	if (!entry) -		goto out_unlock; - -	ret = -EINVAL; -	if (entry->direct != old_addr) -		goto out_unlock; - -	direct = ftrace_find_direct_func(old_addr); -	if (WARN_ON(!direct)) -		goto out_unlock; -	if (direct->count > 1) { -		ret = -ENOMEM; -		new_direct = ftrace_alloc_direct_func(new_addr); -		if (!new_direct) -			goto out_unlock; -		direct->count--; -		new_direct->count++; -	} else { -		direct->addr = new_addr; -	} - -	/* -	 * If there's no other ftrace callback on the rec->ip location, -	 * then it can be changed directly by the architecture. -	 * If there is another caller, then we just need to change the -	 * direct caller helper to point to @new_addr. -	 */ -	if (ftrace_rec_count(rec) == 1) { -		ret = ftrace_modify_direct_caller(entry, rec, old_addr, new_addr); -	} else { -		entry->direct = new_addr; -		ret = 0; -	} - -	if (unlikely(ret && new_direct)) { -		direct->count++; -		list_del_rcu(&new_direct->next); -		synchronize_rcu_tasks(); -		kfree(new_direct); -		ftrace_direct_func_count--; -	} - - out_unlock: -	mutex_unlock(&ftrace_lock); -	mutex_unlock(&direct_mutex); -	return ret; -} -EXPORT_SYMBOL_GPL(modify_ftrace_direct); - -#define MULTI_FLAGS (FTRACE_OPS_FL_DIRECT | FTRACE_OPS_FL_SAVE_REGS) +#define MULTI_FLAGS (FTRACE_OPS_FL_DIRECT | FTRACE_OPS_FL_SAVE_ARGS)  static int check_direct_multi(struct ftrace_ops *ops)  { @@ -5711,7 +5354,7 @@ static void remove_direct_functions_hash(struct ftrace_hash *hash, unsigned long  }  /** - * register_ftrace_direct_multi - Call a custom trampoline directly + * register_ftrace_direct - Call a custom trampoline directly   * for multiple functions registered in @ops   * @ops: The address of the struct ftrace_ops object   * @addr: The address of the trampoline to call at @ops functions @@ -5732,7 +5375,7 @@ static void remove_direct_functions_hash(struct ftrace_hash *hash, unsigned long   *  -ENODEV  - @ip does not point to a ftrace nop location (or not supported)   *  -ENOMEM  - There was an allocation failure.   */ -int register_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr) +int register_ftrace_direct(struct ftrace_ops *ops, unsigned long addr)  {  	struct ftrace_hash *hash, *free_hash = NULL;  	struct ftrace_func_entry *entry, *new; @@ -5774,6 +5417,7 @@ int register_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  	ops->func = call_direct_funcs;  	ops->flags = MULTI_FLAGS;  	ops->trampoline = FTRACE_REGS_ADDR; +	ops->direct_call = addr;  	err = register_ftrace_function_nolock(ops); @@ -5790,11 +5434,11 @@ int register_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  	}  	return err;  } -EXPORT_SYMBOL_GPL(register_ftrace_direct_multi); +EXPORT_SYMBOL_GPL(register_ftrace_direct);  /** - * unregister_ftrace_direct_multi - Remove calls to custom trampoline - * previously registered by register_ftrace_direct_multi for @ops object. + * unregister_ftrace_direct - Remove calls to custom trampoline + * previously registered by register_ftrace_direct for @ops object.   * @ops: The address of the struct ftrace_ops object   *   * This is used to remove a direct calls to @addr from the nop locations @@ -5805,7 +5449,8 @@ EXPORT_SYMBOL_GPL(register_ftrace_direct_multi);   *  0 on success   *  -EINVAL - The @ops object was not properly registered.   */ -int unregister_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr) +int unregister_ftrace_direct(struct ftrace_ops *ops, unsigned long addr, +			     bool free_filters)  {  	struct ftrace_hash *hash = ops->func_hash->filter_hash;  	int err; @@ -5823,12 +5468,15 @@ int unregister_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  	/* cleanup for possible another register call */  	ops->func = NULL;  	ops->trampoline = 0; + +	if (free_filters) +		ftrace_free_filter(ops);  	return err;  } -EXPORT_SYMBOL_GPL(unregister_ftrace_direct_multi); +EXPORT_SYMBOL_GPL(unregister_ftrace_direct);  static int -__modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr) +__modify_ftrace_direct(struct ftrace_ops *ops, unsigned long addr)  {  	struct ftrace_hash *hash;  	struct ftrace_func_entry *entry, *iter; @@ -5844,6 +5492,7 @@ __modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  	/* Enable the tmp_ops to have the same functions as the direct ops */  	ftrace_ops_init(&tmp_ops);  	tmp_ops.func_hash = ops->func_hash; +	tmp_ops.direct_call = addr;  	err = register_ftrace_function_nolock(&tmp_ops);  	if (err) @@ -5865,6 +5514,8 @@ __modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  			entry->direct = addr;  		}  	} +	/* Prevent store tearing if a trampoline concurrently accesses the value */ +	WRITE_ONCE(ops->direct_call, addr);  	mutex_unlock(&ftrace_lock); @@ -5875,7 +5526,7 @@ __modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  }  /** - * modify_ftrace_direct_multi_nolock - Modify an existing direct 'multi' call + * modify_ftrace_direct_nolock - Modify an existing direct 'multi' call   * to call something else   * @ops: The address of the struct ftrace_ops object   * @addr: The address of the new trampoline to call at @ops functions @@ -5892,19 +5543,19 @@ __modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)   * Returns: zero on success. Non zero on error, which includes:   *  -EINVAL - The @ops object was not properly registered.   */ -int modify_ftrace_direct_multi_nolock(struct ftrace_ops *ops, unsigned long addr) +int modify_ftrace_direct_nolock(struct ftrace_ops *ops, unsigned long addr)  {  	if (check_direct_multi(ops))  		return -EINVAL;  	if (!(ops->flags & FTRACE_OPS_FL_ENABLED))  		return -EINVAL; -	return __modify_ftrace_direct_multi(ops, addr); +	return __modify_ftrace_direct(ops, addr);  } -EXPORT_SYMBOL_GPL(modify_ftrace_direct_multi_nolock); +EXPORT_SYMBOL_GPL(modify_ftrace_direct_nolock);  /** - * modify_ftrace_direct_multi - Modify an existing direct 'multi' call + * modify_ftrace_direct - Modify an existing direct 'multi' call   * to call something else   * @ops: The address of the struct ftrace_ops object   * @addr: The address of the new trampoline to call at @ops functions @@ -5918,7 +5569,7 @@ EXPORT_SYMBOL_GPL(modify_ftrace_direct_multi_nolock);   * Returns: zero on success. Non zero on error, which includes:   *  -EINVAL - The @ops object was not properly registered.   */ -int modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr) +int modify_ftrace_direct(struct ftrace_ops *ops, unsigned long addr)  {  	int err; @@ -5928,11 +5579,11 @@ int modify_ftrace_direct_multi(struct ftrace_ops *ops, unsigned long addr)  		return -EINVAL;  	mutex_lock(&direct_mutex); -	err = __modify_ftrace_direct_multi(ops, addr); +	err = __modify_ftrace_direct(ops, addr);  	mutex_unlock(&direct_mutex);  	return err;  } -EXPORT_SYMBOL_GPL(modify_ftrace_direct_multi); +EXPORT_SYMBOL_GPL(modify_ftrace_direct);  #endif /* CONFIG_DYNAMIC_FTRACE_WITH_DIRECT_CALLS */  /** @@ -6258,6 +5909,13 @@ static const struct file_operations ftrace_enabled_fops = {  	.release = seq_release_private,  }; +static const struct file_operations ftrace_touched_fops = { +	.open = ftrace_touched_open, +	.read = seq_read, +	.llseek = seq_lseek, +	.release = seq_release_private, +}; +  static const struct file_operations ftrace_filter_fops = {  	.open = ftrace_filter_open,  	.read = seq_read, @@ -6722,6 +6380,9 @@ static __init int ftrace_init_dyn_tracefs(struct dentry *d_tracer)  	trace_create_file("enabled_functions", TRACE_MODE_READ,  			d_tracer, NULL, &ftrace_enabled_fops); +	trace_create_file("touched_functions", TRACE_MODE_READ, +			d_tracer, NULL, &ftrace_touched_fops); +  	ftrace_create_filter_files(&global_ops, d_tracer);  #ifdef CONFIG_FUNCTION_GRAPH_TRACER @@ -8392,8 +8053,7 @@ struct kallsyms_data {   * and returns 1 in case we resolved all the requested symbols,   * 0 otherwise.   */ -static int kallsyms_callback(void *data, const char *name, -			     struct module *mod, unsigned long addr) +static int kallsyms_callback(void *data, const char *name, unsigned long addr)  {  	struct kallsyms_data *args = data;  	const char **sym; |