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, ¬); 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