tracing: Create generic syscall TRACE_EVENTs
This converts the syscall_enter/exit tracepoints into TRACE_EVENTs, so you can have generic ftrace events that capture all system calls with arguments and return values. These generic events are also renamed to sys_enter/exit, so they're more closely aligned to the specific sys_enter_foo events. Signed-off-by: Josh Stone <jistone@redhat.com> Cc: Jason Baron <jbaron@redhat.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Ingo Molnar <mingo@elte.hu> Cc: Li Zefan <lizf@cn.fujitsu.com> Cc: Steven Rostedt <rostedt@goodmis.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Mathieu Desnoyers <mathieu.desnoyers@polymtl.ca> Cc: Jiaying Zhang <jiayingz@google.com> Cc: Martin Bligh <mbligh@google.com> Cc: Lai Jiangshan <laijs@cn.fujitsu.com> Cc: Paul Mundt <lethal@linux-sh.org> Cc: Martin Schwidefsky <schwidefsky@de.ibm.com> Cc: Heiko Carstens <heiko.carstens@de.ibm.com> LKML-Reference: <1251150194-1713-5-git-send-email-jistone@redhat.com> Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
This commit is contained in:
committed by
Frederic Weisbecker
parent
9741987586
commit
1c569f0264
@@ -51,8 +51,8 @@
|
|||||||
#include "compat_ptrace.h"
|
#include "compat_ptrace.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
DEFINE_TRACE_FN(syscall_enter, syscall_regfunc, syscall_unregfunc);
|
#define CREATE_TRACE_POINTS
|
||||||
DEFINE_TRACE_FN(syscall_exit, syscall_regfunc, syscall_unregfunc);
|
#include <trace/events/syscalls.h>
|
||||||
|
|
||||||
enum s390_regset {
|
enum s390_regset {
|
||||||
REGSET_GENERAL,
|
REGSET_GENERAL,
|
||||||
@@ -665,7 +665,7 @@ asmlinkage long do_syscall_trace_enter(struct pt_regs *regs)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
||||||
trace_syscall_enter(regs, regs->gprs[2]);
|
trace_sys_enter(regs, regs->gprs[2]);
|
||||||
|
|
||||||
if (unlikely(current->audit_context))
|
if (unlikely(current->audit_context))
|
||||||
audit_syscall_entry(is_compat_task() ?
|
audit_syscall_entry(is_compat_task() ?
|
||||||
@@ -683,7 +683,7 @@ asmlinkage void do_syscall_trace_exit(struct pt_regs *regs)
|
|||||||
regs->gprs[2]);
|
regs->gprs[2]);
|
||||||
|
|
||||||
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
||||||
trace_syscall_exit(regs, regs->gprs[2]);
|
trace_sys_exit(regs, regs->gprs[2]);
|
||||||
|
|
||||||
if (test_thread_flag(TIF_SYSCALL_TRACE))
|
if (test_thread_flag(TIF_SYSCALL_TRACE))
|
||||||
tracehook_report_syscall_exit(regs, 0);
|
tracehook_report_syscall_exit(regs, 0);
|
||||||
|
@@ -35,13 +35,11 @@
|
|||||||
#include <asm/proto.h>
|
#include <asm/proto.h>
|
||||||
#include <asm/ds.h>
|
#include <asm/ds.h>
|
||||||
|
|
||||||
#include <trace/syscall.h>
|
|
||||||
|
|
||||||
DEFINE_TRACE_FN(syscall_enter, syscall_regfunc, syscall_unregfunc);
|
|
||||||
DEFINE_TRACE_FN(syscall_exit, syscall_regfunc, syscall_unregfunc);
|
|
||||||
|
|
||||||
#include "tls.h"
|
#include "tls.h"
|
||||||
|
|
||||||
|
#define CREATE_TRACE_POINTS
|
||||||
|
#include <trace/events/syscalls.h>
|
||||||
|
|
||||||
enum x86_regset {
|
enum x86_regset {
|
||||||
REGSET_GENERAL,
|
REGSET_GENERAL,
|
||||||
REGSET_FP,
|
REGSET_FP,
|
||||||
@@ -1501,7 +1499,7 @@ asmregparm long syscall_trace_enter(struct pt_regs *regs)
|
|||||||
ret = -1L;
|
ret = -1L;
|
||||||
|
|
||||||
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
||||||
trace_syscall_enter(regs, regs->orig_ax);
|
trace_sys_enter(regs, regs->orig_ax);
|
||||||
|
|
||||||
if (unlikely(current->audit_context)) {
|
if (unlikely(current->audit_context)) {
|
||||||
if (IS_IA32)
|
if (IS_IA32)
|
||||||
@@ -1527,7 +1525,7 @@ asmregparm void syscall_trace_leave(struct pt_regs *regs)
|
|||||||
audit_syscall_exit(AUDITSC_RESULT(regs->ax), regs->ax);
|
audit_syscall_exit(AUDITSC_RESULT(regs->ax), regs->ax);
|
||||||
|
|
||||||
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
|
||||||
trace_syscall_exit(regs, regs->ax);
|
trace_sys_exit(regs, regs->ax);
|
||||||
|
|
||||||
if (test_thread_flag(TIF_SYSCALL_TRACE))
|
if (test_thread_flag(TIF_SYSCALL_TRACE))
|
||||||
tracehook_report_syscall_exit(regs, 0);
|
tracehook_report_syscall_exit(regs, 0);
|
||||||
|
70
include/trace/events/syscalls.h
Normal file
70
include/trace/events/syscalls.h
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
#undef TRACE_SYSTEM
|
||||||
|
#define TRACE_SYSTEM syscalls
|
||||||
|
|
||||||
|
#if !defined(_TRACE_EVENTS_SYSCALLS_H) || defined(TRACE_HEADER_MULTI_READ)
|
||||||
|
#define _TRACE_EVENTS_SYSCALLS_H
|
||||||
|
|
||||||
|
#include <linux/tracepoint.h>
|
||||||
|
|
||||||
|
#include <asm/ptrace.h>
|
||||||
|
#include <asm/syscall.h>
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef CONFIG_HAVE_SYSCALL_TRACEPOINTS
|
||||||
|
|
||||||
|
extern void syscall_regfunc(void);
|
||||||
|
extern void syscall_unregfunc(void);
|
||||||
|
|
||||||
|
TRACE_EVENT_FN(sys_enter,
|
||||||
|
|
||||||
|
TP_PROTO(struct pt_regs *regs, long id),
|
||||||
|
|
||||||
|
TP_ARGS(regs, id),
|
||||||
|
|
||||||
|
TP_STRUCT__entry(
|
||||||
|
__field( long, id )
|
||||||
|
__array( unsigned long, args, 6 )
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_fast_assign(
|
||||||
|
__entry->id = id;
|
||||||
|
syscall_get_arguments(current, regs, 0, 6, __entry->args);
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_printk("NR %ld (%lx, %lx, %lx, %lx, %lx, %lx)",
|
||||||
|
__entry->id,
|
||||||
|
__entry->args[0], __entry->args[1], __entry->args[2],
|
||||||
|
__entry->args[3], __entry->args[4], __entry->args[5]),
|
||||||
|
|
||||||
|
syscall_regfunc, syscall_unregfunc
|
||||||
|
);
|
||||||
|
|
||||||
|
TRACE_EVENT_FN(sys_exit,
|
||||||
|
|
||||||
|
TP_PROTO(struct pt_regs *regs, long ret),
|
||||||
|
|
||||||
|
TP_ARGS(regs, ret),
|
||||||
|
|
||||||
|
TP_STRUCT__entry(
|
||||||
|
__field( long, id )
|
||||||
|
__field( long, ret )
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_fast_assign(
|
||||||
|
__entry->id = syscall_get_nr(current, regs);
|
||||||
|
__entry->ret = ret;
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_printk("NR %ld = %ld",
|
||||||
|
__entry->id, __entry->ret),
|
||||||
|
|
||||||
|
syscall_regfunc, syscall_unregfunc
|
||||||
|
);
|
||||||
|
|
||||||
|
#endif /* CONFIG_HAVE_SYSCALL_TRACEPOINTS */
|
||||||
|
|
||||||
|
#endif /* _TRACE_EVENTS_SYSCALLS_H */
|
||||||
|
|
||||||
|
/* This part must be outside protection */
|
||||||
|
#include <trace/define_trace.h>
|
||||||
|
|
@@ -8,23 +8,6 @@
|
|||||||
#include <asm/ptrace.h>
|
#include <asm/ptrace.h>
|
||||||
|
|
||||||
|
|
||||||
#ifdef CONFIG_HAVE_SYSCALL_TRACEPOINTS
|
|
||||||
|
|
||||||
extern void syscall_regfunc(void);
|
|
||||||
extern void syscall_unregfunc(void);
|
|
||||||
|
|
||||||
DECLARE_TRACE(syscall_enter,
|
|
||||||
TP_PROTO(struct pt_regs *regs, long id),
|
|
||||||
TP_ARGS(regs, id)
|
|
||||||
);
|
|
||||||
|
|
||||||
DECLARE_TRACE(syscall_exit,
|
|
||||||
TP_PROTO(struct pt_regs *regs, long ret),
|
|
||||||
TP_ARGS(regs, ret)
|
|
||||||
);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A syscall entry in the ftrace syscalls array.
|
* A syscall entry in the ftrace syscalls array.
|
||||||
*
|
*
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
#include <trace/syscall.h>
|
#include <trace/syscall.h>
|
||||||
|
#include <trace/events/syscalls.h>
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/ftrace.h>
|
#include <linux/ftrace.h>
|
||||||
#include <linux/perf_counter.h>
|
#include <linux/perf_counter.h>
|
||||||
@@ -286,7 +287,7 @@ int reg_event_syscall_enter(void *ptr)
|
|||||||
return -ENOSYS;
|
return -ENOSYS;
|
||||||
mutex_lock(&syscall_trace_lock);
|
mutex_lock(&syscall_trace_lock);
|
||||||
if (!sys_refcount_enter)
|
if (!sys_refcount_enter)
|
||||||
ret = register_trace_syscall_enter(ftrace_syscall_enter);
|
ret = register_trace_sys_enter(ftrace_syscall_enter);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
pr_info("event trace: Could not activate"
|
pr_info("event trace: Could not activate"
|
||||||
"syscall entry trace point");
|
"syscall entry trace point");
|
||||||
@@ -311,7 +312,7 @@ void unreg_event_syscall_enter(void *ptr)
|
|||||||
sys_refcount_enter--;
|
sys_refcount_enter--;
|
||||||
clear_bit(num, enabled_enter_syscalls);
|
clear_bit(num, enabled_enter_syscalls);
|
||||||
if (!sys_refcount_enter)
|
if (!sys_refcount_enter)
|
||||||
unregister_trace_syscall_enter(ftrace_syscall_enter);
|
unregister_trace_sys_enter(ftrace_syscall_enter);
|
||||||
mutex_unlock(&syscall_trace_lock);
|
mutex_unlock(&syscall_trace_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -327,7 +328,7 @@ int reg_event_syscall_exit(void *ptr)
|
|||||||
return -ENOSYS;
|
return -ENOSYS;
|
||||||
mutex_lock(&syscall_trace_lock);
|
mutex_lock(&syscall_trace_lock);
|
||||||
if (!sys_refcount_exit)
|
if (!sys_refcount_exit)
|
||||||
ret = register_trace_syscall_exit(ftrace_syscall_exit);
|
ret = register_trace_sys_exit(ftrace_syscall_exit);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
pr_info("event trace: Could not activate"
|
pr_info("event trace: Could not activate"
|
||||||
"syscall exit trace point");
|
"syscall exit trace point");
|
||||||
@@ -352,7 +353,7 @@ void unreg_event_syscall_exit(void *ptr)
|
|||||||
sys_refcount_exit--;
|
sys_refcount_exit--;
|
||||||
clear_bit(num, enabled_exit_syscalls);
|
clear_bit(num, enabled_exit_syscalls);
|
||||||
if (!sys_refcount_exit)
|
if (!sys_refcount_exit)
|
||||||
unregister_trace_syscall_exit(ftrace_syscall_exit);
|
unregister_trace_sys_exit(ftrace_syscall_exit);
|
||||||
mutex_unlock(&syscall_trace_lock);
|
mutex_unlock(&syscall_trace_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -418,7 +419,7 @@ int reg_prof_syscall_enter(char *name)
|
|||||||
|
|
||||||
mutex_lock(&syscall_trace_lock);
|
mutex_lock(&syscall_trace_lock);
|
||||||
if (!sys_prof_refcount_enter)
|
if (!sys_prof_refcount_enter)
|
||||||
ret = register_trace_syscall_enter(prof_syscall_enter);
|
ret = register_trace_sys_enter(prof_syscall_enter);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
pr_info("event trace: Could not activate"
|
pr_info("event trace: Could not activate"
|
||||||
"syscall entry trace point");
|
"syscall entry trace point");
|
||||||
@@ -442,7 +443,7 @@ void unreg_prof_syscall_enter(char *name)
|
|||||||
sys_prof_refcount_enter--;
|
sys_prof_refcount_enter--;
|
||||||
clear_bit(num, enabled_prof_enter_syscalls);
|
clear_bit(num, enabled_prof_enter_syscalls);
|
||||||
if (!sys_prof_refcount_enter)
|
if (!sys_prof_refcount_enter)
|
||||||
unregister_trace_syscall_enter(prof_syscall_enter);
|
unregister_trace_sys_enter(prof_syscall_enter);
|
||||||
mutex_unlock(&syscall_trace_lock);
|
mutex_unlock(&syscall_trace_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -479,7 +480,7 @@ int reg_prof_syscall_exit(char *name)
|
|||||||
|
|
||||||
mutex_lock(&syscall_trace_lock);
|
mutex_lock(&syscall_trace_lock);
|
||||||
if (!sys_prof_refcount_exit)
|
if (!sys_prof_refcount_exit)
|
||||||
ret = register_trace_syscall_exit(prof_syscall_exit);
|
ret = register_trace_sys_exit(prof_syscall_exit);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
pr_info("event trace: Could not activate"
|
pr_info("event trace: Could not activate"
|
||||||
"syscall entry trace point");
|
"syscall entry trace point");
|
||||||
@@ -503,7 +504,7 @@ void unreg_prof_syscall_exit(char *name)
|
|||||||
sys_prof_refcount_exit--;
|
sys_prof_refcount_exit--;
|
||||||
clear_bit(num, enabled_prof_exit_syscalls);
|
clear_bit(num, enabled_prof_exit_syscalls);
|
||||||
if (!sys_prof_refcount_exit)
|
if (!sys_prof_refcount_exit)
|
||||||
unregister_trace_syscall_exit(prof_syscall_exit);
|
unregister_trace_sys_exit(prof_syscall_exit);
|
||||||
mutex_unlock(&syscall_trace_lock);
|
mutex_unlock(&syscall_trace_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user