-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
perf: Unified API to record selective sets of arch registers
This brings a new API to help the selective dump of registers on event sampling, and its implementation for x86 arch. Added HAVE_PERF_REGS config option to determine if the architecture provides perf registers ABI. The information about desired registers will be passed in u64 mask. It's up to the architecture to map the registers into the mask bits. For the x86 arch implementation, both 32 and 64 bit registers bits are defined within single enum to ensure 64 bit system can provide register dump for compat task if needed in the future. Original-patch-by: Frederic Weisbecker <fweisbec@gmail.com> [ Added missing linux/errno.h include ] Signed-off-by: Jiri Olsa <jolsa@redhat.com> Cc: "Frank Ch. Eigler" <fche@redhat.com> Cc: Arun Sharma <asharma@fb.com> Cc: Benjamin Redelings <benjamin.redelings@nescent.org> Cc: Corey Ashford <cjashfor@linux.vnet.ibm.com> Cc: Cyrill Gorcunov <gorcunov@openvz.org> Cc: Frank Ch. Eigler <fche@redhat.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Ingo Molnar <mingo@kernel.org> Cc: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <a.p.zijlstra@chello.nl> Cc: Robert Richter <robert.richter@amd.com> Cc: Stephane Eranian <eranian@google.com> Cc: Tom Zanussi <tzanussi@gmail.com> Cc: Ulrich Drepper <drepper@gmail.com> Link: http://lkml.kernel.org/r/1344345647-11536-2-git-send-email-jolsa@redhat.com Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
- Loading branch information
Jiri Olsa
authored and
Arnaldo Carvalho de Melo
committed
Aug 10, 2012
1 parent
b691f64
commit c5e6319
Showing
6 changed files
with
151 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
#ifndef _ASM_X86_PERF_REGS_H | ||
#define _ASM_X86_PERF_REGS_H | ||
|
||
enum perf_event_x86_regs { | ||
PERF_REG_X86_AX, | ||
PERF_REG_X86_BX, | ||
PERF_REG_X86_CX, | ||
PERF_REG_X86_DX, | ||
PERF_REG_X86_SI, | ||
PERF_REG_X86_DI, | ||
PERF_REG_X86_BP, | ||
PERF_REG_X86_SP, | ||
PERF_REG_X86_IP, | ||
PERF_REG_X86_FLAGS, | ||
PERF_REG_X86_CS, | ||
PERF_REG_X86_SS, | ||
PERF_REG_X86_DS, | ||
PERF_REG_X86_ES, | ||
PERF_REG_X86_FS, | ||
PERF_REG_X86_GS, | ||
PERF_REG_X86_R8, | ||
PERF_REG_X86_R9, | ||
PERF_REG_X86_R10, | ||
PERF_REG_X86_R11, | ||
PERF_REG_X86_R12, | ||
PERF_REG_X86_R13, | ||
PERF_REG_X86_R14, | ||
PERF_REG_X86_R15, | ||
|
||
PERF_REG_X86_32_MAX = PERF_REG_X86_GS + 1, | ||
PERF_REG_X86_64_MAX = PERF_REG_X86_R15 + 1, | ||
}; | ||
#endif /* _ASM_X86_PERF_REGS_H */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,90 @@ | ||
#include <linux/errno.h> | ||
#include <linux/kernel.h> | ||
#include <linux/bug.h> | ||
#include <linux/stddef.h> | ||
#include <asm/perf_regs.h> | ||
#include <asm/ptrace.h> | ||
|
||
#ifdef CONFIG_X86_32 | ||
#define PERF_REG_X86_MAX PERF_REG_X86_32_MAX | ||
#else | ||
#define PERF_REG_X86_MAX PERF_REG_X86_64_MAX | ||
#endif | ||
|
||
#define PT_REGS_OFFSET(id, r) [id] = offsetof(struct pt_regs, r) | ||
|
||
static unsigned int pt_regs_offset[PERF_REG_X86_MAX] = { | ||
PT_REGS_OFFSET(PERF_REG_X86_AX, ax), | ||
PT_REGS_OFFSET(PERF_REG_X86_BX, bx), | ||
PT_REGS_OFFSET(PERF_REG_X86_CX, cx), | ||
PT_REGS_OFFSET(PERF_REG_X86_DX, dx), | ||
PT_REGS_OFFSET(PERF_REG_X86_SI, si), | ||
PT_REGS_OFFSET(PERF_REG_X86_DI, di), | ||
PT_REGS_OFFSET(PERF_REG_X86_BP, bp), | ||
PT_REGS_OFFSET(PERF_REG_X86_SP, sp), | ||
PT_REGS_OFFSET(PERF_REG_X86_IP, ip), | ||
PT_REGS_OFFSET(PERF_REG_X86_FLAGS, flags), | ||
PT_REGS_OFFSET(PERF_REG_X86_CS, cs), | ||
PT_REGS_OFFSET(PERF_REG_X86_SS, ss), | ||
#ifdef CONFIG_X86_32 | ||
PT_REGS_OFFSET(PERF_REG_X86_DS, ds), | ||
PT_REGS_OFFSET(PERF_REG_X86_ES, es), | ||
PT_REGS_OFFSET(PERF_REG_X86_FS, fs), | ||
PT_REGS_OFFSET(PERF_REG_X86_GS, gs), | ||
#else | ||
/* | ||
* The pt_regs struct does not store | ||
* ds, es, fs, gs in 64 bit mode. | ||
*/ | ||
(unsigned int) -1, | ||
(unsigned int) -1, | ||
(unsigned int) -1, | ||
(unsigned int) -1, | ||
#endif | ||
#ifdef CONFIG_X86_64 | ||
PT_REGS_OFFSET(PERF_REG_X86_R8, r8), | ||
PT_REGS_OFFSET(PERF_REG_X86_R9, r9), | ||
PT_REGS_OFFSET(PERF_REG_X86_R10, r10), | ||
PT_REGS_OFFSET(PERF_REG_X86_R11, r11), | ||
PT_REGS_OFFSET(PERF_REG_X86_R12, r12), | ||
PT_REGS_OFFSET(PERF_REG_X86_R13, r13), | ||
PT_REGS_OFFSET(PERF_REG_X86_R14, r14), | ||
PT_REGS_OFFSET(PERF_REG_X86_R15, r15), | ||
#endif | ||
}; | ||
|
||
u64 perf_reg_value(struct pt_regs *regs, int idx) | ||
{ | ||
if (WARN_ON_ONCE(idx > ARRAY_SIZE(pt_regs_offset))) | ||
return 0; | ||
|
||
return regs_get_register(regs, pt_regs_offset[idx]); | ||
} | ||
|
||
#define REG_RESERVED (~((1ULL << PERF_REG_X86_MAX) - 1ULL)) | ||
|
||
#ifdef CONFIG_X86_32 | ||
int perf_reg_validate(u64 mask) | ||
{ | ||
if (!mask || mask & REG_RESERVED) | ||
return -EINVAL; | ||
|
||
return 0; | ||
} | ||
#else /* CONFIG_X86_64 */ | ||
#define REG_NOSUPPORT ((1ULL << PERF_REG_X86_DS) | \ | ||
(1ULL << PERF_REG_X86_ES) | \ | ||
(1ULL << PERF_REG_X86_FS) | \ | ||
(1ULL << PERF_REG_X86_GS)) | ||
|
||
int perf_reg_validate(u64 mask) | ||
{ | ||
if (!mask || mask & REG_RESERVED) | ||
return -EINVAL; | ||
|
||
if (mask & REG_NOSUPPORT) | ||
return -EINVAL; | ||
|
||
return 0; | ||
} | ||
#endif /* CONFIG_X86_32 */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
#ifndef _LINUX_PERF_REGS_H | ||
#define _LINUX_PERF_REGS_H | ||
|
||
#ifdef CONFIG_HAVE_PERF_REGS | ||
#include <asm/perf_regs.h> | ||
u64 perf_reg_value(struct pt_regs *regs, int idx); | ||
int perf_reg_validate(u64 mask); | ||
#else | ||
static inline u64 perf_reg_value(struct pt_regs *regs, int idx) | ||
{ | ||
return 0; | ||
} | ||
|
||
static inline int perf_reg_validate(u64 mask) | ||
{ | ||
return mask ? -ENOSYS : 0; | ||
} | ||
#endif /* CONFIG_HAVE_PERF_REGS */ | ||
#endif /* _LINUX_PERF_REGS_H */ |