xref: /openbmc/linux/kernel/trace/ftrace.c (revision 9f4801e30ad291e27284e873696da1ead92d68fa)
116444a8aSArnaldo Carvalho de Melo /*
216444a8aSArnaldo Carvalho de Melo  * Infrastructure for profiling code inserted by 'gcc -pg'.
316444a8aSArnaldo Carvalho de Melo  *
416444a8aSArnaldo Carvalho de Melo  * Copyright (C) 2007-2008 Steven Rostedt <srostedt@redhat.com>
516444a8aSArnaldo Carvalho de Melo  * Copyright (C) 2004-2008 Ingo Molnar <mingo@redhat.com>
616444a8aSArnaldo Carvalho de Melo  *
716444a8aSArnaldo Carvalho de Melo  * Originally ported from the -rt patch by:
816444a8aSArnaldo Carvalho de Melo  *   Copyright (C) 2007 Arnaldo Carvalho de Melo <acme@redhat.com>
916444a8aSArnaldo Carvalho de Melo  *
1016444a8aSArnaldo Carvalho de Melo  * Based on code in the latency_tracer, that is:
1116444a8aSArnaldo Carvalho de Melo  *
1216444a8aSArnaldo Carvalho de Melo  *  Copyright (C) 2004-2006 Ingo Molnar
1316444a8aSArnaldo Carvalho de Melo  *  Copyright (C) 2004 William Lee Irwin III
1416444a8aSArnaldo Carvalho de Melo  */
1516444a8aSArnaldo Carvalho de Melo 
163d083395SSteven Rostedt #include <linux/stop_machine.h>
173d083395SSteven Rostedt #include <linux/clocksource.h>
183d083395SSteven Rostedt #include <linux/kallsyms.h>
195072c59fSSteven Rostedt #include <linux/seq_file.h>
204a2b8ddaSFrederic Weisbecker #include <linux/suspend.h>
215072c59fSSteven Rostedt #include <linux/debugfs.h>
223d083395SSteven Rostedt #include <linux/hardirq.h>
232d8b820bSIngo Molnar #include <linux/kthread.h>
245072c59fSSteven Rostedt #include <linux/uaccess.h>
25f22f9a89SAbhishek Sagar #include <linux/kprobes.h>
262d8b820bSIngo Molnar #include <linux/ftrace.h>
27b0fc494fSSteven Rostedt #include <linux/sysctl.h>
285072c59fSSteven Rostedt #include <linux/ctype.h>
293d083395SSteven Rostedt #include <linux/list.h>
3016444a8aSArnaldo Carvalho de Melo 
31395a59d0SAbhishek Sagar #include <asm/ftrace.h>
32395a59d0SAbhishek Sagar 
333d083395SSteven Rostedt #include "trace.h"
343d083395SSteven Rostedt 
356912896eSSteven Rostedt #define FTRACE_WARN_ON(cond)			\
366912896eSSteven Rostedt 	do {					\
376912896eSSteven Rostedt 		if (WARN_ON(cond))		\
386912896eSSteven Rostedt 			ftrace_kill();		\
396912896eSSteven Rostedt 	} while (0)
406912896eSSteven Rostedt 
416912896eSSteven Rostedt #define FTRACE_WARN_ON_ONCE(cond)		\
426912896eSSteven Rostedt 	do {					\
436912896eSSteven Rostedt 		if (WARN_ON_ONCE(cond))		\
446912896eSSteven Rostedt 			ftrace_kill();		\
456912896eSSteven Rostedt 	} while (0)
466912896eSSteven Rostedt 
474eebcc81SSteven Rostedt /* ftrace_enabled is a method to turn ftrace on or off */
484eebcc81SSteven Rostedt int ftrace_enabled __read_mostly;
49d61f82d0SSteven Rostedt static int last_ftrace_enabled;
50b0fc494fSSteven Rostedt 
510ef8cde5SSteven Rostedt /* set when tracing only a pid */
52978f3a45SSteven Rostedt struct pid *ftrace_pid_trace;
5321bbecdaSSteven Rostedt static struct pid * const ftrace_swapper_pid = &init_struct_pid;
54df4fc315SSteven Rostedt 
5560a7ecf4SSteven Rostedt /* Quick disabling of function tracer. */
5660a7ecf4SSteven Rostedt int function_trace_stop;
5760a7ecf4SSteven Rostedt 
584eebcc81SSteven Rostedt /*
594eebcc81SSteven Rostedt  * ftrace_disabled is set when an anomaly is discovered.
604eebcc81SSteven Rostedt  * ftrace_disabled is much stronger than ftrace_enabled.
614eebcc81SSteven Rostedt  */
624eebcc81SSteven Rostedt static int ftrace_disabled __read_mostly;
634eebcc81SSteven Rostedt 
643d083395SSteven Rostedt static DEFINE_SPINLOCK(ftrace_lock);
65b0fc494fSSteven Rostedt static DEFINE_MUTEX(ftrace_sysctl_lock);
66df4fc315SSteven Rostedt static DEFINE_MUTEX(ftrace_start_lock);
67b0fc494fSSteven Rostedt 
6816444a8aSArnaldo Carvalho de Melo static struct ftrace_ops ftrace_list_end __read_mostly =
6916444a8aSArnaldo Carvalho de Melo {
7016444a8aSArnaldo Carvalho de Melo 	.func = ftrace_stub,
7116444a8aSArnaldo Carvalho de Melo };
7216444a8aSArnaldo Carvalho de Melo 
7316444a8aSArnaldo Carvalho de Melo static struct ftrace_ops *ftrace_list __read_mostly = &ftrace_list_end;
7416444a8aSArnaldo Carvalho de Melo ftrace_func_t ftrace_trace_function __read_mostly = ftrace_stub;
7560a7ecf4SSteven Rostedt ftrace_func_t __ftrace_trace_function __read_mostly = ftrace_stub;
76df4fc315SSteven Rostedt ftrace_func_t ftrace_pid_function __read_mostly = ftrace_stub;
7716444a8aSArnaldo Carvalho de Melo 
78f2252935SIngo Molnar static void ftrace_list_func(unsigned long ip, unsigned long parent_ip)
7916444a8aSArnaldo Carvalho de Melo {
8016444a8aSArnaldo Carvalho de Melo 	struct ftrace_ops *op = ftrace_list;
8116444a8aSArnaldo Carvalho de Melo 
8216444a8aSArnaldo Carvalho de Melo 	/* in case someone actually ports this to alpha! */
8316444a8aSArnaldo Carvalho de Melo 	read_barrier_depends();
8416444a8aSArnaldo Carvalho de Melo 
8516444a8aSArnaldo Carvalho de Melo 	while (op != &ftrace_list_end) {
8616444a8aSArnaldo Carvalho de Melo 		/* silly alpha */
8716444a8aSArnaldo Carvalho de Melo 		read_barrier_depends();
8816444a8aSArnaldo Carvalho de Melo 		op->func(ip, parent_ip);
8916444a8aSArnaldo Carvalho de Melo 		op = op->next;
9016444a8aSArnaldo Carvalho de Melo 	};
9116444a8aSArnaldo Carvalho de Melo }
9216444a8aSArnaldo Carvalho de Melo 
93df4fc315SSteven Rostedt static void ftrace_pid_func(unsigned long ip, unsigned long parent_ip)
94df4fc315SSteven Rostedt {
950ef8cde5SSteven Rostedt 	if (!test_tsk_trace_trace(current))
96df4fc315SSteven Rostedt 		return;
97df4fc315SSteven Rostedt 
98df4fc315SSteven Rostedt 	ftrace_pid_function(ip, parent_ip);
99df4fc315SSteven Rostedt }
100df4fc315SSteven Rostedt 
101df4fc315SSteven Rostedt static void set_ftrace_pid_function(ftrace_func_t func)
102df4fc315SSteven Rostedt {
103df4fc315SSteven Rostedt 	/* do not set ftrace_pid_function to itself! */
104df4fc315SSteven Rostedt 	if (func != ftrace_pid_func)
105df4fc315SSteven Rostedt 		ftrace_pid_function = func;
106df4fc315SSteven Rostedt }
107df4fc315SSteven Rostedt 
10816444a8aSArnaldo Carvalho de Melo /**
1093d083395SSteven Rostedt  * clear_ftrace_function - reset the ftrace function
11016444a8aSArnaldo Carvalho de Melo  *
1113d083395SSteven Rostedt  * This NULLs the ftrace function and in essence stops
1123d083395SSteven Rostedt  * tracing.  There may be lag
11316444a8aSArnaldo Carvalho de Melo  */
1143d083395SSteven Rostedt void clear_ftrace_function(void)
11516444a8aSArnaldo Carvalho de Melo {
1163d083395SSteven Rostedt 	ftrace_trace_function = ftrace_stub;
11760a7ecf4SSteven Rostedt 	__ftrace_trace_function = ftrace_stub;
118df4fc315SSteven Rostedt 	ftrace_pid_function = ftrace_stub;
1193d083395SSteven Rostedt }
12016444a8aSArnaldo Carvalho de Melo 
12160a7ecf4SSteven Rostedt #ifndef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
12260a7ecf4SSteven Rostedt /*
12360a7ecf4SSteven Rostedt  * For those archs that do not test ftrace_trace_stop in their
12460a7ecf4SSteven Rostedt  * mcount call site, we need to do it from C.
12560a7ecf4SSteven Rostedt  */
12660a7ecf4SSteven Rostedt static void ftrace_test_stop_func(unsigned long ip, unsigned long parent_ip)
12760a7ecf4SSteven Rostedt {
12860a7ecf4SSteven Rostedt 	if (function_trace_stop)
12960a7ecf4SSteven Rostedt 		return;
13060a7ecf4SSteven Rostedt 
13160a7ecf4SSteven Rostedt 	__ftrace_trace_function(ip, parent_ip);
13260a7ecf4SSteven Rostedt }
13360a7ecf4SSteven Rostedt #endif
13460a7ecf4SSteven Rostedt 
135e309b41dSIngo Molnar static int __register_ftrace_function(struct ftrace_ops *ops)
1363d083395SSteven Rostedt {
13799ecdc43SSteven Rostedt 	/* should not be called from interrupt context */
1383d083395SSteven Rostedt 	spin_lock(&ftrace_lock);
1393d083395SSteven Rostedt 
14016444a8aSArnaldo Carvalho de Melo 	ops->next = ftrace_list;
14116444a8aSArnaldo Carvalho de Melo 	/*
14216444a8aSArnaldo Carvalho de Melo 	 * We are entering ops into the ftrace_list but another
14316444a8aSArnaldo Carvalho de Melo 	 * CPU might be walking that list. We need to make sure
14416444a8aSArnaldo Carvalho de Melo 	 * the ops->next pointer is valid before another CPU sees
14516444a8aSArnaldo Carvalho de Melo 	 * the ops pointer included into the ftrace_list.
14616444a8aSArnaldo Carvalho de Melo 	 */
14716444a8aSArnaldo Carvalho de Melo 	smp_wmb();
14816444a8aSArnaldo Carvalho de Melo 	ftrace_list = ops;
1493d083395SSteven Rostedt 
150b0fc494fSSteven Rostedt 	if (ftrace_enabled) {
151df4fc315SSteven Rostedt 		ftrace_func_t func;
152df4fc315SSteven Rostedt 
153df4fc315SSteven Rostedt 		if (ops->next == &ftrace_list_end)
154df4fc315SSteven Rostedt 			func = ops->func;
155df4fc315SSteven Rostedt 		else
156df4fc315SSteven Rostedt 			func = ftrace_list_func;
157df4fc315SSteven Rostedt 
158978f3a45SSteven Rostedt 		if (ftrace_pid_trace) {
159df4fc315SSteven Rostedt 			set_ftrace_pid_function(func);
160df4fc315SSteven Rostedt 			func = ftrace_pid_func;
161df4fc315SSteven Rostedt 		}
162df4fc315SSteven Rostedt 
16316444a8aSArnaldo Carvalho de Melo 		/*
16416444a8aSArnaldo Carvalho de Melo 		 * For one func, simply call it directly.
16516444a8aSArnaldo Carvalho de Melo 		 * For more than one func, call the chain.
16616444a8aSArnaldo Carvalho de Melo 		 */
16760a7ecf4SSteven Rostedt #ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
168df4fc315SSteven Rostedt 		ftrace_trace_function = func;
16960a7ecf4SSteven Rostedt #else
170df4fc315SSteven Rostedt 		__ftrace_trace_function = func;
17160a7ecf4SSteven Rostedt 		ftrace_trace_function = ftrace_test_stop_func;
17260a7ecf4SSteven Rostedt #endif
173b0fc494fSSteven Rostedt 	}
1743d083395SSteven Rostedt 
1753d083395SSteven Rostedt 	spin_unlock(&ftrace_lock);
17616444a8aSArnaldo Carvalho de Melo 
17716444a8aSArnaldo Carvalho de Melo 	return 0;
17816444a8aSArnaldo Carvalho de Melo }
17916444a8aSArnaldo Carvalho de Melo 
180e309b41dSIngo Molnar static int __unregister_ftrace_function(struct ftrace_ops *ops)
18116444a8aSArnaldo Carvalho de Melo {
18216444a8aSArnaldo Carvalho de Melo 	struct ftrace_ops **p;
18316444a8aSArnaldo Carvalho de Melo 	int ret = 0;
18416444a8aSArnaldo Carvalho de Melo 
18599ecdc43SSteven Rostedt 	/* should not be called from interrupt context */
1863d083395SSteven Rostedt 	spin_lock(&ftrace_lock);
18716444a8aSArnaldo Carvalho de Melo 
18816444a8aSArnaldo Carvalho de Melo 	/*
1893d083395SSteven Rostedt 	 * If we are removing the last function, then simply point
1903d083395SSteven Rostedt 	 * to the ftrace_stub.
19116444a8aSArnaldo Carvalho de Melo 	 */
19216444a8aSArnaldo Carvalho de Melo 	if (ftrace_list == ops && ops->next == &ftrace_list_end) {
19316444a8aSArnaldo Carvalho de Melo 		ftrace_trace_function = ftrace_stub;
19416444a8aSArnaldo Carvalho de Melo 		ftrace_list = &ftrace_list_end;
19516444a8aSArnaldo Carvalho de Melo 		goto out;
19616444a8aSArnaldo Carvalho de Melo 	}
19716444a8aSArnaldo Carvalho de Melo 
19816444a8aSArnaldo Carvalho de Melo 	for (p = &ftrace_list; *p != &ftrace_list_end; p = &(*p)->next)
19916444a8aSArnaldo Carvalho de Melo 		if (*p == ops)
20016444a8aSArnaldo Carvalho de Melo 			break;
20116444a8aSArnaldo Carvalho de Melo 
20216444a8aSArnaldo Carvalho de Melo 	if (*p != ops) {
20316444a8aSArnaldo Carvalho de Melo 		ret = -1;
20416444a8aSArnaldo Carvalho de Melo 		goto out;
20516444a8aSArnaldo Carvalho de Melo 	}
20616444a8aSArnaldo Carvalho de Melo 
20716444a8aSArnaldo Carvalho de Melo 	*p = (*p)->next;
20816444a8aSArnaldo Carvalho de Melo 
209b0fc494fSSteven Rostedt 	if (ftrace_enabled) {
21016444a8aSArnaldo Carvalho de Melo 		/* If we only have one func left, then call that directly */
211df4fc315SSteven Rostedt 		if (ftrace_list->next == &ftrace_list_end) {
212df4fc315SSteven Rostedt 			ftrace_func_t func = ftrace_list->func;
213df4fc315SSteven Rostedt 
214978f3a45SSteven Rostedt 			if (ftrace_pid_trace) {
215df4fc315SSteven Rostedt 				set_ftrace_pid_function(func);
216df4fc315SSteven Rostedt 				func = ftrace_pid_func;
217df4fc315SSteven Rostedt 			}
218df4fc315SSteven Rostedt #ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
219df4fc315SSteven Rostedt 			ftrace_trace_function = func;
220df4fc315SSteven Rostedt #else
221df4fc315SSteven Rostedt 			__ftrace_trace_function = func;
222df4fc315SSteven Rostedt #endif
223df4fc315SSteven Rostedt 		}
224b0fc494fSSteven Rostedt 	}
22516444a8aSArnaldo Carvalho de Melo 
22616444a8aSArnaldo Carvalho de Melo  out:
2273d083395SSteven Rostedt 	spin_unlock(&ftrace_lock);
2283d083395SSteven Rostedt 
2293d083395SSteven Rostedt 	return ret;
2303d083395SSteven Rostedt }
2313d083395SSteven Rostedt 
232df4fc315SSteven Rostedt static void ftrace_update_pid_func(void)
233df4fc315SSteven Rostedt {
234df4fc315SSteven Rostedt 	ftrace_func_t func;
235df4fc315SSteven Rostedt 
236df4fc315SSteven Rostedt 	/* should not be called from interrupt context */
237df4fc315SSteven Rostedt 	spin_lock(&ftrace_lock);
238df4fc315SSteven Rostedt 
239df4fc315SSteven Rostedt 	if (ftrace_trace_function == ftrace_stub)
240df4fc315SSteven Rostedt 		goto out;
241df4fc315SSteven Rostedt 
242df4fc315SSteven Rostedt 	func = ftrace_trace_function;
243df4fc315SSteven Rostedt 
244978f3a45SSteven Rostedt 	if (ftrace_pid_trace) {
245df4fc315SSteven Rostedt 		set_ftrace_pid_function(func);
246df4fc315SSteven Rostedt 		func = ftrace_pid_func;
247df4fc315SSteven Rostedt 	} else {
24866eafebcSLiming Wang 		if (func == ftrace_pid_func)
24966eafebcSLiming Wang 			func = ftrace_pid_function;
250df4fc315SSteven Rostedt 	}
251df4fc315SSteven Rostedt 
252df4fc315SSteven Rostedt #ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
253df4fc315SSteven Rostedt 	ftrace_trace_function = func;
254df4fc315SSteven Rostedt #else
255df4fc315SSteven Rostedt 	__ftrace_trace_function = func;
256df4fc315SSteven Rostedt #endif
257df4fc315SSteven Rostedt 
258df4fc315SSteven Rostedt  out:
259df4fc315SSteven Rostedt 	spin_unlock(&ftrace_lock);
260df4fc315SSteven Rostedt }
261df4fc315SSteven Rostedt 
2623d083395SSteven Rostedt #ifdef CONFIG_DYNAMIC_FTRACE
26399ecdc43SSteven Rostedt #ifndef CONFIG_FTRACE_MCOUNT_RECORD
264cb7be3b2SSteven Rostedt # error Dynamic ftrace depends on MCOUNT_RECORD
26599ecdc43SSteven Rostedt #endif
26699ecdc43SSteven Rostedt 
267d61f82d0SSteven Rostedt enum {
268d61f82d0SSteven Rostedt 	FTRACE_ENABLE_CALLS		= (1 << 0),
269d61f82d0SSteven Rostedt 	FTRACE_DISABLE_CALLS		= (1 << 1),
270d61f82d0SSteven Rostedt 	FTRACE_UPDATE_TRACE_FUNC	= (1 << 2),
271d61f82d0SSteven Rostedt 	FTRACE_ENABLE_MCOUNT		= (1 << 3),
272d61f82d0SSteven Rostedt 	FTRACE_DISABLE_MCOUNT		= (1 << 4),
2735a45cfe1SSteven Rostedt 	FTRACE_START_FUNC_RET		= (1 << 5),
2745a45cfe1SSteven Rostedt 	FTRACE_STOP_FUNC_RET		= (1 << 6),
275d61f82d0SSteven Rostedt };
276d61f82d0SSteven Rostedt 
2775072c59fSSteven Rostedt static int ftrace_filtered;
2785072c59fSSteven Rostedt 
27908f5ac90SSteven Rostedt static LIST_HEAD(ftrace_new_addrs);
2803d083395SSteven Rostedt 
28141c52c0dSSteven Rostedt static DEFINE_MUTEX(ftrace_regex_lock);
2823d083395SSteven Rostedt 
2833c1720f0SSteven Rostedt struct ftrace_page {
2843c1720f0SSteven Rostedt 	struct ftrace_page	*next;
285431aa3fbSSteven Rostedt 	int			index;
2863c1720f0SSteven Rostedt 	struct dyn_ftrace	records[];
287aa5e5ceaSDavid Miller };
2883c1720f0SSteven Rostedt 
2893c1720f0SSteven Rostedt #define ENTRIES_PER_PAGE \
2903c1720f0SSteven Rostedt   ((PAGE_SIZE - sizeof(struct ftrace_page)) / sizeof(struct dyn_ftrace))
2913c1720f0SSteven Rostedt 
2923c1720f0SSteven Rostedt /* estimate from running different kernels */
2933c1720f0SSteven Rostedt #define NR_TO_INIT		10000
2943c1720f0SSteven Rostedt 
2953c1720f0SSteven Rostedt static struct ftrace_page	*ftrace_pages_start;
2963c1720f0SSteven Rostedt static struct ftrace_page	*ftrace_pages;
2973c1720f0SSteven Rostedt 
29837ad5084SSteven Rostedt static struct dyn_ftrace *ftrace_free_records;
29937ad5084SSteven Rostedt 
300265c831cSSteven Rostedt /*
301265c831cSSteven Rostedt  * This is a double for. Do not use 'break' to break out of the loop,
302265c831cSSteven Rostedt  * you must use a goto.
303265c831cSSteven Rostedt  */
304265c831cSSteven Rostedt #define do_for_each_ftrace_rec(pg, rec)					\
305265c831cSSteven Rostedt 	for (pg = ftrace_pages_start; pg; pg = pg->next) {		\
306265c831cSSteven Rostedt 		int _____i;						\
307265c831cSSteven Rostedt 		for (_____i = 0; _____i < pg->index; _____i++) {	\
308265c831cSSteven Rostedt 			rec = &pg->records[_____i];
309265c831cSSteven Rostedt 
310265c831cSSteven Rostedt #define while_for_each_ftrace_rec()		\
311265c831cSSteven Rostedt 		}				\
312265c831cSSteven Rostedt 	}
313ecea656dSAbhishek Sagar 
314ecea656dSAbhishek Sagar #ifdef CONFIG_KPROBES
315f17845e5SIngo Molnar 
316f17845e5SIngo Molnar static int frozen_record_count;
317f17845e5SIngo Molnar 
318ecea656dSAbhishek Sagar static inline void freeze_record(struct dyn_ftrace *rec)
319ecea656dSAbhishek Sagar {
320ecea656dSAbhishek Sagar 	if (!(rec->flags & FTRACE_FL_FROZEN)) {
321ecea656dSAbhishek Sagar 		rec->flags |= FTRACE_FL_FROZEN;
322ecea656dSAbhishek Sagar 		frozen_record_count++;
323ecea656dSAbhishek Sagar 	}
324ecea656dSAbhishek Sagar }
325ecea656dSAbhishek Sagar 
326ecea656dSAbhishek Sagar static inline void unfreeze_record(struct dyn_ftrace *rec)
327ecea656dSAbhishek Sagar {
328ecea656dSAbhishek Sagar 	if (rec->flags & FTRACE_FL_FROZEN) {
329ecea656dSAbhishek Sagar 		rec->flags &= ~FTRACE_FL_FROZEN;
330ecea656dSAbhishek Sagar 		frozen_record_count--;
331ecea656dSAbhishek Sagar 	}
332ecea656dSAbhishek Sagar }
333ecea656dSAbhishek Sagar 
334ecea656dSAbhishek Sagar static inline int record_frozen(struct dyn_ftrace *rec)
335ecea656dSAbhishek Sagar {
336ecea656dSAbhishek Sagar 	return rec->flags & FTRACE_FL_FROZEN;
337ecea656dSAbhishek Sagar }
338ecea656dSAbhishek Sagar #else
339ecea656dSAbhishek Sagar # define freeze_record(rec)			({ 0; })
340ecea656dSAbhishek Sagar # define unfreeze_record(rec)			({ 0; })
341ecea656dSAbhishek Sagar # define record_frozen(rec)			({ 0; })
342ecea656dSAbhishek Sagar #endif /* CONFIG_KPROBES */
343ecea656dSAbhishek Sagar 
344e309b41dSIngo Molnar static void ftrace_free_rec(struct dyn_ftrace *rec)
34537ad5084SSteven Rostedt {
34637ad5084SSteven Rostedt 	rec->ip = (unsigned long)ftrace_free_records;
34737ad5084SSteven Rostedt 	ftrace_free_records = rec;
34837ad5084SSteven Rostedt 	rec->flags |= FTRACE_FL_FREE;
34937ad5084SSteven Rostedt }
35037ad5084SSteven Rostedt 
351fed1939cSSteven Rostedt void ftrace_release(void *start, unsigned long size)
352fed1939cSSteven Rostedt {
353fed1939cSSteven Rostedt 	struct dyn_ftrace *rec;
354fed1939cSSteven Rostedt 	struct ftrace_page *pg;
355fed1939cSSteven Rostedt 	unsigned long s = (unsigned long)start;
356fed1939cSSteven Rostedt 	unsigned long e = s + size;
357fed1939cSSteven Rostedt 
35800fd61aeSSteven Rostedt 	if (ftrace_disabled || !start)
359fed1939cSSteven Rostedt 		return;
360fed1939cSSteven Rostedt 
36199ecdc43SSteven Rostedt 	/* should not be called from interrupt context */
362fed1939cSSteven Rostedt 	spin_lock(&ftrace_lock);
363fed1939cSSteven Rostedt 
364265c831cSSteven Rostedt 	do_for_each_ftrace_rec(pg, rec) {
365fed1939cSSteven Rostedt 		if ((rec->ip >= s) && (rec->ip < e))
366fed1939cSSteven Rostedt 			ftrace_free_rec(rec);
367265c831cSSteven Rostedt 	} while_for_each_ftrace_rec();
368265c831cSSteven Rostedt 
369fed1939cSSteven Rostedt 	spin_unlock(&ftrace_lock);
370fed1939cSSteven Rostedt }
371fed1939cSSteven Rostedt 
372e309b41dSIngo Molnar static struct dyn_ftrace *ftrace_alloc_dyn_node(unsigned long ip)
3733c1720f0SSteven Rostedt {
37437ad5084SSteven Rostedt 	struct dyn_ftrace *rec;
37537ad5084SSteven Rostedt 
37637ad5084SSteven Rostedt 	/* First check for freed records */
37737ad5084SSteven Rostedt 	if (ftrace_free_records) {
37837ad5084SSteven Rostedt 		rec = ftrace_free_records;
37937ad5084SSteven Rostedt 
38037ad5084SSteven Rostedt 		if (unlikely(!(rec->flags & FTRACE_FL_FREE))) {
3816912896eSSteven Rostedt 			FTRACE_WARN_ON_ONCE(1);
38237ad5084SSteven Rostedt 			ftrace_free_records = NULL;
38337ad5084SSteven Rostedt 			return NULL;
38437ad5084SSteven Rostedt 		}
38537ad5084SSteven Rostedt 
38637ad5084SSteven Rostedt 		ftrace_free_records = (void *)rec->ip;
38737ad5084SSteven Rostedt 		memset(rec, 0, sizeof(*rec));
38837ad5084SSteven Rostedt 		return rec;
38937ad5084SSteven Rostedt 	}
39037ad5084SSteven Rostedt 
3913c1720f0SSteven Rostedt 	if (ftrace_pages->index == ENTRIES_PER_PAGE) {
39208f5ac90SSteven Rostedt 		if (!ftrace_pages->next) {
39308f5ac90SSteven Rostedt 			/* allocate another page */
39408f5ac90SSteven Rostedt 			ftrace_pages->next =
39508f5ac90SSteven Rostedt 				(void *)get_zeroed_page(GFP_KERNEL);
3963c1720f0SSteven Rostedt 			if (!ftrace_pages->next)
3973c1720f0SSteven Rostedt 				return NULL;
39808f5ac90SSteven Rostedt 		}
3993c1720f0SSteven Rostedt 		ftrace_pages = ftrace_pages->next;
4003c1720f0SSteven Rostedt 	}
4013c1720f0SSteven Rostedt 
4023c1720f0SSteven Rostedt 	return &ftrace_pages->records[ftrace_pages->index++];
4033c1720f0SSteven Rostedt }
4043c1720f0SSteven Rostedt 
40508f5ac90SSteven Rostedt static struct dyn_ftrace *
406d61f82d0SSteven Rostedt ftrace_record_ip(unsigned long ip)
4073d083395SSteven Rostedt {
40808f5ac90SSteven Rostedt 	struct dyn_ftrace *rec;
4093d083395SSteven Rostedt 
410f3c7ac40SSteven Rostedt 	if (ftrace_disabled)
41108f5ac90SSteven Rostedt 		return NULL;
412d61f82d0SSteven Rostedt 
41308f5ac90SSteven Rostedt 	rec = ftrace_alloc_dyn_node(ip);
41408f5ac90SSteven Rostedt 	if (!rec)
41508f5ac90SSteven Rostedt 		return NULL;
4163d083395SSteven Rostedt 
41708f5ac90SSteven Rostedt 	rec->ip = ip;
4183d083395SSteven Rostedt 
41908f5ac90SSteven Rostedt 	list_add(&rec->list, &ftrace_new_addrs);
4203d083395SSteven Rostedt 
42108f5ac90SSteven Rostedt 	return rec;
4223d083395SSteven Rostedt }
4233d083395SSteven Rostedt 
42405736a42SSteven Rostedt static void print_ip_ins(const char *fmt, unsigned char *p)
42505736a42SSteven Rostedt {
42605736a42SSteven Rostedt 	int i;
42705736a42SSteven Rostedt 
42805736a42SSteven Rostedt 	printk(KERN_CONT "%s", fmt);
42905736a42SSteven Rostedt 
43005736a42SSteven Rostedt 	for (i = 0; i < MCOUNT_INSN_SIZE; i++)
43105736a42SSteven Rostedt 		printk(KERN_CONT "%s%02x", i ? ":" : "", p[i]);
43205736a42SSteven Rostedt }
43305736a42SSteven Rostedt 
43431e88909SSteven Rostedt static void ftrace_bug(int failed, unsigned long ip)
4353c1720f0SSteven Rostedt {
436b17e8a37SSteven Rostedt 	switch (failed) {
437593eb8a2SSteven Rostedt 	case -EFAULT:
4386912896eSSteven Rostedt 		FTRACE_WARN_ON_ONCE(1);
43905736a42SSteven Rostedt 		pr_info("ftrace faulted on modifying ");
44005736a42SSteven Rostedt 		print_ip_sym(ip);
44105736a42SSteven Rostedt 		break;
442593eb8a2SSteven Rostedt 	case -EINVAL:
4436912896eSSteven Rostedt 		FTRACE_WARN_ON_ONCE(1);
44405736a42SSteven Rostedt 		pr_info("ftrace failed to modify ");
44505736a42SSteven Rostedt 		print_ip_sym(ip);
44605736a42SSteven Rostedt 		print_ip_ins(" actual: ", (unsigned char *)ip);
44705736a42SSteven Rostedt 		printk(KERN_CONT "\n");
44805736a42SSteven Rostedt 		break;
449593eb8a2SSteven Rostedt 	case -EPERM:
4506912896eSSteven Rostedt 		FTRACE_WARN_ON_ONCE(1);
451593eb8a2SSteven Rostedt 		pr_info("ftrace faulted on writing ");
452593eb8a2SSteven Rostedt 		print_ip_sym(ip);
453593eb8a2SSteven Rostedt 		break;
454593eb8a2SSteven Rostedt 	default:
4556912896eSSteven Rostedt 		FTRACE_WARN_ON_ONCE(1);
456593eb8a2SSteven Rostedt 		pr_info("ftrace faulted on unknown error ");
457593eb8a2SSteven Rostedt 		print_ip_sym(ip);
45805736a42SSteven Rostedt 	}
459b17e8a37SSteven Rostedt }
46005736a42SSteven Rostedt 
4613d083395SSteven Rostedt 
4623d083395SSteven Rostedt static int
46331e88909SSteven Rostedt __ftrace_replace_code(struct dyn_ftrace *rec, int enable)
4643d083395SSteven Rostedt {
4653d083395SSteven Rostedt 	unsigned long ip, fl;
466e7d3737eSFrederic Weisbecker 	unsigned long ftrace_addr;
467e7d3737eSFrederic Weisbecker 
468f0001207SShaohua Li 	ftrace_addr = (unsigned long)FTRACE_ADDR;
4693d083395SSteven Rostedt 
4703d083395SSteven Rostedt 	ip = rec->ip;
4713d083395SSteven Rostedt 
4723d083395SSteven Rostedt 	/*
473982c350bSSteven Rostedt 	 * If this record is not to be traced and
474982c350bSSteven Rostedt 	 * it is not enabled then do nothing.
4753d083395SSteven Rostedt 	 *
476982c350bSSteven Rostedt 	 * If this record is not to be traced and
47757794a9dSWenji Huang 	 * it is enabled then disable it.
4783d083395SSteven Rostedt 	 *
4793d083395SSteven Rostedt 	 */
480982c350bSSteven Rostedt 	if (rec->flags & FTRACE_FL_NOTRACE) {
481982c350bSSteven Rostedt 		if (rec->flags & FTRACE_FL_ENABLED)
4823d083395SSteven Rostedt 			rec->flags &= ~FTRACE_FL_ENABLED;
483982c350bSSteven Rostedt 		else
4843d083395SSteven Rostedt 			return 0;
485982c350bSSteven Rostedt 
486982c350bSSteven Rostedt 	} else if (ftrace_filtered && enable) {
487982c350bSSteven Rostedt 		/*
488982c350bSSteven Rostedt 		 * Filtering is on:
489982c350bSSteven Rostedt 		 */
490982c350bSSteven Rostedt 
491982c350bSSteven Rostedt 		fl = rec->flags & (FTRACE_FL_FILTER | FTRACE_FL_ENABLED);
492982c350bSSteven Rostedt 
493982c350bSSteven Rostedt 		/* Record is filtered and enabled, do nothing */
494982c350bSSteven Rostedt 		if (fl == (FTRACE_FL_FILTER | FTRACE_FL_ENABLED))
495982c350bSSteven Rostedt 			return 0;
496982c350bSSteven Rostedt 
49757794a9dSWenji Huang 		/* Record is not filtered or enabled, do nothing */
498982c350bSSteven Rostedt 		if (!fl)
499982c350bSSteven Rostedt 			return 0;
500982c350bSSteven Rostedt 
501982c350bSSteven Rostedt 		/* Record is not filtered but enabled, disable it */
502982c350bSSteven Rostedt 		if (fl == FTRACE_FL_ENABLED)
503982c350bSSteven Rostedt 			rec->flags &= ~FTRACE_FL_ENABLED;
504982c350bSSteven Rostedt 		else
505982c350bSSteven Rostedt 		/* Otherwise record is filtered but not enabled, enable it */
506982c350bSSteven Rostedt 			rec->flags |= FTRACE_FL_ENABLED;
507982c350bSSteven Rostedt 	} else {
508982c350bSSteven Rostedt 		/* Disable or not filtered */
5093d083395SSteven Rostedt 
5103d083395SSteven Rostedt 		if (enable) {
511982c350bSSteven Rostedt 			/* if record is enabled, do nothing */
5123d083395SSteven Rostedt 			if (rec->flags & FTRACE_FL_ENABLED)
5133d083395SSteven Rostedt 				return 0;
514982c350bSSteven Rostedt 
5153d083395SSteven Rostedt 			rec->flags |= FTRACE_FL_ENABLED;
516982c350bSSteven Rostedt 
5173d083395SSteven Rostedt 		} else {
518982c350bSSteven Rostedt 
51957794a9dSWenji Huang 			/* if record is not enabled, do nothing */
5203d083395SSteven Rostedt 			if (!(rec->flags & FTRACE_FL_ENABLED))
5213d083395SSteven Rostedt 				return 0;
522982c350bSSteven Rostedt 
5233d083395SSteven Rostedt 			rec->flags &= ~FTRACE_FL_ENABLED;
5243d083395SSteven Rostedt 		}
5253d083395SSteven Rostedt 	}
5263d083395SSteven Rostedt 
527982c350bSSteven Rostedt 	if (rec->flags & FTRACE_FL_ENABLED)
528e7d3737eSFrederic Weisbecker 		return ftrace_make_call(rec, ftrace_addr);
52931e88909SSteven Rostedt 	else
530e7d3737eSFrederic Weisbecker 		return ftrace_make_nop(NULL, rec, ftrace_addr);
5313d083395SSteven Rostedt }
5323d083395SSteven Rostedt 
5333d083395SSteven Rostedt static void ftrace_replace_code(int enable)
5343d083395SSteven Rostedt {
535265c831cSSteven Rostedt 	int failed;
5363d083395SSteven Rostedt 	struct dyn_ftrace *rec;
5373d083395SSteven Rostedt 	struct ftrace_page *pg;
5383d083395SSteven Rostedt 
539265c831cSSteven Rostedt 	do_for_each_ftrace_rec(pg, rec) {
540918c1154SSteven Rostedt 		/*
541918c1154SSteven Rostedt 		 * Skip over free records and records that have
542918c1154SSteven Rostedt 		 * failed.
543918c1154SSteven Rostedt 		 */
544918c1154SSteven Rostedt 		if (rec->flags & FTRACE_FL_FREE ||
545918c1154SSteven Rostedt 		    rec->flags & FTRACE_FL_FAILED)
5463d083395SSteven Rostedt 			continue;
5473d083395SSteven Rostedt 
5483d083395SSteven Rostedt 		/* ignore updates to this record's mcount site */
5493d083395SSteven Rostedt 		if (get_kprobe((void *)rec->ip)) {
5503d083395SSteven Rostedt 			freeze_record(rec);
5513d083395SSteven Rostedt 			continue;
5523d083395SSteven Rostedt 		} else {
5533d083395SSteven Rostedt 			unfreeze_record(rec);
5543d083395SSteven Rostedt 		}
5553d083395SSteven Rostedt 
55631e88909SSteven Rostedt 		failed = __ftrace_replace_code(rec, enable);
5573d083395SSteven Rostedt 		if (failed && (rec->flags & FTRACE_FL_CONVERTED)) {
5583d083395SSteven Rostedt 			rec->flags |= FTRACE_FL_FAILED;
5593d083395SSteven Rostedt 			if ((system_state == SYSTEM_BOOTING) ||
5603d083395SSteven Rostedt 			    !core_kernel_text(rec->ip)) {
5613d083395SSteven Rostedt 				ftrace_free_rec(rec);
562b17e8a37SSteven Rostedt 			} else
56331e88909SSteven Rostedt 				ftrace_bug(failed, rec->ip);
5643d083395SSteven Rostedt 		}
565265c831cSSteven Rostedt 	} while_for_each_ftrace_rec();
5663d083395SSteven Rostedt }
5673c1720f0SSteven Rostedt 
5683b47bfc1SSteven Rostedt static int
56931e88909SSteven Rostedt ftrace_code_disable(struct module *mod, struct dyn_ftrace *rec)
5703c1720f0SSteven Rostedt {
57137ad5084SSteven Rostedt 	unsigned long ip;
572492a7ea5SAbhishek Sagar 	int ret;
57337ad5084SSteven Rostedt 
574492a7ea5SAbhishek Sagar 	ip = rec->ip;
5753c1720f0SSteven Rostedt 
57625aac9dcSShaohua Li 	ret = ftrace_make_nop(mod, rec, MCOUNT_ADDR);
5773d083395SSteven Rostedt 	if (ret) {
57831e88909SSteven Rostedt 		ftrace_bug(ret, ip);
5793d083395SSteven Rostedt 		rec->flags |= FTRACE_FL_FAILED;
5803d083395SSteven Rostedt 		return 0;
581d61f82d0SSteven Rostedt 	}
582d61f82d0SSteven Rostedt 	return 1;
5833d083395SSteven Rostedt }
5843d083395SSteven Rostedt 
5853d083395SSteven Rostedt static int __ftrace_modify_code(void *data)
5863d083395SSteven Rostedt {
5873d083395SSteven Rostedt 	int *command = data;
5883d083395SSteven Rostedt 
589a3583244SSteven Rostedt 	if (*command & FTRACE_ENABLE_CALLS)
5903d083395SSteven Rostedt 		ftrace_replace_code(1);
591a3583244SSteven Rostedt 	else if (*command & FTRACE_DISABLE_CALLS)
5923d083395SSteven Rostedt 		ftrace_replace_code(0);
5933d083395SSteven Rostedt 
5943d083395SSteven Rostedt 	if (*command & FTRACE_UPDATE_TRACE_FUNC)
5953d083395SSteven Rostedt 		ftrace_update_ftrace_func(ftrace_trace_function);
5963d083395SSteven Rostedt 
5975a45cfe1SSteven Rostedt 	if (*command & FTRACE_START_FUNC_RET)
5985a45cfe1SSteven Rostedt 		ftrace_enable_ftrace_graph_caller();
5995a45cfe1SSteven Rostedt 	else if (*command & FTRACE_STOP_FUNC_RET)
6005a45cfe1SSteven Rostedt 		ftrace_disable_ftrace_graph_caller();
6015a45cfe1SSteven Rostedt 
6023d083395SSteven Rostedt 	return 0;
6033d083395SSteven Rostedt }
6043d083395SSteven Rostedt 
605e309b41dSIngo Molnar static void ftrace_run_update_code(int command)
6063d083395SSteven Rostedt {
607784e2d76SRusty Russell 	stop_machine(__ftrace_modify_code, &command, NULL);
6083d083395SSteven Rostedt }
6093d083395SSteven Rostedt 
6103d083395SSteven Rostedt static ftrace_func_t saved_ftrace_func;
61160a7ecf4SSteven Rostedt static int ftrace_start_up;
612df4fc315SSteven Rostedt 
613df4fc315SSteven Rostedt static void ftrace_startup_enable(int command)
614df4fc315SSteven Rostedt {
615df4fc315SSteven Rostedt 	if (saved_ftrace_func != ftrace_trace_function) {
616df4fc315SSteven Rostedt 		saved_ftrace_func = ftrace_trace_function;
617df4fc315SSteven Rostedt 		command |= FTRACE_UPDATE_TRACE_FUNC;
618df4fc315SSteven Rostedt 	}
619df4fc315SSteven Rostedt 
620df4fc315SSteven Rostedt 	if (!command || !ftrace_enabled)
621df4fc315SSteven Rostedt 		return;
622df4fc315SSteven Rostedt 
623df4fc315SSteven Rostedt 	ftrace_run_update_code(command);
624df4fc315SSteven Rostedt }
6253d083395SSteven Rostedt 
6265a45cfe1SSteven Rostedt static void ftrace_startup(int command)
6273d083395SSteven Rostedt {
6284eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
6294eebcc81SSteven Rostedt 		return;
6304eebcc81SSteven Rostedt 
631cb7be3b2SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
63260a7ecf4SSteven Rostedt 	ftrace_start_up++;
633d61f82d0SSteven Rostedt 	command |= FTRACE_ENABLE_CALLS;
6343d083395SSteven Rostedt 
635df4fc315SSteven Rostedt 	ftrace_startup_enable(command);
636d61f82d0SSteven Rostedt 
637cb7be3b2SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
6383d083395SSteven Rostedt }
6393d083395SSteven Rostedt 
6405a45cfe1SSteven Rostedt static void ftrace_shutdown(int command)
6413d083395SSteven Rostedt {
6424eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
6434eebcc81SSteven Rostedt 		return;
6444eebcc81SSteven Rostedt 
645cb7be3b2SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
64660a7ecf4SSteven Rostedt 	ftrace_start_up--;
64760a7ecf4SSteven Rostedt 	if (!ftrace_start_up)
648d61f82d0SSteven Rostedt 		command |= FTRACE_DISABLE_CALLS;
649d61f82d0SSteven Rostedt 
650d61f82d0SSteven Rostedt 	if (saved_ftrace_func != ftrace_trace_function) {
651d61f82d0SSteven Rostedt 		saved_ftrace_func = ftrace_trace_function;
652d61f82d0SSteven Rostedt 		command |= FTRACE_UPDATE_TRACE_FUNC;
653d61f82d0SSteven Rostedt 	}
654d61f82d0SSteven Rostedt 
655d61f82d0SSteven Rostedt 	if (!command || !ftrace_enabled)
6563d083395SSteven Rostedt 		goto out;
6573d083395SSteven Rostedt 
658d61f82d0SSteven Rostedt 	ftrace_run_update_code(command);
6593d083395SSteven Rostedt  out:
660cb7be3b2SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
6613d083395SSteven Rostedt }
6623d083395SSteven Rostedt 
663e309b41dSIngo Molnar static void ftrace_startup_sysctl(void)
664b0fc494fSSteven Rostedt {
665d61f82d0SSteven Rostedt 	int command = FTRACE_ENABLE_MCOUNT;
666d61f82d0SSteven Rostedt 
6674eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
6684eebcc81SSteven Rostedt 		return;
6694eebcc81SSteven Rostedt 
670cb7be3b2SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
671d61f82d0SSteven Rostedt 	/* Force update next time */
672d61f82d0SSteven Rostedt 	saved_ftrace_func = NULL;
67360a7ecf4SSteven Rostedt 	/* ftrace_start_up is true if we want ftrace running */
67460a7ecf4SSteven Rostedt 	if (ftrace_start_up)
675d61f82d0SSteven Rostedt 		command |= FTRACE_ENABLE_CALLS;
676d61f82d0SSteven Rostedt 
677d61f82d0SSteven Rostedt 	ftrace_run_update_code(command);
678cb7be3b2SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
679b0fc494fSSteven Rostedt }
680b0fc494fSSteven Rostedt 
681e309b41dSIngo Molnar static void ftrace_shutdown_sysctl(void)
682b0fc494fSSteven Rostedt {
683d61f82d0SSteven Rostedt 	int command = FTRACE_DISABLE_MCOUNT;
684d61f82d0SSteven Rostedt 
6854eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
6864eebcc81SSteven Rostedt 		return;
6874eebcc81SSteven Rostedt 
688cb7be3b2SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
68960a7ecf4SSteven Rostedt 	/* ftrace_start_up is true if ftrace is running */
69060a7ecf4SSteven Rostedt 	if (ftrace_start_up)
691d61f82d0SSteven Rostedt 		command |= FTRACE_DISABLE_CALLS;
692d61f82d0SSteven Rostedt 
693d61f82d0SSteven Rostedt 	ftrace_run_update_code(command);
694cb7be3b2SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
695b0fc494fSSteven Rostedt }
696b0fc494fSSteven Rostedt 
6973d083395SSteven Rostedt static cycle_t		ftrace_update_time;
6983d083395SSteven Rostedt static unsigned long	ftrace_update_cnt;
6993d083395SSteven Rostedt unsigned long		ftrace_update_tot_cnt;
7003d083395SSteven Rostedt 
70131e88909SSteven Rostedt static int ftrace_update_code(struct module *mod)
7023d083395SSteven Rostedt {
70308f5ac90SSteven Rostedt 	struct dyn_ftrace *p, *t;
704f22f9a89SAbhishek Sagar 	cycle_t start, stop;
7053d083395SSteven Rostedt 
706750ed1a4SIngo Molnar 	start = ftrace_now(raw_smp_processor_id());
7073d083395SSteven Rostedt 	ftrace_update_cnt = 0;
7083d083395SSteven Rostedt 
70908f5ac90SSteven Rostedt 	list_for_each_entry_safe(p, t, &ftrace_new_addrs, list) {
710f22f9a89SAbhishek Sagar 
71108f5ac90SSteven Rostedt 		/* If something went wrong, bail without enabling anything */
71208f5ac90SSteven Rostedt 		if (unlikely(ftrace_disabled))
71308f5ac90SSteven Rostedt 			return -1;
7143d083395SSteven Rostedt 
71508f5ac90SSteven Rostedt 		list_del_init(&p->list);
716f22f9a89SAbhishek Sagar 
717f22f9a89SAbhishek Sagar 		/* convert record (i.e, patch mcount-call with NOP) */
71831e88909SSteven Rostedt 		if (ftrace_code_disable(mod, p)) {
7190eb96701SAbhishek Sagar 			p->flags |= FTRACE_FL_CONVERTED;
7203d083395SSteven Rostedt 			ftrace_update_cnt++;
72108f5ac90SSteven Rostedt 		} else
7220eb96701SAbhishek Sagar 			ftrace_free_rec(p);
7233d083395SSteven Rostedt 	}
7243d083395SSteven Rostedt 
725750ed1a4SIngo Molnar 	stop = ftrace_now(raw_smp_processor_id());
7263d083395SSteven Rostedt 	ftrace_update_time = stop - start;
7273d083395SSteven Rostedt 	ftrace_update_tot_cnt += ftrace_update_cnt;
7283d083395SSteven Rostedt 
72916444a8aSArnaldo Carvalho de Melo 	return 0;
73016444a8aSArnaldo Carvalho de Melo }
73116444a8aSArnaldo Carvalho de Melo 
73268bf21aaSSteven Rostedt static int __init ftrace_dyn_table_alloc(unsigned long num_to_init)
7333c1720f0SSteven Rostedt {
7343c1720f0SSteven Rostedt 	struct ftrace_page *pg;
7353c1720f0SSteven Rostedt 	int cnt;
7363c1720f0SSteven Rostedt 	int i;
7373c1720f0SSteven Rostedt 
7383c1720f0SSteven Rostedt 	/* allocate a few pages */
7393c1720f0SSteven Rostedt 	ftrace_pages_start = (void *)get_zeroed_page(GFP_KERNEL);
7403c1720f0SSteven Rostedt 	if (!ftrace_pages_start)
7413c1720f0SSteven Rostedt 		return -1;
7423c1720f0SSteven Rostedt 
7433c1720f0SSteven Rostedt 	/*
7443c1720f0SSteven Rostedt 	 * Allocate a few more pages.
7453c1720f0SSteven Rostedt 	 *
7463c1720f0SSteven Rostedt 	 * TODO: have some parser search vmlinux before
7473c1720f0SSteven Rostedt 	 *   final linking to find all calls to ftrace.
7483c1720f0SSteven Rostedt 	 *   Then we can:
7493c1720f0SSteven Rostedt 	 *    a) know how many pages to allocate.
7503c1720f0SSteven Rostedt 	 *     and/or
7513c1720f0SSteven Rostedt 	 *    b) set up the table then.
7523c1720f0SSteven Rostedt 	 *
7533c1720f0SSteven Rostedt 	 *  The dynamic code is still necessary for
7543c1720f0SSteven Rostedt 	 *  modules.
7553c1720f0SSteven Rostedt 	 */
7563c1720f0SSteven Rostedt 
7573c1720f0SSteven Rostedt 	pg = ftrace_pages = ftrace_pages_start;
7583c1720f0SSteven Rostedt 
75968bf21aaSSteven Rostedt 	cnt = num_to_init / ENTRIES_PER_PAGE;
76008f5ac90SSteven Rostedt 	pr_info("ftrace: allocating %ld entries in %d pages\n",
7615821e1b7Swalimis 		num_to_init, cnt + 1);
7623c1720f0SSteven Rostedt 
7633c1720f0SSteven Rostedt 	for (i = 0; i < cnt; i++) {
7643c1720f0SSteven Rostedt 		pg->next = (void *)get_zeroed_page(GFP_KERNEL);
7653c1720f0SSteven Rostedt 
7663c1720f0SSteven Rostedt 		/* If we fail, we'll try later anyway */
7673c1720f0SSteven Rostedt 		if (!pg->next)
7683c1720f0SSteven Rostedt 			break;
7693c1720f0SSteven Rostedt 
7703c1720f0SSteven Rostedt 		pg = pg->next;
7713c1720f0SSteven Rostedt 	}
7723c1720f0SSteven Rostedt 
7733c1720f0SSteven Rostedt 	return 0;
7743c1720f0SSteven Rostedt }
7753c1720f0SSteven Rostedt 
7765072c59fSSteven Rostedt enum {
7775072c59fSSteven Rostedt 	FTRACE_ITER_FILTER	= (1 << 0),
7785072c59fSSteven Rostedt 	FTRACE_ITER_CONT	= (1 << 1),
77941c52c0dSSteven Rostedt 	FTRACE_ITER_NOTRACE	= (1 << 2),
780eb9a7bf0SAbhishek Sagar 	FTRACE_ITER_FAILURES	= (1 << 3),
7810c75a3edSSteven Rostedt 	FTRACE_ITER_PRINTALL	= (1 << 4),
7825072c59fSSteven Rostedt };
7835072c59fSSteven Rostedt 
7845072c59fSSteven Rostedt #define FTRACE_BUFF_MAX (KSYM_SYMBOL_LEN+4) /* room for wildcards */
7855072c59fSSteven Rostedt 
7865072c59fSSteven Rostedt struct ftrace_iterator {
7875072c59fSSteven Rostedt 	struct ftrace_page	*pg;
788431aa3fbSSteven Rostedt 	int			idx;
7895072c59fSSteven Rostedt 	unsigned		flags;
7905072c59fSSteven Rostedt 	unsigned char		buffer[FTRACE_BUFF_MAX+1];
7915072c59fSSteven Rostedt 	unsigned		buffer_idx;
7925072c59fSSteven Rostedt 	unsigned		filtered;
7935072c59fSSteven Rostedt };
7945072c59fSSteven Rostedt 
795e309b41dSIngo Molnar static void *
7965072c59fSSteven Rostedt t_next(struct seq_file *m, void *v, loff_t *pos)
7975072c59fSSteven Rostedt {
7985072c59fSSteven Rostedt 	struct ftrace_iterator *iter = m->private;
7995072c59fSSteven Rostedt 	struct dyn_ftrace *rec = NULL;
8005072c59fSSteven Rostedt 
8015072c59fSSteven Rostedt 	(*pos)++;
8025072c59fSSteven Rostedt 
8030c75a3edSSteven Rostedt 	if (iter->flags & FTRACE_ITER_PRINTALL)
8040c75a3edSSteven Rostedt 		return NULL;
8050c75a3edSSteven Rostedt 
80699ecdc43SSteven Rostedt 	/* should not be called from interrupt context */
80799ecdc43SSteven Rostedt 	spin_lock(&ftrace_lock);
8085072c59fSSteven Rostedt  retry:
8095072c59fSSteven Rostedt 	if (iter->idx >= iter->pg->index) {
8105072c59fSSteven Rostedt 		if (iter->pg->next) {
8115072c59fSSteven Rostedt 			iter->pg = iter->pg->next;
8125072c59fSSteven Rostedt 			iter->idx = 0;
8135072c59fSSteven Rostedt 			goto retry;
81450cdaf08SLiming Wang 		} else {
81550cdaf08SLiming Wang 			iter->idx = -1;
8165072c59fSSteven Rostedt 		}
8175072c59fSSteven Rostedt 	} else {
8185072c59fSSteven Rostedt 		rec = &iter->pg->records[iter->idx++];
819a9fdda33SSteven Rostedt 		if ((rec->flags & FTRACE_FL_FREE) ||
820a9fdda33SSteven Rostedt 
821a9fdda33SSteven Rostedt 		    (!(iter->flags & FTRACE_ITER_FAILURES) &&
822eb9a7bf0SAbhishek Sagar 		     (rec->flags & FTRACE_FL_FAILED)) ||
823eb9a7bf0SAbhishek Sagar 
824eb9a7bf0SAbhishek Sagar 		    ((iter->flags & FTRACE_ITER_FAILURES) &&
825a9fdda33SSteven Rostedt 		     !(rec->flags & FTRACE_FL_FAILED)) ||
826eb9a7bf0SAbhishek Sagar 
8270183fb1cSSteven Rostedt 		    ((iter->flags & FTRACE_ITER_FILTER) &&
8280183fb1cSSteven Rostedt 		     !(rec->flags & FTRACE_FL_FILTER)) ||
8290183fb1cSSteven Rostedt 
83041c52c0dSSteven Rostedt 		    ((iter->flags & FTRACE_ITER_NOTRACE) &&
83141c52c0dSSteven Rostedt 		     !(rec->flags & FTRACE_FL_NOTRACE))) {
8325072c59fSSteven Rostedt 			rec = NULL;
8335072c59fSSteven Rostedt 			goto retry;
8345072c59fSSteven Rostedt 		}
8355072c59fSSteven Rostedt 	}
83699ecdc43SSteven Rostedt 	spin_unlock(&ftrace_lock);
8375072c59fSSteven Rostedt 
8385072c59fSSteven Rostedt 	return rec;
8395072c59fSSteven Rostedt }
8405072c59fSSteven Rostedt 
8415072c59fSSteven Rostedt static void *t_start(struct seq_file *m, loff_t *pos)
8425072c59fSSteven Rostedt {
8435072c59fSSteven Rostedt 	struct ftrace_iterator *iter = m->private;
8445072c59fSSteven Rostedt 	void *p = NULL;
8455072c59fSSteven Rostedt 
8460c75a3edSSteven Rostedt 	/*
8470c75a3edSSteven Rostedt 	 * For set_ftrace_filter reading, if we have the filter
8480c75a3edSSteven Rostedt 	 * off, we can short cut and just print out that all
8490c75a3edSSteven Rostedt 	 * functions are enabled.
8500c75a3edSSteven Rostedt 	 */
8510c75a3edSSteven Rostedt 	if (iter->flags & FTRACE_ITER_FILTER && !ftrace_filtered) {
8520c75a3edSSteven Rostedt 		if (*pos > 0)
8530c75a3edSSteven Rostedt 			return NULL;
8540c75a3edSSteven Rostedt 		iter->flags |= FTRACE_ITER_PRINTALL;
8550c75a3edSSteven Rostedt 		(*pos)++;
8560c75a3edSSteven Rostedt 		return iter;
8570c75a3edSSteven Rostedt 	}
8580c75a3edSSteven Rostedt 
85950cdaf08SLiming Wang 	if (*pos > 0) {
86050cdaf08SLiming Wang 		if (iter->idx < 0)
86150cdaf08SLiming Wang 			return p;
86250cdaf08SLiming Wang 		(*pos)--;
86350cdaf08SLiming Wang 		iter->idx--;
86450cdaf08SLiming Wang 	}
8655821e1b7Swalimis 
86650cdaf08SLiming Wang 	p = t_next(m, p, pos);
8675072c59fSSteven Rostedt 
8685072c59fSSteven Rostedt 	return p;
8695072c59fSSteven Rostedt }
8705072c59fSSteven Rostedt 
8715072c59fSSteven Rostedt static void t_stop(struct seq_file *m, void *p)
8725072c59fSSteven Rostedt {
8735072c59fSSteven Rostedt }
8745072c59fSSteven Rostedt 
8755072c59fSSteven Rostedt static int t_show(struct seq_file *m, void *v)
8765072c59fSSteven Rostedt {
8770c75a3edSSteven Rostedt 	struct ftrace_iterator *iter = m->private;
8785072c59fSSteven Rostedt 	struct dyn_ftrace *rec = v;
8795072c59fSSteven Rostedt 	char str[KSYM_SYMBOL_LEN];
8805072c59fSSteven Rostedt 
8810c75a3edSSteven Rostedt 	if (iter->flags & FTRACE_ITER_PRINTALL) {
8820c75a3edSSteven Rostedt 		seq_printf(m, "#### all functions enabled ####\n");
8830c75a3edSSteven Rostedt 		return 0;
8840c75a3edSSteven Rostedt 	}
8850c75a3edSSteven Rostedt 
8865072c59fSSteven Rostedt 	if (!rec)
8875072c59fSSteven Rostedt 		return 0;
8885072c59fSSteven Rostedt 
8895072c59fSSteven Rostedt 	kallsyms_lookup(rec->ip, NULL, NULL, NULL, str);
8905072c59fSSteven Rostedt 
89150cdaf08SLiming Wang 	seq_printf(m, "%s\n", str);
8925072c59fSSteven Rostedt 
8935072c59fSSteven Rostedt 	return 0;
8945072c59fSSteven Rostedt }
8955072c59fSSteven Rostedt 
8965072c59fSSteven Rostedt static struct seq_operations show_ftrace_seq_ops = {
8975072c59fSSteven Rostedt 	.start = t_start,
8985072c59fSSteven Rostedt 	.next = t_next,
8995072c59fSSteven Rostedt 	.stop = t_stop,
9005072c59fSSteven Rostedt 	.show = t_show,
9015072c59fSSteven Rostedt };
9025072c59fSSteven Rostedt 
903e309b41dSIngo Molnar static int
9045072c59fSSteven Rostedt ftrace_avail_open(struct inode *inode, struct file *file)
9055072c59fSSteven Rostedt {
9065072c59fSSteven Rostedt 	struct ftrace_iterator *iter;
9075072c59fSSteven Rostedt 	int ret;
9085072c59fSSteven Rostedt 
9094eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
9104eebcc81SSteven Rostedt 		return -ENODEV;
9114eebcc81SSteven Rostedt 
9125072c59fSSteven Rostedt 	iter = kzalloc(sizeof(*iter), GFP_KERNEL);
9135072c59fSSteven Rostedt 	if (!iter)
9145072c59fSSteven Rostedt 		return -ENOMEM;
9155072c59fSSteven Rostedt 
9165072c59fSSteven Rostedt 	iter->pg = ftrace_pages_start;
9175072c59fSSteven Rostedt 
9185072c59fSSteven Rostedt 	ret = seq_open(file, &show_ftrace_seq_ops);
9195072c59fSSteven Rostedt 	if (!ret) {
9205072c59fSSteven Rostedt 		struct seq_file *m = file->private_data;
9214bf39a94SIngo Molnar 
9225072c59fSSteven Rostedt 		m->private = iter;
9234bf39a94SIngo Molnar 	} else {
9245072c59fSSteven Rostedt 		kfree(iter);
9254bf39a94SIngo Molnar 	}
9265072c59fSSteven Rostedt 
9275072c59fSSteven Rostedt 	return ret;
9285072c59fSSteven Rostedt }
9295072c59fSSteven Rostedt 
9305072c59fSSteven Rostedt int ftrace_avail_release(struct inode *inode, struct file *file)
9315072c59fSSteven Rostedt {
9325072c59fSSteven Rostedt 	struct seq_file *m = (struct seq_file *)file->private_data;
9335072c59fSSteven Rostedt 	struct ftrace_iterator *iter = m->private;
9345072c59fSSteven Rostedt 
9355072c59fSSteven Rostedt 	seq_release(inode, file);
9365072c59fSSteven Rostedt 	kfree(iter);
9374bf39a94SIngo Molnar 
9385072c59fSSteven Rostedt 	return 0;
9395072c59fSSteven Rostedt }
9405072c59fSSteven Rostedt 
941eb9a7bf0SAbhishek Sagar static int
942eb9a7bf0SAbhishek Sagar ftrace_failures_open(struct inode *inode, struct file *file)
943eb9a7bf0SAbhishek Sagar {
944eb9a7bf0SAbhishek Sagar 	int ret;
945eb9a7bf0SAbhishek Sagar 	struct seq_file *m;
946eb9a7bf0SAbhishek Sagar 	struct ftrace_iterator *iter;
947eb9a7bf0SAbhishek Sagar 
948eb9a7bf0SAbhishek Sagar 	ret = ftrace_avail_open(inode, file);
949eb9a7bf0SAbhishek Sagar 	if (!ret) {
950eb9a7bf0SAbhishek Sagar 		m = (struct seq_file *)file->private_data;
951eb9a7bf0SAbhishek Sagar 		iter = (struct ftrace_iterator *)m->private;
952eb9a7bf0SAbhishek Sagar 		iter->flags = FTRACE_ITER_FAILURES;
953eb9a7bf0SAbhishek Sagar 	}
954eb9a7bf0SAbhishek Sagar 
955eb9a7bf0SAbhishek Sagar 	return ret;
956eb9a7bf0SAbhishek Sagar }
957eb9a7bf0SAbhishek Sagar 
958eb9a7bf0SAbhishek Sagar 
95941c52c0dSSteven Rostedt static void ftrace_filter_reset(int enable)
9605072c59fSSteven Rostedt {
9615072c59fSSteven Rostedt 	struct ftrace_page *pg;
9625072c59fSSteven Rostedt 	struct dyn_ftrace *rec;
96341c52c0dSSteven Rostedt 	unsigned long type = enable ? FTRACE_FL_FILTER : FTRACE_FL_NOTRACE;
9645072c59fSSteven Rostedt 
96599ecdc43SSteven Rostedt 	/* should not be called from interrupt context */
96699ecdc43SSteven Rostedt 	spin_lock(&ftrace_lock);
96741c52c0dSSteven Rostedt 	if (enable)
9685072c59fSSteven Rostedt 		ftrace_filtered = 0;
969265c831cSSteven Rostedt 	do_for_each_ftrace_rec(pg, rec) {
9705072c59fSSteven Rostedt 		if (rec->flags & FTRACE_FL_FAILED)
9715072c59fSSteven Rostedt 			continue;
97241c52c0dSSteven Rostedt 		rec->flags &= ~type;
973265c831cSSteven Rostedt 	} while_for_each_ftrace_rec();
974265c831cSSteven Rostedt 
97599ecdc43SSteven Rostedt 	spin_unlock(&ftrace_lock);
9765072c59fSSteven Rostedt }
9775072c59fSSteven Rostedt 
978e309b41dSIngo Molnar static int
97941c52c0dSSteven Rostedt ftrace_regex_open(struct inode *inode, struct file *file, int enable)
9805072c59fSSteven Rostedt {
9815072c59fSSteven Rostedt 	struct ftrace_iterator *iter;
9825072c59fSSteven Rostedt 	int ret = 0;
9835072c59fSSteven Rostedt 
9844eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
9854eebcc81SSteven Rostedt 		return -ENODEV;
9864eebcc81SSteven Rostedt 
9875072c59fSSteven Rostedt 	iter = kzalloc(sizeof(*iter), GFP_KERNEL);
9885072c59fSSteven Rostedt 	if (!iter)
9895072c59fSSteven Rostedt 		return -ENOMEM;
9905072c59fSSteven Rostedt 
99141c52c0dSSteven Rostedt 	mutex_lock(&ftrace_regex_lock);
9925072c59fSSteven Rostedt 	if ((file->f_mode & FMODE_WRITE) &&
9935072c59fSSteven Rostedt 	    !(file->f_flags & O_APPEND))
99441c52c0dSSteven Rostedt 		ftrace_filter_reset(enable);
9955072c59fSSteven Rostedt 
9965072c59fSSteven Rostedt 	if (file->f_mode & FMODE_READ) {
9975072c59fSSteven Rostedt 		iter->pg = ftrace_pages_start;
99841c52c0dSSteven Rostedt 		iter->flags = enable ? FTRACE_ITER_FILTER :
99941c52c0dSSteven Rostedt 			FTRACE_ITER_NOTRACE;
10005072c59fSSteven Rostedt 
10015072c59fSSteven Rostedt 		ret = seq_open(file, &show_ftrace_seq_ops);
10025072c59fSSteven Rostedt 		if (!ret) {
10035072c59fSSteven Rostedt 			struct seq_file *m = file->private_data;
10045072c59fSSteven Rostedt 			m->private = iter;
10055072c59fSSteven Rostedt 		} else
10065072c59fSSteven Rostedt 			kfree(iter);
10075072c59fSSteven Rostedt 	} else
10085072c59fSSteven Rostedt 		file->private_data = iter;
100941c52c0dSSteven Rostedt 	mutex_unlock(&ftrace_regex_lock);
10105072c59fSSteven Rostedt 
10115072c59fSSteven Rostedt 	return ret;
10125072c59fSSteven Rostedt }
10135072c59fSSteven Rostedt 
101441c52c0dSSteven Rostedt static int
101541c52c0dSSteven Rostedt ftrace_filter_open(struct inode *inode, struct file *file)
101641c52c0dSSteven Rostedt {
101741c52c0dSSteven Rostedt 	return ftrace_regex_open(inode, file, 1);
101841c52c0dSSteven Rostedt }
101941c52c0dSSteven Rostedt 
102041c52c0dSSteven Rostedt static int
102141c52c0dSSteven Rostedt ftrace_notrace_open(struct inode *inode, struct file *file)
102241c52c0dSSteven Rostedt {
102341c52c0dSSteven Rostedt 	return ftrace_regex_open(inode, file, 0);
102441c52c0dSSteven Rostedt }
102541c52c0dSSteven Rostedt 
1026e309b41dSIngo Molnar static ssize_t
102741c52c0dSSteven Rostedt ftrace_regex_read(struct file *file, char __user *ubuf,
10285072c59fSSteven Rostedt 		       size_t cnt, loff_t *ppos)
10295072c59fSSteven Rostedt {
10305072c59fSSteven Rostedt 	if (file->f_mode & FMODE_READ)
10315072c59fSSteven Rostedt 		return seq_read(file, ubuf, cnt, ppos);
10325072c59fSSteven Rostedt 	else
10335072c59fSSteven Rostedt 		return -EPERM;
10345072c59fSSteven Rostedt }
10355072c59fSSteven Rostedt 
1036e309b41dSIngo Molnar static loff_t
103741c52c0dSSteven Rostedt ftrace_regex_lseek(struct file *file, loff_t offset, int origin)
10385072c59fSSteven Rostedt {
10395072c59fSSteven Rostedt 	loff_t ret;
10405072c59fSSteven Rostedt 
10415072c59fSSteven Rostedt 	if (file->f_mode & FMODE_READ)
10425072c59fSSteven Rostedt 		ret = seq_lseek(file, offset, origin);
10435072c59fSSteven Rostedt 	else
10445072c59fSSteven Rostedt 		file->f_pos = ret = 1;
10455072c59fSSteven Rostedt 
10465072c59fSSteven Rostedt 	return ret;
10475072c59fSSteven Rostedt }
10485072c59fSSteven Rostedt 
10495072c59fSSteven Rostedt enum {
10505072c59fSSteven Rostedt 	MATCH_FULL,
10515072c59fSSteven Rostedt 	MATCH_FRONT_ONLY,
10525072c59fSSteven Rostedt 	MATCH_MIDDLE_ONLY,
10535072c59fSSteven Rostedt 	MATCH_END_ONLY,
10545072c59fSSteven Rostedt };
10555072c59fSSteven Rostedt 
1056*9f4801e3SSteven Rostedt /*
1057*9f4801e3SSteven Rostedt  * (static function - no need for kernel doc)
1058*9f4801e3SSteven Rostedt  *
1059*9f4801e3SSteven Rostedt  * Pass in a buffer containing a glob and this function will
1060*9f4801e3SSteven Rostedt  * set search to point to the search part of the buffer and
1061*9f4801e3SSteven Rostedt  * return the type of search it is (see enum above).
1062*9f4801e3SSteven Rostedt  * This does modify buff.
1063*9f4801e3SSteven Rostedt  *
1064*9f4801e3SSteven Rostedt  * Returns enum type.
1065*9f4801e3SSteven Rostedt  *  search returns the pointer to use for comparison.
1066*9f4801e3SSteven Rostedt  *  not returns 1 if buff started with a '!'
1067*9f4801e3SSteven Rostedt  *     0 otherwise.
1068*9f4801e3SSteven Rostedt  */
1069*9f4801e3SSteven Rostedt static int
1070*9f4801e3SSteven Rostedt ftrace_setup_glob(unsigned char *buff, int len, char **search, int *not)
10715072c59fSSteven Rostedt {
10725072c59fSSteven Rostedt 	int type = MATCH_FULL;
1073*9f4801e3SSteven Rostedt 	int i;
1074ea3a6d6dSSteven Rostedt 
1075ea3a6d6dSSteven Rostedt 	if (buff[0] == '!') {
1076*9f4801e3SSteven Rostedt 		*not = 1;
1077ea3a6d6dSSteven Rostedt 		buff++;
1078ea3a6d6dSSteven Rostedt 		len--;
1079*9f4801e3SSteven Rostedt 	} else
1080*9f4801e3SSteven Rostedt 		*not = 0;
1081*9f4801e3SSteven Rostedt 
1082*9f4801e3SSteven Rostedt 	*search = buff;
10835072c59fSSteven Rostedt 
10845072c59fSSteven Rostedt 	for (i = 0; i < len; i++) {
10855072c59fSSteven Rostedt 		if (buff[i] == '*') {
10865072c59fSSteven Rostedt 			if (!i) {
1087*9f4801e3SSteven Rostedt 				*search = buff + 1;
10885072c59fSSteven Rostedt 				type = MATCH_END_ONLY;
10895072c59fSSteven Rostedt 			} else {
1090*9f4801e3SSteven Rostedt 				if (type == MATCH_END_ONLY)
10915072c59fSSteven Rostedt 					type = MATCH_MIDDLE_ONLY;
1092*9f4801e3SSteven Rostedt 				else
10935072c59fSSteven Rostedt 					type = MATCH_FRONT_ONLY;
10945072c59fSSteven Rostedt 				buff[i] = 0;
10955072c59fSSteven Rostedt 				break;
10965072c59fSSteven Rostedt 			}
10975072c59fSSteven Rostedt 		}
10985072c59fSSteven Rostedt 	}
10995072c59fSSteven Rostedt 
1100*9f4801e3SSteven Rostedt 	return type;
1101*9f4801e3SSteven Rostedt }
1102*9f4801e3SSteven Rostedt 
1103*9f4801e3SSteven Rostedt static int
1104*9f4801e3SSteven Rostedt ftrace_match_record(struct dyn_ftrace *rec, char *regex, int len, int type)
1105*9f4801e3SSteven Rostedt {
1106*9f4801e3SSteven Rostedt 	char str[KSYM_SYMBOL_LEN];
1107*9f4801e3SSteven Rostedt 	int matched = 0;
1108*9f4801e3SSteven Rostedt 	char *ptr;
1109*9f4801e3SSteven Rostedt 
1110*9f4801e3SSteven Rostedt 	kallsyms_lookup(rec->ip, NULL, NULL, NULL, str);
1111*9f4801e3SSteven Rostedt 	switch (type) {
1112*9f4801e3SSteven Rostedt 	case MATCH_FULL:
1113*9f4801e3SSteven Rostedt 		if (strcmp(str, regex) == 0)
1114*9f4801e3SSteven Rostedt 			matched = 1;
1115*9f4801e3SSteven Rostedt 		break;
1116*9f4801e3SSteven Rostedt 	case MATCH_FRONT_ONLY:
1117*9f4801e3SSteven Rostedt 		if (strncmp(str, regex, len) == 0)
1118*9f4801e3SSteven Rostedt 			matched = 1;
1119*9f4801e3SSteven Rostedt 		break;
1120*9f4801e3SSteven Rostedt 	case MATCH_MIDDLE_ONLY:
1121*9f4801e3SSteven Rostedt 		if (strstr(str, regex))
1122*9f4801e3SSteven Rostedt 			matched = 1;
1123*9f4801e3SSteven Rostedt 		break;
1124*9f4801e3SSteven Rostedt 	case MATCH_END_ONLY:
1125*9f4801e3SSteven Rostedt 		ptr = strstr(str, regex);
1126*9f4801e3SSteven Rostedt 		if (ptr && (ptr[len] == 0))
1127*9f4801e3SSteven Rostedt 			matched = 1;
1128*9f4801e3SSteven Rostedt 		break;
1129*9f4801e3SSteven Rostedt 	}
1130*9f4801e3SSteven Rostedt 
1131*9f4801e3SSteven Rostedt 	return matched;
1132*9f4801e3SSteven Rostedt }
1133*9f4801e3SSteven Rostedt 
1134*9f4801e3SSteven Rostedt static void ftrace_match_records(char *buff, int len, int enable)
1135*9f4801e3SSteven Rostedt {
1136*9f4801e3SSteven Rostedt 	char *search;
1137*9f4801e3SSteven Rostedt 	struct ftrace_page *pg;
1138*9f4801e3SSteven Rostedt 	struct dyn_ftrace *rec;
1139*9f4801e3SSteven Rostedt 	int type;
1140*9f4801e3SSteven Rostedt 	unsigned long flag = enable ? FTRACE_FL_FILTER : FTRACE_FL_NOTRACE;
1141*9f4801e3SSteven Rostedt 	unsigned search_len;
1142*9f4801e3SSteven Rostedt 	int not;
1143*9f4801e3SSteven Rostedt 
1144*9f4801e3SSteven Rostedt 	type = ftrace_setup_glob(buff, len, &search, &not);
1145*9f4801e3SSteven Rostedt 
1146*9f4801e3SSteven Rostedt 	search_len = strlen(search);
1147*9f4801e3SSteven Rostedt 
114899ecdc43SSteven Rostedt 	/* should not be called from interrupt context */
114999ecdc43SSteven Rostedt 	spin_lock(&ftrace_lock);
115041c52c0dSSteven Rostedt 	if (enable)
11515072c59fSSteven Rostedt 		ftrace_filtered = 1;
1152265c831cSSteven Rostedt 	do_for_each_ftrace_rec(pg, rec) {
11535072c59fSSteven Rostedt 
11545072c59fSSteven Rostedt 		if (rec->flags & FTRACE_FL_FAILED)
11555072c59fSSteven Rostedt 			continue;
1156*9f4801e3SSteven Rostedt 
1157*9f4801e3SSteven Rostedt 		if (ftrace_match_record(rec, search, search_len, type)) {
1158ea3a6d6dSSteven Rostedt 			if (not)
1159ea3a6d6dSSteven Rostedt 				rec->flags &= ~flag;
1160ea3a6d6dSSteven Rostedt 			else
116141c52c0dSSteven Rostedt 				rec->flags |= flag;
11625072c59fSSteven Rostedt 		}
1163*9f4801e3SSteven Rostedt 
1164265c831cSSteven Rostedt 	} while_for_each_ftrace_rec();
116599ecdc43SSteven Rostedt 	spin_unlock(&ftrace_lock);
11665072c59fSSteven Rostedt }
11675072c59fSSteven Rostedt 
1168e309b41dSIngo Molnar static ssize_t
116941c52c0dSSteven Rostedt ftrace_regex_write(struct file *file, const char __user *ubuf,
117041c52c0dSSteven Rostedt 		   size_t cnt, loff_t *ppos, int enable)
11715072c59fSSteven Rostedt {
11725072c59fSSteven Rostedt 	struct ftrace_iterator *iter;
11735072c59fSSteven Rostedt 	char ch;
11745072c59fSSteven Rostedt 	size_t read = 0;
11755072c59fSSteven Rostedt 	ssize_t ret;
11765072c59fSSteven Rostedt 
11775072c59fSSteven Rostedt 	if (!cnt || cnt < 0)
11785072c59fSSteven Rostedt 		return 0;
11795072c59fSSteven Rostedt 
118041c52c0dSSteven Rostedt 	mutex_lock(&ftrace_regex_lock);
11815072c59fSSteven Rostedt 
11825072c59fSSteven Rostedt 	if (file->f_mode & FMODE_READ) {
11835072c59fSSteven Rostedt 		struct seq_file *m = file->private_data;
11845072c59fSSteven Rostedt 		iter = m->private;
11855072c59fSSteven Rostedt 	} else
11865072c59fSSteven Rostedt 		iter = file->private_data;
11875072c59fSSteven Rostedt 
11885072c59fSSteven Rostedt 	if (!*ppos) {
11895072c59fSSteven Rostedt 		iter->flags &= ~FTRACE_ITER_CONT;
11905072c59fSSteven Rostedt 		iter->buffer_idx = 0;
11915072c59fSSteven Rostedt 	}
11925072c59fSSteven Rostedt 
11935072c59fSSteven Rostedt 	ret = get_user(ch, ubuf++);
11945072c59fSSteven Rostedt 	if (ret)
11955072c59fSSteven Rostedt 		goto out;
11965072c59fSSteven Rostedt 	read++;
11975072c59fSSteven Rostedt 	cnt--;
11985072c59fSSteven Rostedt 
11995072c59fSSteven Rostedt 	if (!(iter->flags & ~FTRACE_ITER_CONT)) {
12005072c59fSSteven Rostedt 		/* skip white space */
12015072c59fSSteven Rostedt 		while (cnt && isspace(ch)) {
12025072c59fSSteven Rostedt 			ret = get_user(ch, ubuf++);
12035072c59fSSteven Rostedt 			if (ret)
12045072c59fSSteven Rostedt 				goto out;
12055072c59fSSteven Rostedt 			read++;
12065072c59fSSteven Rostedt 			cnt--;
12075072c59fSSteven Rostedt 		}
12085072c59fSSteven Rostedt 
12095072c59fSSteven Rostedt 		if (isspace(ch)) {
12105072c59fSSteven Rostedt 			file->f_pos += read;
12115072c59fSSteven Rostedt 			ret = read;
12125072c59fSSteven Rostedt 			goto out;
12135072c59fSSteven Rostedt 		}
12145072c59fSSteven Rostedt 
12155072c59fSSteven Rostedt 		iter->buffer_idx = 0;
12165072c59fSSteven Rostedt 	}
12175072c59fSSteven Rostedt 
12185072c59fSSteven Rostedt 	while (cnt && !isspace(ch)) {
12195072c59fSSteven Rostedt 		if (iter->buffer_idx < FTRACE_BUFF_MAX)
12205072c59fSSteven Rostedt 			iter->buffer[iter->buffer_idx++] = ch;
12215072c59fSSteven Rostedt 		else {
12225072c59fSSteven Rostedt 			ret = -EINVAL;
12235072c59fSSteven Rostedt 			goto out;
12245072c59fSSteven Rostedt 		}
12255072c59fSSteven Rostedt 		ret = get_user(ch, ubuf++);
12265072c59fSSteven Rostedt 		if (ret)
12275072c59fSSteven Rostedt 			goto out;
12285072c59fSSteven Rostedt 		read++;
12295072c59fSSteven Rostedt 		cnt--;
12305072c59fSSteven Rostedt 	}
12315072c59fSSteven Rostedt 
12325072c59fSSteven Rostedt 	if (isspace(ch)) {
12335072c59fSSteven Rostedt 		iter->filtered++;
12345072c59fSSteven Rostedt 		iter->buffer[iter->buffer_idx] = 0;
12357f24b31bSSteven Rostedt 		ftrace_match_records(iter->buffer, iter->buffer_idx, enable);
12365072c59fSSteven Rostedt 		iter->buffer_idx = 0;
12375072c59fSSteven Rostedt 	} else
12385072c59fSSteven Rostedt 		iter->flags |= FTRACE_ITER_CONT;
12395072c59fSSteven Rostedt 
12405072c59fSSteven Rostedt 
12415072c59fSSteven Rostedt 	file->f_pos += read;
12425072c59fSSteven Rostedt 
12435072c59fSSteven Rostedt 	ret = read;
12445072c59fSSteven Rostedt  out:
124541c52c0dSSteven Rostedt 	mutex_unlock(&ftrace_regex_lock);
12465072c59fSSteven Rostedt 
12475072c59fSSteven Rostedt 	return ret;
12485072c59fSSteven Rostedt }
12495072c59fSSteven Rostedt 
125041c52c0dSSteven Rostedt static ssize_t
125141c52c0dSSteven Rostedt ftrace_filter_write(struct file *file, const char __user *ubuf,
125241c52c0dSSteven Rostedt 		    size_t cnt, loff_t *ppos)
125341c52c0dSSteven Rostedt {
125441c52c0dSSteven Rostedt 	return ftrace_regex_write(file, ubuf, cnt, ppos, 1);
125541c52c0dSSteven Rostedt }
125641c52c0dSSteven Rostedt 
125741c52c0dSSteven Rostedt static ssize_t
125841c52c0dSSteven Rostedt ftrace_notrace_write(struct file *file, const char __user *ubuf,
125941c52c0dSSteven Rostedt 		     size_t cnt, loff_t *ppos)
126041c52c0dSSteven Rostedt {
126141c52c0dSSteven Rostedt 	return ftrace_regex_write(file, ubuf, cnt, ppos, 0);
126241c52c0dSSteven Rostedt }
126341c52c0dSSteven Rostedt 
126441c52c0dSSteven Rostedt static void
126541c52c0dSSteven Rostedt ftrace_set_regex(unsigned char *buf, int len, int reset, int enable)
126641c52c0dSSteven Rostedt {
126741c52c0dSSteven Rostedt 	if (unlikely(ftrace_disabled))
126841c52c0dSSteven Rostedt 		return;
126941c52c0dSSteven Rostedt 
127041c52c0dSSteven Rostedt 	mutex_lock(&ftrace_regex_lock);
127141c52c0dSSteven Rostedt 	if (reset)
127241c52c0dSSteven Rostedt 		ftrace_filter_reset(enable);
127341c52c0dSSteven Rostedt 	if (buf)
12747f24b31bSSteven Rostedt 		ftrace_match_records(buf, len, enable);
127541c52c0dSSteven Rostedt 	mutex_unlock(&ftrace_regex_lock);
127641c52c0dSSteven Rostedt }
127741c52c0dSSteven Rostedt 
127877a2b37dSSteven Rostedt /**
127977a2b37dSSteven Rostedt  * ftrace_set_filter - set a function to filter on in ftrace
128077a2b37dSSteven Rostedt  * @buf - the string that holds the function filter text.
128177a2b37dSSteven Rostedt  * @len - the length of the string.
128277a2b37dSSteven Rostedt  * @reset - non zero to reset all filters before applying this filter.
128377a2b37dSSteven Rostedt  *
128477a2b37dSSteven Rostedt  * Filters denote which functions should be enabled when tracing is enabled.
128577a2b37dSSteven Rostedt  * If @buf is NULL and reset is set, all functions will be enabled for tracing.
128677a2b37dSSteven Rostedt  */
1287e309b41dSIngo Molnar void ftrace_set_filter(unsigned char *buf, int len, int reset)
128877a2b37dSSteven Rostedt {
128941c52c0dSSteven Rostedt 	ftrace_set_regex(buf, len, reset, 1);
129041c52c0dSSteven Rostedt }
12914eebcc81SSteven Rostedt 
129241c52c0dSSteven Rostedt /**
129341c52c0dSSteven Rostedt  * ftrace_set_notrace - set a function to not trace in ftrace
129441c52c0dSSteven Rostedt  * @buf - the string that holds the function notrace text.
129541c52c0dSSteven Rostedt  * @len - the length of the string.
129641c52c0dSSteven Rostedt  * @reset - non zero to reset all filters before applying this filter.
129741c52c0dSSteven Rostedt  *
129841c52c0dSSteven Rostedt  * Notrace Filters denote which functions should not be enabled when tracing
129941c52c0dSSteven Rostedt  * is enabled. If @buf is NULL and reset is set, all functions will be enabled
130041c52c0dSSteven Rostedt  * for tracing.
130141c52c0dSSteven Rostedt  */
130241c52c0dSSteven Rostedt void ftrace_set_notrace(unsigned char *buf, int len, int reset)
130341c52c0dSSteven Rostedt {
130441c52c0dSSteven Rostedt 	ftrace_set_regex(buf, len, reset, 0);
130577a2b37dSSteven Rostedt }
130677a2b37dSSteven Rostedt 
1307e309b41dSIngo Molnar static int
130841c52c0dSSteven Rostedt ftrace_regex_release(struct inode *inode, struct file *file, int enable)
13095072c59fSSteven Rostedt {
13105072c59fSSteven Rostedt 	struct seq_file *m = (struct seq_file *)file->private_data;
13115072c59fSSteven Rostedt 	struct ftrace_iterator *iter;
13125072c59fSSteven Rostedt 
131341c52c0dSSteven Rostedt 	mutex_lock(&ftrace_regex_lock);
13145072c59fSSteven Rostedt 	if (file->f_mode & FMODE_READ) {
13155072c59fSSteven Rostedt 		iter = m->private;
13165072c59fSSteven Rostedt 
13175072c59fSSteven Rostedt 		seq_release(inode, file);
13185072c59fSSteven Rostedt 	} else
13195072c59fSSteven Rostedt 		iter = file->private_data;
13205072c59fSSteven Rostedt 
13215072c59fSSteven Rostedt 	if (iter->buffer_idx) {
13225072c59fSSteven Rostedt 		iter->filtered++;
13235072c59fSSteven Rostedt 		iter->buffer[iter->buffer_idx] = 0;
13247f24b31bSSteven Rostedt 		ftrace_match_records(iter->buffer, iter->buffer_idx, enable);
13255072c59fSSteven Rostedt 	}
13265072c59fSSteven Rostedt 
13275072c59fSSteven Rostedt 	mutex_lock(&ftrace_sysctl_lock);
1328cb7be3b2SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
1329ee02a2e5SSteven Rostedt 	if (ftrace_start_up && ftrace_enabled)
13305072c59fSSteven Rostedt 		ftrace_run_update_code(FTRACE_ENABLE_CALLS);
1331cb7be3b2SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
13325072c59fSSteven Rostedt 	mutex_unlock(&ftrace_sysctl_lock);
13335072c59fSSteven Rostedt 
13345072c59fSSteven Rostedt 	kfree(iter);
133541c52c0dSSteven Rostedt 	mutex_unlock(&ftrace_regex_lock);
13365072c59fSSteven Rostedt 	return 0;
13375072c59fSSteven Rostedt }
13385072c59fSSteven Rostedt 
133941c52c0dSSteven Rostedt static int
134041c52c0dSSteven Rostedt ftrace_filter_release(struct inode *inode, struct file *file)
134141c52c0dSSteven Rostedt {
134241c52c0dSSteven Rostedt 	return ftrace_regex_release(inode, file, 1);
134341c52c0dSSteven Rostedt }
134441c52c0dSSteven Rostedt 
134541c52c0dSSteven Rostedt static int
134641c52c0dSSteven Rostedt ftrace_notrace_release(struct inode *inode, struct file *file)
134741c52c0dSSteven Rostedt {
134841c52c0dSSteven Rostedt 	return ftrace_regex_release(inode, file, 0);
134941c52c0dSSteven Rostedt }
135041c52c0dSSteven Rostedt 
13515072c59fSSteven Rostedt static struct file_operations ftrace_avail_fops = {
13525072c59fSSteven Rostedt 	.open = ftrace_avail_open,
13535072c59fSSteven Rostedt 	.read = seq_read,
13545072c59fSSteven Rostedt 	.llseek = seq_lseek,
13555072c59fSSteven Rostedt 	.release = ftrace_avail_release,
13565072c59fSSteven Rostedt };
13575072c59fSSteven Rostedt 
1358eb9a7bf0SAbhishek Sagar static struct file_operations ftrace_failures_fops = {
1359eb9a7bf0SAbhishek Sagar 	.open = ftrace_failures_open,
1360eb9a7bf0SAbhishek Sagar 	.read = seq_read,
1361eb9a7bf0SAbhishek Sagar 	.llseek = seq_lseek,
1362eb9a7bf0SAbhishek Sagar 	.release = ftrace_avail_release,
1363eb9a7bf0SAbhishek Sagar };
1364eb9a7bf0SAbhishek Sagar 
13655072c59fSSteven Rostedt static struct file_operations ftrace_filter_fops = {
13665072c59fSSteven Rostedt 	.open = ftrace_filter_open,
136741c52c0dSSteven Rostedt 	.read = ftrace_regex_read,
13685072c59fSSteven Rostedt 	.write = ftrace_filter_write,
136941c52c0dSSteven Rostedt 	.llseek = ftrace_regex_lseek,
13705072c59fSSteven Rostedt 	.release = ftrace_filter_release,
13715072c59fSSteven Rostedt };
13725072c59fSSteven Rostedt 
137341c52c0dSSteven Rostedt static struct file_operations ftrace_notrace_fops = {
137441c52c0dSSteven Rostedt 	.open = ftrace_notrace_open,
137541c52c0dSSteven Rostedt 	.read = ftrace_regex_read,
137641c52c0dSSteven Rostedt 	.write = ftrace_notrace_write,
137741c52c0dSSteven Rostedt 	.llseek = ftrace_regex_lseek,
137841c52c0dSSteven Rostedt 	.release = ftrace_notrace_release,
137941c52c0dSSteven Rostedt };
138041c52c0dSSteven Rostedt 
1381ea4e2bc4SSteven Rostedt #ifdef CONFIG_FUNCTION_GRAPH_TRACER
1382ea4e2bc4SSteven Rostedt 
1383ea4e2bc4SSteven Rostedt static DEFINE_MUTEX(graph_lock);
1384ea4e2bc4SSteven Rostedt 
1385ea4e2bc4SSteven Rostedt int ftrace_graph_count;
1386ea4e2bc4SSteven Rostedt unsigned long ftrace_graph_funcs[FTRACE_GRAPH_MAX_FUNCS] __read_mostly;
1387ea4e2bc4SSteven Rostedt 
1388ea4e2bc4SSteven Rostedt static void *
1389ea4e2bc4SSteven Rostedt g_next(struct seq_file *m, void *v, loff_t *pos)
1390ea4e2bc4SSteven Rostedt {
1391ea4e2bc4SSteven Rostedt 	unsigned long *array = m->private;
1392ea4e2bc4SSteven Rostedt 	int index = *pos;
1393ea4e2bc4SSteven Rostedt 
1394ea4e2bc4SSteven Rostedt 	(*pos)++;
1395ea4e2bc4SSteven Rostedt 
1396ea4e2bc4SSteven Rostedt 	if (index >= ftrace_graph_count)
1397ea4e2bc4SSteven Rostedt 		return NULL;
1398ea4e2bc4SSteven Rostedt 
1399ea4e2bc4SSteven Rostedt 	return &array[index];
1400ea4e2bc4SSteven Rostedt }
1401ea4e2bc4SSteven Rostedt 
1402ea4e2bc4SSteven Rostedt static void *g_start(struct seq_file *m, loff_t *pos)
1403ea4e2bc4SSteven Rostedt {
1404ea4e2bc4SSteven Rostedt 	void *p = NULL;
1405ea4e2bc4SSteven Rostedt 
1406ea4e2bc4SSteven Rostedt 	mutex_lock(&graph_lock);
1407ea4e2bc4SSteven Rostedt 
1408ea4e2bc4SSteven Rostedt 	p = g_next(m, p, pos);
1409ea4e2bc4SSteven Rostedt 
1410ea4e2bc4SSteven Rostedt 	return p;
1411ea4e2bc4SSteven Rostedt }
1412ea4e2bc4SSteven Rostedt 
1413ea4e2bc4SSteven Rostedt static void g_stop(struct seq_file *m, void *p)
1414ea4e2bc4SSteven Rostedt {
1415ea4e2bc4SSteven Rostedt 	mutex_unlock(&graph_lock);
1416ea4e2bc4SSteven Rostedt }
1417ea4e2bc4SSteven Rostedt 
1418ea4e2bc4SSteven Rostedt static int g_show(struct seq_file *m, void *v)
1419ea4e2bc4SSteven Rostedt {
1420ea4e2bc4SSteven Rostedt 	unsigned long *ptr = v;
1421ea4e2bc4SSteven Rostedt 	char str[KSYM_SYMBOL_LEN];
1422ea4e2bc4SSteven Rostedt 
1423ea4e2bc4SSteven Rostedt 	if (!ptr)
1424ea4e2bc4SSteven Rostedt 		return 0;
1425ea4e2bc4SSteven Rostedt 
1426ea4e2bc4SSteven Rostedt 	kallsyms_lookup(*ptr, NULL, NULL, NULL, str);
1427ea4e2bc4SSteven Rostedt 
1428ea4e2bc4SSteven Rostedt 	seq_printf(m, "%s\n", str);
1429ea4e2bc4SSteven Rostedt 
1430ea4e2bc4SSteven Rostedt 	return 0;
1431ea4e2bc4SSteven Rostedt }
1432ea4e2bc4SSteven Rostedt 
1433ea4e2bc4SSteven Rostedt static struct seq_operations ftrace_graph_seq_ops = {
1434ea4e2bc4SSteven Rostedt 	.start = g_start,
1435ea4e2bc4SSteven Rostedt 	.next = g_next,
1436ea4e2bc4SSteven Rostedt 	.stop = g_stop,
1437ea4e2bc4SSteven Rostedt 	.show = g_show,
1438ea4e2bc4SSteven Rostedt };
1439ea4e2bc4SSteven Rostedt 
1440ea4e2bc4SSteven Rostedt static int
1441ea4e2bc4SSteven Rostedt ftrace_graph_open(struct inode *inode, struct file *file)
1442ea4e2bc4SSteven Rostedt {
1443ea4e2bc4SSteven Rostedt 	int ret = 0;
1444ea4e2bc4SSteven Rostedt 
1445ea4e2bc4SSteven Rostedt 	if (unlikely(ftrace_disabled))
1446ea4e2bc4SSteven Rostedt 		return -ENODEV;
1447ea4e2bc4SSteven Rostedt 
1448ea4e2bc4SSteven Rostedt 	mutex_lock(&graph_lock);
1449ea4e2bc4SSteven Rostedt 	if ((file->f_mode & FMODE_WRITE) &&
1450ea4e2bc4SSteven Rostedt 	    !(file->f_flags & O_APPEND)) {
1451ea4e2bc4SSteven Rostedt 		ftrace_graph_count = 0;
1452ea4e2bc4SSteven Rostedt 		memset(ftrace_graph_funcs, 0, sizeof(ftrace_graph_funcs));
1453ea4e2bc4SSteven Rostedt 	}
1454ea4e2bc4SSteven Rostedt 
1455ea4e2bc4SSteven Rostedt 	if (file->f_mode & FMODE_READ) {
1456ea4e2bc4SSteven Rostedt 		ret = seq_open(file, &ftrace_graph_seq_ops);
1457ea4e2bc4SSteven Rostedt 		if (!ret) {
1458ea4e2bc4SSteven Rostedt 			struct seq_file *m = file->private_data;
1459ea4e2bc4SSteven Rostedt 			m->private = ftrace_graph_funcs;
1460ea4e2bc4SSteven Rostedt 		}
1461ea4e2bc4SSteven Rostedt 	} else
1462ea4e2bc4SSteven Rostedt 		file->private_data = ftrace_graph_funcs;
1463ea4e2bc4SSteven Rostedt 	mutex_unlock(&graph_lock);
1464ea4e2bc4SSteven Rostedt 
1465ea4e2bc4SSteven Rostedt 	return ret;
1466ea4e2bc4SSteven Rostedt }
1467ea4e2bc4SSteven Rostedt 
1468ea4e2bc4SSteven Rostedt static ssize_t
1469ea4e2bc4SSteven Rostedt ftrace_graph_read(struct file *file, char __user *ubuf,
1470ea4e2bc4SSteven Rostedt 		       size_t cnt, loff_t *ppos)
1471ea4e2bc4SSteven Rostedt {
1472ea4e2bc4SSteven Rostedt 	if (file->f_mode & FMODE_READ)
1473ea4e2bc4SSteven Rostedt 		return seq_read(file, ubuf, cnt, ppos);
1474ea4e2bc4SSteven Rostedt 	else
1475ea4e2bc4SSteven Rostedt 		return -EPERM;
1476ea4e2bc4SSteven Rostedt }
1477ea4e2bc4SSteven Rostedt 
1478ea4e2bc4SSteven Rostedt static int
1479ea4e2bc4SSteven Rostedt ftrace_set_func(unsigned long *array, int idx, char *buffer)
1480ea4e2bc4SSteven Rostedt {
1481ea4e2bc4SSteven Rostedt 	char str[KSYM_SYMBOL_LEN];
1482ea4e2bc4SSteven Rostedt 	struct dyn_ftrace *rec;
1483ea4e2bc4SSteven Rostedt 	struct ftrace_page *pg;
1484ea4e2bc4SSteven Rostedt 	int found = 0;
1485265c831cSSteven Rostedt 	int j;
1486ea4e2bc4SSteven Rostedt 
1487ea4e2bc4SSteven Rostedt 	if (ftrace_disabled)
1488ea4e2bc4SSteven Rostedt 		return -ENODEV;
1489ea4e2bc4SSteven Rostedt 
1490ea4e2bc4SSteven Rostedt 	/* should not be called from interrupt context */
1491ea4e2bc4SSteven Rostedt 	spin_lock(&ftrace_lock);
1492ea4e2bc4SSteven Rostedt 
1493265c831cSSteven Rostedt 	do_for_each_ftrace_rec(pg, rec) {
1494ea4e2bc4SSteven Rostedt 
1495ea4e2bc4SSteven Rostedt 		if (rec->flags & (FTRACE_FL_FAILED | FTRACE_FL_FREE))
1496ea4e2bc4SSteven Rostedt 			continue;
1497ea4e2bc4SSteven Rostedt 
1498ea4e2bc4SSteven Rostedt 		kallsyms_lookup(rec->ip, NULL, NULL, NULL, str);
1499ea4e2bc4SSteven Rostedt 		if (strcmp(str, buffer) == 0) {
1500265c831cSSteven Rostedt 			/* Return 1 if we add it to the array */
1501ea4e2bc4SSteven Rostedt 			found = 1;
1502faec2ec5SLiming Wang 			for (j = 0; j < idx; j++)
1503faec2ec5SLiming Wang 				if (array[j] == rec->ip) {
1504faec2ec5SLiming Wang 					found = 0;
1505faec2ec5SLiming Wang 					break;
1506faec2ec5SLiming Wang 				}
1507faec2ec5SLiming Wang 			if (found)
1508ea4e2bc4SSteven Rostedt 				array[idx] = rec->ip;
1509265c831cSSteven Rostedt 			goto out;
1510ea4e2bc4SSteven Rostedt 		}
1511265c831cSSteven Rostedt 	} while_for_each_ftrace_rec();
1512265c831cSSteven Rostedt  out:
1513ea4e2bc4SSteven Rostedt 	spin_unlock(&ftrace_lock);
1514ea4e2bc4SSteven Rostedt 
1515ea4e2bc4SSteven Rostedt 	return found ? 0 : -EINVAL;
1516ea4e2bc4SSteven Rostedt }
1517ea4e2bc4SSteven Rostedt 
1518ea4e2bc4SSteven Rostedt static ssize_t
1519ea4e2bc4SSteven Rostedt ftrace_graph_write(struct file *file, const char __user *ubuf,
1520ea4e2bc4SSteven Rostedt 		   size_t cnt, loff_t *ppos)
1521ea4e2bc4SSteven Rostedt {
1522ea4e2bc4SSteven Rostedt 	unsigned char buffer[FTRACE_BUFF_MAX+1];
1523ea4e2bc4SSteven Rostedt 	unsigned long *array;
1524ea4e2bc4SSteven Rostedt 	size_t read = 0;
1525ea4e2bc4SSteven Rostedt 	ssize_t ret;
1526ea4e2bc4SSteven Rostedt 	int index = 0;
1527ea4e2bc4SSteven Rostedt 	char ch;
1528ea4e2bc4SSteven Rostedt 
1529ea4e2bc4SSteven Rostedt 	if (!cnt || cnt < 0)
1530ea4e2bc4SSteven Rostedt 		return 0;
1531ea4e2bc4SSteven Rostedt 
1532ea4e2bc4SSteven Rostedt 	mutex_lock(&graph_lock);
1533ea4e2bc4SSteven Rostedt 
1534ea4e2bc4SSteven Rostedt 	if (ftrace_graph_count >= FTRACE_GRAPH_MAX_FUNCS) {
1535ea4e2bc4SSteven Rostedt 		ret = -EBUSY;
1536ea4e2bc4SSteven Rostedt 		goto out;
1537ea4e2bc4SSteven Rostedt 	}
1538ea4e2bc4SSteven Rostedt 
1539ea4e2bc4SSteven Rostedt 	if (file->f_mode & FMODE_READ) {
1540ea4e2bc4SSteven Rostedt 		struct seq_file *m = file->private_data;
1541ea4e2bc4SSteven Rostedt 		array = m->private;
1542ea4e2bc4SSteven Rostedt 	} else
1543ea4e2bc4SSteven Rostedt 		array = file->private_data;
1544ea4e2bc4SSteven Rostedt 
1545ea4e2bc4SSteven Rostedt 	ret = get_user(ch, ubuf++);
1546ea4e2bc4SSteven Rostedt 	if (ret)
1547ea4e2bc4SSteven Rostedt 		goto out;
1548ea4e2bc4SSteven Rostedt 	read++;
1549ea4e2bc4SSteven Rostedt 	cnt--;
1550ea4e2bc4SSteven Rostedt 
1551ea4e2bc4SSteven Rostedt 	/* skip white space */
1552ea4e2bc4SSteven Rostedt 	while (cnt && isspace(ch)) {
1553ea4e2bc4SSteven Rostedt 		ret = get_user(ch, ubuf++);
1554ea4e2bc4SSteven Rostedt 		if (ret)
1555ea4e2bc4SSteven Rostedt 			goto out;
1556ea4e2bc4SSteven Rostedt 		read++;
1557ea4e2bc4SSteven Rostedt 		cnt--;
1558ea4e2bc4SSteven Rostedt 	}
1559ea4e2bc4SSteven Rostedt 
1560ea4e2bc4SSteven Rostedt 	if (isspace(ch)) {
1561ea4e2bc4SSteven Rostedt 		*ppos += read;
1562ea4e2bc4SSteven Rostedt 		ret = read;
1563ea4e2bc4SSteven Rostedt 		goto out;
1564ea4e2bc4SSteven Rostedt 	}
1565ea4e2bc4SSteven Rostedt 
1566ea4e2bc4SSteven Rostedt 	while (cnt && !isspace(ch)) {
1567ea4e2bc4SSteven Rostedt 		if (index < FTRACE_BUFF_MAX)
1568ea4e2bc4SSteven Rostedt 			buffer[index++] = ch;
1569ea4e2bc4SSteven Rostedt 		else {
1570ea4e2bc4SSteven Rostedt 			ret = -EINVAL;
1571ea4e2bc4SSteven Rostedt 			goto out;
1572ea4e2bc4SSteven Rostedt 		}
1573ea4e2bc4SSteven Rostedt 		ret = get_user(ch, ubuf++);
1574ea4e2bc4SSteven Rostedt 		if (ret)
1575ea4e2bc4SSteven Rostedt 			goto out;
1576ea4e2bc4SSteven Rostedt 		read++;
1577ea4e2bc4SSteven Rostedt 		cnt--;
1578ea4e2bc4SSteven Rostedt 	}
1579ea4e2bc4SSteven Rostedt 	buffer[index] = 0;
1580ea4e2bc4SSteven Rostedt 
1581ea4e2bc4SSteven Rostedt 	/* we allow only one at a time */
1582ea4e2bc4SSteven Rostedt 	ret = ftrace_set_func(array, ftrace_graph_count, buffer);
1583ea4e2bc4SSteven Rostedt 	if (ret)
1584ea4e2bc4SSteven Rostedt 		goto out;
1585ea4e2bc4SSteven Rostedt 
1586ea4e2bc4SSteven Rostedt 	ftrace_graph_count++;
1587ea4e2bc4SSteven Rostedt 
1588ea4e2bc4SSteven Rostedt 	file->f_pos += read;
1589ea4e2bc4SSteven Rostedt 
1590ea4e2bc4SSteven Rostedt 	ret = read;
1591ea4e2bc4SSteven Rostedt  out:
1592ea4e2bc4SSteven Rostedt 	mutex_unlock(&graph_lock);
1593ea4e2bc4SSteven Rostedt 
1594ea4e2bc4SSteven Rostedt 	return ret;
1595ea4e2bc4SSteven Rostedt }
1596ea4e2bc4SSteven Rostedt 
1597ea4e2bc4SSteven Rostedt static const struct file_operations ftrace_graph_fops = {
1598ea4e2bc4SSteven Rostedt 	.open = ftrace_graph_open,
1599ea4e2bc4SSteven Rostedt 	.read = ftrace_graph_read,
1600ea4e2bc4SSteven Rostedt 	.write = ftrace_graph_write,
1601ea4e2bc4SSteven Rostedt };
1602ea4e2bc4SSteven Rostedt #endif /* CONFIG_FUNCTION_GRAPH_TRACER */
1603ea4e2bc4SSteven Rostedt 
1604df4fc315SSteven Rostedt static __init int ftrace_init_dyn_debugfs(struct dentry *d_tracer)
16055072c59fSSteven Rostedt {
16065072c59fSSteven Rostedt 	struct dentry *entry;
16075072c59fSSteven Rostedt 
16085072c59fSSteven Rostedt 	entry = debugfs_create_file("available_filter_functions", 0444,
16095072c59fSSteven Rostedt 				    d_tracer, NULL, &ftrace_avail_fops);
16105072c59fSSteven Rostedt 	if (!entry)
16115072c59fSSteven Rostedt 		pr_warning("Could not create debugfs "
16125072c59fSSteven Rostedt 			   "'available_filter_functions' entry\n");
16135072c59fSSteven Rostedt 
1614eb9a7bf0SAbhishek Sagar 	entry = debugfs_create_file("failures", 0444,
1615eb9a7bf0SAbhishek Sagar 				    d_tracer, NULL, &ftrace_failures_fops);
1616eb9a7bf0SAbhishek Sagar 	if (!entry)
1617eb9a7bf0SAbhishek Sagar 		pr_warning("Could not create debugfs 'failures' entry\n");
1618eb9a7bf0SAbhishek Sagar 
16195072c59fSSteven Rostedt 	entry = debugfs_create_file("set_ftrace_filter", 0644, d_tracer,
16205072c59fSSteven Rostedt 				    NULL, &ftrace_filter_fops);
16215072c59fSSteven Rostedt 	if (!entry)
16225072c59fSSteven Rostedt 		pr_warning("Could not create debugfs "
16235072c59fSSteven Rostedt 			   "'set_ftrace_filter' entry\n");
162441c52c0dSSteven Rostedt 
162541c52c0dSSteven Rostedt 	entry = debugfs_create_file("set_ftrace_notrace", 0644, d_tracer,
162641c52c0dSSteven Rostedt 				    NULL, &ftrace_notrace_fops);
162741c52c0dSSteven Rostedt 	if (!entry)
162841c52c0dSSteven Rostedt 		pr_warning("Could not create debugfs "
162941c52c0dSSteven Rostedt 			   "'set_ftrace_notrace' entry\n");
1630ad90c0e3SSteven Rostedt 
1631ea4e2bc4SSteven Rostedt #ifdef CONFIG_FUNCTION_GRAPH_TRACER
1632ea4e2bc4SSteven Rostedt 	entry = debugfs_create_file("set_graph_function", 0444, d_tracer,
1633ea4e2bc4SSteven Rostedt 				    NULL,
1634ea4e2bc4SSteven Rostedt 				    &ftrace_graph_fops);
1635ea4e2bc4SSteven Rostedt 	if (!entry)
1636ea4e2bc4SSteven Rostedt 		pr_warning("Could not create debugfs "
1637ea4e2bc4SSteven Rostedt 			   "'set_graph_function' entry\n");
1638ea4e2bc4SSteven Rostedt #endif /* CONFIG_FUNCTION_GRAPH_TRACER */
1639ea4e2bc4SSteven Rostedt 
16405072c59fSSteven Rostedt 	return 0;
16415072c59fSSteven Rostedt }
16425072c59fSSteven Rostedt 
164331e88909SSteven Rostedt static int ftrace_convert_nops(struct module *mod,
164431e88909SSteven Rostedt 			       unsigned long *start,
164568bf21aaSSteven Rostedt 			       unsigned long *end)
164668bf21aaSSteven Rostedt {
164768bf21aaSSteven Rostedt 	unsigned long *p;
164868bf21aaSSteven Rostedt 	unsigned long addr;
164968bf21aaSSteven Rostedt 	unsigned long flags;
165068bf21aaSSteven Rostedt 
165108f5ac90SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
165268bf21aaSSteven Rostedt 	p = start;
165368bf21aaSSteven Rostedt 	while (p < end) {
165468bf21aaSSteven Rostedt 		addr = ftrace_call_adjust(*p++);
165520e5227eSSteven Rostedt 		/*
165620e5227eSSteven Rostedt 		 * Some architecture linkers will pad between
165720e5227eSSteven Rostedt 		 * the different mcount_loc sections of different
165820e5227eSSteven Rostedt 		 * object files to satisfy alignments.
165920e5227eSSteven Rostedt 		 * Skip any NULL pointers.
166020e5227eSSteven Rostedt 		 */
166120e5227eSSteven Rostedt 		if (!addr)
166220e5227eSSteven Rostedt 			continue;
166368bf21aaSSteven Rostedt 		ftrace_record_ip(addr);
166468bf21aaSSteven Rostedt 	}
166568bf21aaSSteven Rostedt 
166608f5ac90SSteven Rostedt 	/* disable interrupts to prevent kstop machine */
166768bf21aaSSteven Rostedt 	local_irq_save(flags);
166831e88909SSteven Rostedt 	ftrace_update_code(mod);
166968bf21aaSSteven Rostedt 	local_irq_restore(flags);
167008f5ac90SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
167168bf21aaSSteven Rostedt 
167268bf21aaSSteven Rostedt 	return 0;
167368bf21aaSSteven Rostedt }
167468bf21aaSSteven Rostedt 
167531e88909SSteven Rostedt void ftrace_init_module(struct module *mod,
167631e88909SSteven Rostedt 			unsigned long *start, unsigned long *end)
167790d595feSSteven Rostedt {
167800fd61aeSSteven Rostedt 	if (ftrace_disabled || start == end)
1679fed1939cSSteven Rostedt 		return;
168031e88909SSteven Rostedt 	ftrace_convert_nops(mod, start, end);
168190d595feSSteven Rostedt }
168290d595feSSteven Rostedt 
168368bf21aaSSteven Rostedt extern unsigned long __start_mcount_loc[];
168468bf21aaSSteven Rostedt extern unsigned long __stop_mcount_loc[];
168568bf21aaSSteven Rostedt 
168668bf21aaSSteven Rostedt void __init ftrace_init(void)
168768bf21aaSSteven Rostedt {
168868bf21aaSSteven Rostedt 	unsigned long count, addr, flags;
168968bf21aaSSteven Rostedt 	int ret;
169068bf21aaSSteven Rostedt 
169168bf21aaSSteven Rostedt 	/* Keep the ftrace pointer to the stub */
169268bf21aaSSteven Rostedt 	addr = (unsigned long)ftrace_stub;
169368bf21aaSSteven Rostedt 
169468bf21aaSSteven Rostedt 	local_irq_save(flags);
169568bf21aaSSteven Rostedt 	ftrace_dyn_arch_init(&addr);
169668bf21aaSSteven Rostedt 	local_irq_restore(flags);
169768bf21aaSSteven Rostedt 
169868bf21aaSSteven Rostedt 	/* ftrace_dyn_arch_init places the return code in addr */
169968bf21aaSSteven Rostedt 	if (addr)
170068bf21aaSSteven Rostedt 		goto failed;
170168bf21aaSSteven Rostedt 
170268bf21aaSSteven Rostedt 	count = __stop_mcount_loc - __start_mcount_loc;
170368bf21aaSSteven Rostedt 
170468bf21aaSSteven Rostedt 	ret = ftrace_dyn_table_alloc(count);
170568bf21aaSSteven Rostedt 	if (ret)
170668bf21aaSSteven Rostedt 		goto failed;
170768bf21aaSSteven Rostedt 
170868bf21aaSSteven Rostedt 	last_ftrace_enabled = ftrace_enabled = 1;
170968bf21aaSSteven Rostedt 
171031e88909SSteven Rostedt 	ret = ftrace_convert_nops(NULL,
171131e88909SSteven Rostedt 				  __start_mcount_loc,
171268bf21aaSSteven Rostedt 				  __stop_mcount_loc);
171368bf21aaSSteven Rostedt 
171468bf21aaSSteven Rostedt 	return;
171568bf21aaSSteven Rostedt  failed:
171668bf21aaSSteven Rostedt 	ftrace_disabled = 1;
171768bf21aaSSteven Rostedt }
171868bf21aaSSteven Rostedt 
17193d083395SSteven Rostedt #else
17200b6e4d56SFrederic Weisbecker 
17210b6e4d56SFrederic Weisbecker static int __init ftrace_nodyn_init(void)
17220b6e4d56SFrederic Weisbecker {
17230b6e4d56SFrederic Weisbecker 	ftrace_enabled = 1;
17240b6e4d56SFrederic Weisbecker 	return 0;
17250b6e4d56SFrederic Weisbecker }
17260b6e4d56SFrederic Weisbecker device_initcall(ftrace_nodyn_init);
17270b6e4d56SFrederic Weisbecker 
1728df4fc315SSteven Rostedt static inline int ftrace_init_dyn_debugfs(struct dentry *d_tracer) { return 0; }
1729df4fc315SSteven Rostedt static inline void ftrace_startup_enable(int command) { }
17305a45cfe1SSteven Rostedt /* Keep as macros so we do not need to define the commands */
17315a45cfe1SSteven Rostedt # define ftrace_startup(command)	do { } while (0)
17325a45cfe1SSteven Rostedt # define ftrace_shutdown(command)	do { } while (0)
1733b0fc494fSSteven Rostedt # define ftrace_startup_sysctl()	do { } while (0)
1734b0fc494fSSteven Rostedt # define ftrace_shutdown_sysctl()	do { } while (0)
17353d083395SSteven Rostedt #endif /* CONFIG_DYNAMIC_FTRACE */
17363d083395SSteven Rostedt 
1737df4fc315SSteven Rostedt static ssize_t
1738df4fc315SSteven Rostedt ftrace_pid_read(struct file *file, char __user *ubuf,
1739df4fc315SSteven Rostedt 		       size_t cnt, loff_t *ppos)
1740df4fc315SSteven Rostedt {
1741df4fc315SSteven Rostedt 	char buf[64];
1742df4fc315SSteven Rostedt 	int r;
1743df4fc315SSteven Rostedt 
1744e32d8956SSteven Rostedt 	if (ftrace_pid_trace == ftrace_swapper_pid)
1745e32d8956SSteven Rostedt 		r = sprintf(buf, "swapper tasks\n");
1746e32d8956SSteven Rostedt 	else if (ftrace_pid_trace)
1747978f3a45SSteven Rostedt 		r = sprintf(buf, "%u\n", pid_nr(ftrace_pid_trace));
1748df4fc315SSteven Rostedt 	else
1749df4fc315SSteven Rostedt 		r = sprintf(buf, "no pid\n");
1750df4fc315SSteven Rostedt 
1751df4fc315SSteven Rostedt 	return simple_read_from_buffer(ubuf, cnt, ppos, buf, r);
1752df4fc315SSteven Rostedt }
1753df4fc315SSteven Rostedt 
1754e32d8956SSteven Rostedt static void clear_ftrace_swapper(void)
1755e32d8956SSteven Rostedt {
1756e32d8956SSteven Rostedt 	struct task_struct *p;
1757e32d8956SSteven Rostedt 	int cpu;
1758e32d8956SSteven Rostedt 
1759e32d8956SSteven Rostedt 	get_online_cpus();
1760e32d8956SSteven Rostedt 	for_each_online_cpu(cpu) {
1761e32d8956SSteven Rostedt 		p = idle_task(cpu);
1762e32d8956SSteven Rostedt 		clear_tsk_trace_trace(p);
1763e32d8956SSteven Rostedt 	}
1764e32d8956SSteven Rostedt 	put_online_cpus();
1765e32d8956SSteven Rostedt }
1766e32d8956SSteven Rostedt 
1767e32d8956SSteven Rostedt static void set_ftrace_swapper(void)
1768e32d8956SSteven Rostedt {
1769e32d8956SSteven Rostedt 	struct task_struct *p;
1770e32d8956SSteven Rostedt 	int cpu;
1771e32d8956SSteven Rostedt 
1772e32d8956SSteven Rostedt 	get_online_cpus();
1773e32d8956SSteven Rostedt 	for_each_online_cpu(cpu) {
1774e32d8956SSteven Rostedt 		p = idle_task(cpu);
1775e32d8956SSteven Rostedt 		set_tsk_trace_trace(p);
1776e32d8956SSteven Rostedt 	}
1777e32d8956SSteven Rostedt 	put_online_cpus();
1778e32d8956SSteven Rostedt }
1779e32d8956SSteven Rostedt 
1780e32d8956SSteven Rostedt static void clear_ftrace_pid(struct pid *pid)
1781978f3a45SSteven Rostedt {
1782978f3a45SSteven Rostedt 	struct task_struct *p;
1783978f3a45SSteven Rostedt 
1784229c4ef8SOleg Nesterov 	rcu_read_lock();
1785e32d8956SSteven Rostedt 	do_each_pid_task(pid, PIDTYPE_PID, p) {
1786978f3a45SSteven Rostedt 		clear_tsk_trace_trace(p);
1787e32d8956SSteven Rostedt 	} while_each_pid_task(pid, PIDTYPE_PID, p);
1788229c4ef8SOleg Nesterov 	rcu_read_unlock();
1789229c4ef8SOleg Nesterov 
1790e32d8956SSteven Rostedt 	put_pid(pid);
1791978f3a45SSteven Rostedt }
1792978f3a45SSteven Rostedt 
1793e32d8956SSteven Rostedt static void set_ftrace_pid(struct pid *pid)
1794978f3a45SSteven Rostedt {
1795978f3a45SSteven Rostedt 	struct task_struct *p;
1796978f3a45SSteven Rostedt 
1797229c4ef8SOleg Nesterov 	rcu_read_lock();
1798978f3a45SSteven Rostedt 	do_each_pid_task(pid, PIDTYPE_PID, p) {
1799978f3a45SSteven Rostedt 		set_tsk_trace_trace(p);
1800978f3a45SSteven Rostedt 	} while_each_pid_task(pid, PIDTYPE_PID, p);
1801229c4ef8SOleg Nesterov 	rcu_read_unlock();
1802978f3a45SSteven Rostedt }
1803978f3a45SSteven Rostedt 
1804e32d8956SSteven Rostedt static void clear_ftrace_pid_task(struct pid **pid)
1805e32d8956SSteven Rostedt {
1806e32d8956SSteven Rostedt 	if (*pid == ftrace_swapper_pid)
1807e32d8956SSteven Rostedt 		clear_ftrace_swapper();
1808e32d8956SSteven Rostedt 	else
1809e32d8956SSteven Rostedt 		clear_ftrace_pid(*pid);
1810e32d8956SSteven Rostedt 
1811e32d8956SSteven Rostedt 	*pid = NULL;
1812e32d8956SSteven Rostedt }
1813e32d8956SSteven Rostedt 
1814e32d8956SSteven Rostedt static void set_ftrace_pid_task(struct pid *pid)
1815e32d8956SSteven Rostedt {
1816e32d8956SSteven Rostedt 	if (pid == ftrace_swapper_pid)
1817e32d8956SSteven Rostedt 		set_ftrace_swapper();
1818e32d8956SSteven Rostedt 	else
1819e32d8956SSteven Rostedt 		set_ftrace_pid(pid);
1820e32d8956SSteven Rostedt }
1821e32d8956SSteven Rostedt 
1822df4fc315SSteven Rostedt static ssize_t
1823df4fc315SSteven Rostedt ftrace_pid_write(struct file *filp, const char __user *ubuf,
1824df4fc315SSteven Rostedt 		   size_t cnt, loff_t *ppos)
1825df4fc315SSteven Rostedt {
1826978f3a45SSteven Rostedt 	struct pid *pid;
1827df4fc315SSteven Rostedt 	char buf[64];
1828df4fc315SSteven Rostedt 	long val;
1829df4fc315SSteven Rostedt 	int ret;
1830df4fc315SSteven Rostedt 
1831df4fc315SSteven Rostedt 	if (cnt >= sizeof(buf))
1832df4fc315SSteven Rostedt 		return -EINVAL;
1833df4fc315SSteven Rostedt 
1834df4fc315SSteven Rostedt 	if (copy_from_user(&buf, ubuf, cnt))
1835df4fc315SSteven Rostedt 		return -EFAULT;
1836df4fc315SSteven Rostedt 
1837df4fc315SSteven Rostedt 	buf[cnt] = 0;
1838df4fc315SSteven Rostedt 
1839df4fc315SSteven Rostedt 	ret = strict_strtol(buf, 10, &val);
1840df4fc315SSteven Rostedt 	if (ret < 0)
1841df4fc315SSteven Rostedt 		return ret;
1842df4fc315SSteven Rostedt 
1843df4fc315SSteven Rostedt 	mutex_lock(&ftrace_start_lock);
1844978f3a45SSteven Rostedt 	if (val < 0) {
1845df4fc315SSteven Rostedt 		/* disable pid tracing */
1846978f3a45SSteven Rostedt 		if (!ftrace_pid_trace)
1847df4fc315SSteven Rostedt 			goto out;
1848978f3a45SSteven Rostedt 
1849978f3a45SSteven Rostedt 		clear_ftrace_pid_task(&ftrace_pid_trace);
1850df4fc315SSteven Rostedt 
1851df4fc315SSteven Rostedt 	} else {
1852e32d8956SSteven Rostedt 		/* swapper task is special */
1853e32d8956SSteven Rostedt 		if (!val) {
1854e32d8956SSteven Rostedt 			pid = ftrace_swapper_pid;
1855e32d8956SSteven Rostedt 			if (pid == ftrace_pid_trace)
1856e32d8956SSteven Rostedt 				goto out;
1857e32d8956SSteven Rostedt 		} else {
1858978f3a45SSteven Rostedt 			pid = find_get_pid(val);
1859df4fc315SSteven Rostedt 
1860978f3a45SSteven Rostedt 			if (pid == ftrace_pid_trace) {
1861978f3a45SSteven Rostedt 				put_pid(pid);
1862978f3a45SSteven Rostedt 				goto out;
1863978f3a45SSteven Rostedt 			}
1864e32d8956SSteven Rostedt 		}
1865978f3a45SSteven Rostedt 
1866978f3a45SSteven Rostedt 		if (ftrace_pid_trace)
1867978f3a45SSteven Rostedt 			clear_ftrace_pid_task(&ftrace_pid_trace);
1868978f3a45SSteven Rostedt 
1869978f3a45SSteven Rostedt 		if (!pid)
1870df4fc315SSteven Rostedt 			goto out;
1871df4fc315SSteven Rostedt 
1872978f3a45SSteven Rostedt 		ftrace_pid_trace = pid;
18730ef8cde5SSteven Rostedt 
1874978f3a45SSteven Rostedt 		set_ftrace_pid_task(ftrace_pid_trace);
1875df4fc315SSteven Rostedt 	}
1876df4fc315SSteven Rostedt 
1877df4fc315SSteven Rostedt 	/* update the function call */
1878df4fc315SSteven Rostedt 	ftrace_update_pid_func();
1879df4fc315SSteven Rostedt 	ftrace_startup_enable(0);
1880df4fc315SSteven Rostedt 
1881df4fc315SSteven Rostedt  out:
1882df4fc315SSteven Rostedt 	mutex_unlock(&ftrace_start_lock);
1883df4fc315SSteven Rostedt 
1884df4fc315SSteven Rostedt 	return cnt;
1885df4fc315SSteven Rostedt }
1886df4fc315SSteven Rostedt 
1887df4fc315SSteven Rostedt static struct file_operations ftrace_pid_fops = {
1888df4fc315SSteven Rostedt 	.read = ftrace_pid_read,
1889df4fc315SSteven Rostedt 	.write = ftrace_pid_write,
1890df4fc315SSteven Rostedt };
1891df4fc315SSteven Rostedt 
1892df4fc315SSteven Rostedt static __init int ftrace_init_debugfs(void)
1893df4fc315SSteven Rostedt {
1894df4fc315SSteven Rostedt 	struct dentry *d_tracer;
1895df4fc315SSteven Rostedt 	struct dentry *entry;
1896df4fc315SSteven Rostedt 
1897df4fc315SSteven Rostedt 	d_tracer = tracing_init_dentry();
1898df4fc315SSteven Rostedt 	if (!d_tracer)
1899df4fc315SSteven Rostedt 		return 0;
1900df4fc315SSteven Rostedt 
1901df4fc315SSteven Rostedt 	ftrace_init_dyn_debugfs(d_tracer);
1902df4fc315SSteven Rostedt 
1903df4fc315SSteven Rostedt 	entry = debugfs_create_file("set_ftrace_pid", 0644, d_tracer,
1904df4fc315SSteven Rostedt 				    NULL, &ftrace_pid_fops);
1905df4fc315SSteven Rostedt 	if (!entry)
1906df4fc315SSteven Rostedt 		pr_warning("Could not create debugfs "
1907df4fc315SSteven Rostedt 			   "'set_ftrace_pid' entry\n");
1908df4fc315SSteven Rostedt 	return 0;
1909df4fc315SSteven Rostedt }
1910df4fc315SSteven Rostedt 
1911df4fc315SSteven Rostedt fs_initcall(ftrace_init_debugfs);
1912df4fc315SSteven Rostedt 
19133d083395SSteven Rostedt /**
191481adbdc0SSteven Rostedt  * ftrace_kill - kill ftrace
1915a2bb6a3dSSteven Rostedt  *
1916a2bb6a3dSSteven Rostedt  * This function should be used by panic code. It stops ftrace
1917a2bb6a3dSSteven Rostedt  * but in a not so nice way. If you need to simply kill ftrace
1918a2bb6a3dSSteven Rostedt  * from a non-atomic section, use ftrace_kill.
1919a2bb6a3dSSteven Rostedt  */
192081adbdc0SSteven Rostedt void ftrace_kill(void)
1921a2bb6a3dSSteven Rostedt {
1922a2bb6a3dSSteven Rostedt 	ftrace_disabled = 1;
1923a2bb6a3dSSteven Rostedt 	ftrace_enabled = 0;
1924a2bb6a3dSSteven Rostedt 	clear_ftrace_function();
1925a2bb6a3dSSteven Rostedt }
1926a2bb6a3dSSteven Rostedt 
1927a2bb6a3dSSteven Rostedt /**
19283d083395SSteven Rostedt  * register_ftrace_function - register a function for profiling
19293d083395SSteven Rostedt  * @ops - ops structure that holds the function for profiling.
19303d083395SSteven Rostedt  *
19313d083395SSteven Rostedt  * Register a function to be called by all functions in the
19323d083395SSteven Rostedt  * kernel.
19333d083395SSteven Rostedt  *
19343d083395SSteven Rostedt  * Note: @ops->func and all the functions it calls must be labeled
19353d083395SSteven Rostedt  *       with "notrace", otherwise it will go into a
19363d083395SSteven Rostedt  *       recursive loop.
19373d083395SSteven Rostedt  */
19383d083395SSteven Rostedt int register_ftrace_function(struct ftrace_ops *ops)
19393d083395SSteven Rostedt {
1940b0fc494fSSteven Rostedt 	int ret;
1941b0fc494fSSteven Rostedt 
19424eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
19434eebcc81SSteven Rostedt 		return -1;
19444eebcc81SSteven Rostedt 
1945b0fc494fSSteven Rostedt 	mutex_lock(&ftrace_sysctl_lock);
1946e7d3737eSFrederic Weisbecker 
1947b0fc494fSSteven Rostedt 	ret = __register_ftrace_function(ops);
19485a45cfe1SSteven Rostedt 	ftrace_startup(0);
1949b0fc494fSSteven Rostedt 
1950e7d3737eSFrederic Weisbecker 	mutex_unlock(&ftrace_sysctl_lock);
1951b0fc494fSSteven Rostedt 	return ret;
19523d083395SSteven Rostedt }
19533d083395SSteven Rostedt 
19543d083395SSteven Rostedt /**
195532632920SUwe Kleine-Koenig  * unregister_ftrace_function - unregister a function for profiling.
19563d083395SSteven Rostedt  * @ops - ops structure that holds the function to unregister
19573d083395SSteven Rostedt  *
19583d083395SSteven Rostedt  * Unregister a function that was added to be called by ftrace profiling.
19593d083395SSteven Rostedt  */
19603d083395SSteven Rostedt int unregister_ftrace_function(struct ftrace_ops *ops)
19613d083395SSteven Rostedt {
19623d083395SSteven Rostedt 	int ret;
19633d083395SSteven Rostedt 
1964b0fc494fSSteven Rostedt 	mutex_lock(&ftrace_sysctl_lock);
19653d083395SSteven Rostedt 	ret = __unregister_ftrace_function(ops);
19665a45cfe1SSteven Rostedt 	ftrace_shutdown(0);
1967b0fc494fSSteven Rostedt 	mutex_unlock(&ftrace_sysctl_lock);
1968b0fc494fSSteven Rostedt 
1969b0fc494fSSteven Rostedt 	return ret;
1970b0fc494fSSteven Rostedt }
1971b0fc494fSSteven Rostedt 
1972e309b41dSIngo Molnar int
1973b0fc494fSSteven Rostedt ftrace_enable_sysctl(struct ctl_table *table, int write,
19745072c59fSSteven Rostedt 		     struct file *file, void __user *buffer, size_t *lenp,
1975b0fc494fSSteven Rostedt 		     loff_t *ppos)
1976b0fc494fSSteven Rostedt {
1977b0fc494fSSteven Rostedt 	int ret;
1978b0fc494fSSteven Rostedt 
19794eebcc81SSteven Rostedt 	if (unlikely(ftrace_disabled))
19804eebcc81SSteven Rostedt 		return -ENODEV;
19814eebcc81SSteven Rostedt 
1982b0fc494fSSteven Rostedt 	mutex_lock(&ftrace_sysctl_lock);
1983b0fc494fSSteven Rostedt 
19845072c59fSSteven Rostedt 	ret  = proc_dointvec(table, write, file, buffer, lenp, ppos);
1985b0fc494fSSteven Rostedt 
1986b0fc494fSSteven Rostedt 	if (ret || !write || (last_ftrace_enabled == ftrace_enabled))
1987b0fc494fSSteven Rostedt 		goto out;
1988b0fc494fSSteven Rostedt 
1989b0fc494fSSteven Rostedt 	last_ftrace_enabled = ftrace_enabled;
1990b0fc494fSSteven Rostedt 
1991b0fc494fSSteven Rostedt 	if (ftrace_enabled) {
1992b0fc494fSSteven Rostedt 
1993b0fc494fSSteven Rostedt 		ftrace_startup_sysctl();
1994b0fc494fSSteven Rostedt 
1995b0fc494fSSteven Rostedt 		/* we are starting ftrace again */
1996b0fc494fSSteven Rostedt 		if (ftrace_list != &ftrace_list_end) {
1997b0fc494fSSteven Rostedt 			if (ftrace_list->next == &ftrace_list_end)
1998b0fc494fSSteven Rostedt 				ftrace_trace_function = ftrace_list->func;
1999b0fc494fSSteven Rostedt 			else
2000b0fc494fSSteven Rostedt 				ftrace_trace_function = ftrace_list_func;
2001b0fc494fSSteven Rostedt 		}
2002b0fc494fSSteven Rostedt 
2003b0fc494fSSteven Rostedt 	} else {
2004b0fc494fSSteven Rostedt 		/* stopping ftrace calls (just send to ftrace_stub) */
2005b0fc494fSSteven Rostedt 		ftrace_trace_function = ftrace_stub;
2006b0fc494fSSteven Rostedt 
2007b0fc494fSSteven Rostedt 		ftrace_shutdown_sysctl();
2008b0fc494fSSteven Rostedt 	}
2009b0fc494fSSteven Rostedt 
2010b0fc494fSSteven Rostedt  out:
2011b0fc494fSSteven Rostedt 	mutex_unlock(&ftrace_sysctl_lock);
20123d083395SSteven Rostedt 	return ret;
201316444a8aSArnaldo Carvalho de Melo }
2014f17845e5SIngo Molnar 
2015fb52607aSFrederic Weisbecker #ifdef CONFIG_FUNCTION_GRAPH_TRACER
2016e7d3737eSFrederic Weisbecker 
2017287b6e68SFrederic Weisbecker static atomic_t ftrace_graph_active;
20184a2b8ddaSFrederic Weisbecker static struct notifier_block ftrace_suspend_notifier;
2019f201ae23SFrederic Weisbecker 
2020e49dc19cSSteven Rostedt int ftrace_graph_entry_stub(struct ftrace_graph_ent *trace)
2021e49dc19cSSteven Rostedt {
2022e49dc19cSSteven Rostedt 	return 0;
2023e49dc19cSSteven Rostedt }
2024e49dc19cSSteven Rostedt 
2025287b6e68SFrederic Weisbecker /* The callbacks that hook a function */
2026287b6e68SFrederic Weisbecker trace_func_graph_ret_t ftrace_graph_return =
2027287b6e68SFrederic Weisbecker 			(trace_func_graph_ret_t)ftrace_stub;
2028e49dc19cSSteven Rostedt trace_func_graph_ent_t ftrace_graph_entry = ftrace_graph_entry_stub;
2029f201ae23SFrederic Weisbecker 
2030f201ae23SFrederic Weisbecker /* Try to assign a return stack array on FTRACE_RETSTACK_ALLOC_SIZE tasks. */
2031f201ae23SFrederic Weisbecker static int alloc_retstack_tasklist(struct ftrace_ret_stack **ret_stack_list)
2032f201ae23SFrederic Weisbecker {
2033f201ae23SFrederic Weisbecker 	int i;
2034f201ae23SFrederic Weisbecker 	int ret = 0;
2035f201ae23SFrederic Weisbecker 	unsigned long flags;
2036f201ae23SFrederic Weisbecker 	int start = 0, end = FTRACE_RETSTACK_ALLOC_SIZE;
2037f201ae23SFrederic Weisbecker 	struct task_struct *g, *t;
2038f201ae23SFrederic Weisbecker 
2039f201ae23SFrederic Weisbecker 	for (i = 0; i < FTRACE_RETSTACK_ALLOC_SIZE; i++) {
2040f201ae23SFrederic Weisbecker 		ret_stack_list[i] = kmalloc(FTRACE_RETFUNC_DEPTH
2041f201ae23SFrederic Weisbecker 					* sizeof(struct ftrace_ret_stack),
2042f201ae23SFrederic Weisbecker 					GFP_KERNEL);
2043f201ae23SFrederic Weisbecker 		if (!ret_stack_list[i]) {
2044f201ae23SFrederic Weisbecker 			start = 0;
2045f201ae23SFrederic Weisbecker 			end = i;
2046f201ae23SFrederic Weisbecker 			ret = -ENOMEM;
2047f201ae23SFrederic Weisbecker 			goto free;
2048f201ae23SFrederic Weisbecker 		}
2049f201ae23SFrederic Weisbecker 	}
2050f201ae23SFrederic Weisbecker 
2051f201ae23SFrederic Weisbecker 	read_lock_irqsave(&tasklist_lock, flags);
2052f201ae23SFrederic Weisbecker 	do_each_thread(g, t) {
2053f201ae23SFrederic Weisbecker 		if (start == end) {
2054f201ae23SFrederic Weisbecker 			ret = -EAGAIN;
2055f201ae23SFrederic Weisbecker 			goto unlock;
2056f201ae23SFrederic Weisbecker 		}
2057f201ae23SFrederic Weisbecker 
2058f201ae23SFrederic Weisbecker 		if (t->ret_stack == NULL) {
2059f201ae23SFrederic Weisbecker 			t->curr_ret_stack = -1;
206048d68b20SFrederic Weisbecker 			/* Make sure IRQs see the -1 first: */
206148d68b20SFrederic Weisbecker 			barrier();
206248d68b20SFrederic Weisbecker 			t->ret_stack = ret_stack_list[start++];
2063380c4b14SFrederic Weisbecker 			atomic_set(&t->tracing_graph_pause, 0);
2064f201ae23SFrederic Weisbecker 			atomic_set(&t->trace_overrun, 0);
2065f201ae23SFrederic Weisbecker 		}
2066f201ae23SFrederic Weisbecker 	} while_each_thread(g, t);
2067f201ae23SFrederic Weisbecker 
2068f201ae23SFrederic Weisbecker unlock:
2069f201ae23SFrederic Weisbecker 	read_unlock_irqrestore(&tasklist_lock, flags);
2070f201ae23SFrederic Weisbecker free:
2071f201ae23SFrederic Weisbecker 	for (i = start; i < end; i++)
2072f201ae23SFrederic Weisbecker 		kfree(ret_stack_list[i]);
2073f201ae23SFrederic Weisbecker 	return ret;
2074f201ae23SFrederic Weisbecker }
2075f201ae23SFrederic Weisbecker 
2076f201ae23SFrederic Weisbecker /* Allocate a return stack for each task */
2077fb52607aSFrederic Weisbecker static int start_graph_tracing(void)
2078f201ae23SFrederic Weisbecker {
2079f201ae23SFrederic Weisbecker 	struct ftrace_ret_stack **ret_stack_list;
2080f201ae23SFrederic Weisbecker 	int ret;
2081f201ae23SFrederic Weisbecker 
2082f201ae23SFrederic Weisbecker 	ret_stack_list = kmalloc(FTRACE_RETSTACK_ALLOC_SIZE *
2083f201ae23SFrederic Weisbecker 				sizeof(struct ftrace_ret_stack *),
2084f201ae23SFrederic Weisbecker 				GFP_KERNEL);
2085f201ae23SFrederic Weisbecker 
2086f201ae23SFrederic Weisbecker 	if (!ret_stack_list)
2087f201ae23SFrederic Weisbecker 		return -ENOMEM;
2088f201ae23SFrederic Weisbecker 
2089f201ae23SFrederic Weisbecker 	do {
2090f201ae23SFrederic Weisbecker 		ret = alloc_retstack_tasklist(ret_stack_list);
2091f201ae23SFrederic Weisbecker 	} while (ret == -EAGAIN);
2092f201ae23SFrederic Weisbecker 
2093f201ae23SFrederic Weisbecker 	kfree(ret_stack_list);
2094f201ae23SFrederic Weisbecker 	return ret;
2095f201ae23SFrederic Weisbecker }
2096f201ae23SFrederic Weisbecker 
20974a2b8ddaSFrederic Weisbecker /*
20984a2b8ddaSFrederic Weisbecker  * Hibernation protection.
20994a2b8ddaSFrederic Weisbecker  * The state of the current task is too much unstable during
21004a2b8ddaSFrederic Weisbecker  * suspend/restore to disk. We want to protect against that.
21014a2b8ddaSFrederic Weisbecker  */
21024a2b8ddaSFrederic Weisbecker static int
21034a2b8ddaSFrederic Weisbecker ftrace_suspend_notifier_call(struct notifier_block *bl, unsigned long state,
21044a2b8ddaSFrederic Weisbecker 							void *unused)
21054a2b8ddaSFrederic Weisbecker {
21064a2b8ddaSFrederic Weisbecker 	switch (state) {
21074a2b8ddaSFrederic Weisbecker 	case PM_HIBERNATION_PREPARE:
21084a2b8ddaSFrederic Weisbecker 		pause_graph_tracing();
21094a2b8ddaSFrederic Weisbecker 		break;
21104a2b8ddaSFrederic Weisbecker 
21114a2b8ddaSFrederic Weisbecker 	case PM_POST_HIBERNATION:
21124a2b8ddaSFrederic Weisbecker 		unpause_graph_tracing();
21134a2b8ddaSFrederic Weisbecker 		break;
21144a2b8ddaSFrederic Weisbecker 	}
21154a2b8ddaSFrederic Weisbecker 	return NOTIFY_DONE;
21164a2b8ddaSFrederic Weisbecker }
21174a2b8ddaSFrederic Weisbecker 
2118287b6e68SFrederic Weisbecker int register_ftrace_graph(trace_func_graph_ret_t retfunc,
2119287b6e68SFrederic Weisbecker 			trace_func_graph_ent_t entryfunc)
212015e6cb36SFrederic Weisbecker {
2121e7d3737eSFrederic Weisbecker 	int ret = 0;
2122e7d3737eSFrederic Weisbecker 
2123e7d3737eSFrederic Weisbecker 	mutex_lock(&ftrace_sysctl_lock);
2124e7d3737eSFrederic Weisbecker 
21254a2b8ddaSFrederic Weisbecker 	ftrace_suspend_notifier.notifier_call = ftrace_suspend_notifier_call;
21264a2b8ddaSFrederic Weisbecker 	register_pm_notifier(&ftrace_suspend_notifier);
21274a2b8ddaSFrederic Weisbecker 
2128287b6e68SFrederic Weisbecker 	atomic_inc(&ftrace_graph_active);
2129fb52607aSFrederic Weisbecker 	ret = start_graph_tracing();
2130f201ae23SFrederic Weisbecker 	if (ret) {
2131287b6e68SFrederic Weisbecker 		atomic_dec(&ftrace_graph_active);
2132f201ae23SFrederic Weisbecker 		goto out;
2133f201ae23SFrederic Weisbecker 	}
2134e53a6319SSteven Rostedt 
2135287b6e68SFrederic Weisbecker 	ftrace_graph_return = retfunc;
2136287b6e68SFrederic Weisbecker 	ftrace_graph_entry = entryfunc;
2137e53a6319SSteven Rostedt 
21385a45cfe1SSteven Rostedt 	ftrace_startup(FTRACE_START_FUNC_RET);
2139e7d3737eSFrederic Weisbecker 
2140e7d3737eSFrederic Weisbecker out:
2141e7d3737eSFrederic Weisbecker 	mutex_unlock(&ftrace_sysctl_lock);
2142e7d3737eSFrederic Weisbecker 	return ret;
214315e6cb36SFrederic Weisbecker }
214415e6cb36SFrederic Weisbecker 
2145fb52607aSFrederic Weisbecker void unregister_ftrace_graph(void)
214615e6cb36SFrederic Weisbecker {
2147e7d3737eSFrederic Weisbecker 	mutex_lock(&ftrace_sysctl_lock);
2148e7d3737eSFrederic Weisbecker 
2149287b6e68SFrederic Weisbecker 	atomic_dec(&ftrace_graph_active);
2150287b6e68SFrederic Weisbecker 	ftrace_graph_return = (trace_func_graph_ret_t)ftrace_stub;
2151e49dc19cSSteven Rostedt 	ftrace_graph_entry = ftrace_graph_entry_stub;
21525a45cfe1SSteven Rostedt 	ftrace_shutdown(FTRACE_STOP_FUNC_RET);
21534a2b8ddaSFrederic Weisbecker 	unregister_pm_notifier(&ftrace_suspend_notifier);
2154e7d3737eSFrederic Weisbecker 
2155e7d3737eSFrederic Weisbecker 	mutex_unlock(&ftrace_sysctl_lock);
215615e6cb36SFrederic Weisbecker }
2157f201ae23SFrederic Weisbecker 
2158f201ae23SFrederic Weisbecker /* Allocate a return stack for newly created task */
2159fb52607aSFrederic Weisbecker void ftrace_graph_init_task(struct task_struct *t)
2160f201ae23SFrederic Weisbecker {
2161287b6e68SFrederic Weisbecker 	if (atomic_read(&ftrace_graph_active)) {
2162f201ae23SFrederic Weisbecker 		t->ret_stack = kmalloc(FTRACE_RETFUNC_DEPTH
2163f201ae23SFrederic Weisbecker 				* sizeof(struct ftrace_ret_stack),
2164f201ae23SFrederic Weisbecker 				GFP_KERNEL);
2165f201ae23SFrederic Weisbecker 		if (!t->ret_stack)
2166f201ae23SFrederic Weisbecker 			return;
2167f201ae23SFrederic Weisbecker 		t->curr_ret_stack = -1;
2168380c4b14SFrederic Weisbecker 		atomic_set(&t->tracing_graph_pause, 0);
2169f201ae23SFrederic Weisbecker 		atomic_set(&t->trace_overrun, 0);
2170f201ae23SFrederic Weisbecker 	} else
2171f201ae23SFrederic Weisbecker 		t->ret_stack = NULL;
2172f201ae23SFrederic Weisbecker }
2173f201ae23SFrederic Weisbecker 
2174fb52607aSFrederic Weisbecker void ftrace_graph_exit_task(struct task_struct *t)
2175f201ae23SFrederic Weisbecker {
2176eae849caSFrederic Weisbecker 	struct ftrace_ret_stack	*ret_stack = t->ret_stack;
2177eae849caSFrederic Weisbecker 
2178f201ae23SFrederic Weisbecker 	t->ret_stack = NULL;
2179eae849caSFrederic Weisbecker 	/* NULL must become visible to IRQs before we free it: */
2180eae849caSFrederic Weisbecker 	barrier();
2181eae849caSFrederic Weisbecker 
2182eae849caSFrederic Weisbecker 	kfree(ret_stack);
2183f201ae23SFrederic Weisbecker }
218414a866c5SSteven Rostedt 
218514a866c5SSteven Rostedt void ftrace_graph_stop(void)
218614a866c5SSteven Rostedt {
218714a866c5SSteven Rostedt 	ftrace_stop();
218814a866c5SSteven Rostedt }
218915e6cb36SFrederic Weisbecker #endif
219015e6cb36SFrederic Weisbecker 
2191