aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/ftrace.c
diff options
context:
space:
mode:
authorFrederic Weisbecker <fweisbec@gmail.com>2008-12-01 18:20:39 -0500
committerIngo Molnar <mingo@elte.hu>2008-12-02 03:47:48 -0500
commit48d68b20d00865035b8b65e69af343d0f53fac9d (patch)
treeeca27c0cf9486ae83c7e3193709abae099d1f019 /arch/x86/kernel/ftrace.c
parent222658e08f72cd539d01f3aabdc258c596f487e2 (diff)
tracing/function-graph-tracer: support for x86-64
Impact: extend and enable the function graph tracer to 64-bit x86 This patch implements the support for function graph tracer under x86-64. Both static and dynamic tracing are supported. This causes some small CPP conditional asm on arch/x86/kernel/ftrace.c I wanted to use probe_kernel_read/write to make the return address saving/patching code more generic but it causes tracing recursion. That would be perhaps useful to implement a notrace version of these function for other archs ports. Note that arch/x86/process_64.c is not traced, as in X86-32. I first thought __switch_to() was responsible of crashes during tracing because I believed current task were changed inside but that's actually not the case (actually yes, but not the "current" pointer). So I will have to investigate to find the functions that harm here, to enable tracing of the other functions inside (but there is no issue at this time, while process_64.c stays out of -pg flags). A little possible race condition is fixed inside this patch too. When the tracer allocate a return stack dynamically, the current depth is not initialized before but after. An interrupt could occur at this time and, after seeing that the return stack is allocated, the tracer could try to trace it with a random uninitialized depth. It's a prevention, even if I hadn't problems with it. Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com> Cc: Steven Rostedt <rostedt@goodmis.org> Cc: Tim Bird <tim.bird@am.sony.com> Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86/kernel/ftrace.c')
-rw-r--r--arch/x86/kernel/ftrace.c11
1 files changed, 10 insertions, 1 deletions
diff --git a/arch/x86/kernel/ftrace.c b/arch/x86/kernel/ftrace.c
index 7ef914e6a2f6..58832478b94e 100644
--- a/arch/x86/kernel/ftrace.c
+++ b/arch/x86/kernel/ftrace.c
@@ -467,8 +467,13 @@ void prepare_ftrace_return(unsigned long *parent, unsigned long self_addr)
467 * ignore such a protection. 467 * ignore such a protection.
468 */ 468 */
469 asm volatile( 469 asm volatile(
470#ifdef CONFIG_X86_64
471 "1: movq (%[parent_old]), %[old]\n"
472 "2: movq %[return_hooker], (%[parent_replaced])\n"
473#else
470 "1: movl (%[parent_old]), %[old]\n" 474 "1: movl (%[parent_old]), %[old]\n"
471 "2: movl %[return_hooker], (%[parent_replaced])\n" 475 "2: movl %[return_hooker], (%[parent_replaced])\n"
476#endif
472 " movl $0, %[faulted]\n" 477 " movl $0, %[faulted]\n"
473 478
474 ".section .fixup, \"ax\"\n" 479 ".section .fixup, \"ax\"\n"
@@ -476,8 +481,13 @@ void prepare_ftrace_return(unsigned long *parent, unsigned long self_addr)
476 ".previous\n" 481 ".previous\n"
477 482
478 ".section __ex_table, \"a\"\n" 483 ".section __ex_table, \"a\"\n"
484#ifdef CONFIG_X86_64
485 " .quad 1b, 3b\n"
486 " .quad 2b, 3b\n"
487#else
479 " .long 1b, 3b\n" 488 " .long 1b, 3b\n"
480 " .long 2b, 3b\n" 489 " .long 2b, 3b\n"
490#endif
481 ".previous\n" 491 ".previous\n"
482 492
483 : [parent_replaced] "=r" (parent), [old] "=r" (old), 493 : [parent_replaced] "=r" (parent), [old] "=r" (old),
@@ -509,5 +519,4 @@ void prepare_ftrace_return(unsigned long *parent, unsigned long self_addr)
509 ftrace_graph_entry(&trace); 519 ftrace_graph_entry(&trace);
510 520
511} 521}
512
513#endif /* CONFIG_FUNCTION_GRAPH_TRACER */ 522#endif /* CONFIG_FUNCTION_GRAPH_TRACER */