diff options
Diffstat (limited to 'arch/x86/include/asm')
-rw-r--r-- | arch/x86/include/asm/Kbuild | 1 | ||||
-rw-r--r-- | arch/x86/include/asm/debugreg.h | 11 | ||||
-rw-r--r-- | arch/x86/include/asm/hw_breakpoint.h | 58 | ||||
-rw-r--r-- | arch/x86/include/asm/processor.h | 12 |
4 files changed, 50 insertions, 32 deletions
diff --git a/arch/x86/include/asm/Kbuild b/arch/x86/include/asm/Kbuild index 4a8e80cdcfa5..9f828f87ca35 100644 --- a/arch/x86/include/asm/Kbuild +++ b/arch/x86/include/asm/Kbuild | |||
@@ -10,6 +10,7 @@ header-y += ptrace-abi.h | |||
10 | header-y += sigcontext32.h | 10 | header-y += sigcontext32.h |
11 | header-y += ucontext.h | 11 | header-y += ucontext.h |
12 | header-y += processor-flags.h | 12 | header-y += processor-flags.h |
13 | header-y += hw_breakpoint.h | ||
13 | 14 | ||
14 | unifdef-y += e820.h | 15 | unifdef-y += e820.h |
15 | unifdef-y += ist.h | 16 | unifdef-y += ist.h |
diff --git a/arch/x86/include/asm/debugreg.h b/arch/x86/include/asm/debugreg.h index 23439fbb1d0e..9a3333c91f9a 100644 --- a/arch/x86/include/asm/debugreg.h +++ b/arch/x86/include/asm/debugreg.h | |||
@@ -75,13 +75,8 @@ | |||
75 | */ | 75 | */ |
76 | #ifdef __KERNEL__ | 76 | #ifdef __KERNEL__ |
77 | 77 | ||
78 | /* For process management */ | 78 | DECLARE_PER_CPU(unsigned long, dr7); |
79 | extern void flush_thread_hw_breakpoint(struct task_struct *tsk); | ||
80 | extern int copy_thread_hw_breakpoint(struct task_struct *tsk, | ||
81 | struct task_struct *child, unsigned long clone_flags); | ||
82 | 79 | ||
83 | /* For CPU management */ | ||
84 | extern void load_debug_registers(void); | ||
85 | static inline void hw_breakpoint_disable(void) | 80 | static inline void hw_breakpoint_disable(void) |
86 | { | 81 | { |
87 | /* Zero the control register for HW Breakpoint */ | 82 | /* Zero the control register for HW Breakpoint */ |
@@ -94,6 +89,10 @@ static inline void hw_breakpoint_disable(void) | |||
94 | set_debugreg(0UL, 3); | 89 | set_debugreg(0UL, 3); |
95 | } | 90 | } |
96 | 91 | ||
92 | #ifdef CONFIG_KVM | ||
93 | extern void hw_breakpoint_restore(void); | ||
94 | #endif | ||
95 | |||
97 | #endif /* __KERNEL__ */ | 96 | #endif /* __KERNEL__ */ |
98 | 97 | ||
99 | #endif /* _ASM_X86_DEBUGREG_H */ | 98 | #endif /* _ASM_X86_DEBUGREG_H */ |
diff --git a/arch/x86/include/asm/hw_breakpoint.h b/arch/x86/include/asm/hw_breakpoint.h index 3cfca8e2b5f6..0675a7c4c20e 100644 --- a/arch/x86/include/asm/hw_breakpoint.h +++ b/arch/x86/include/asm/hw_breakpoint.h | |||
@@ -4,6 +4,11 @@ | |||
4 | #ifdef __KERNEL__ | 4 | #ifdef __KERNEL__ |
5 | #define __ARCH_HW_BREAKPOINT_H | 5 | #define __ARCH_HW_BREAKPOINT_H |
6 | 6 | ||
7 | /* | ||
8 | * The name should probably be something dealt in | ||
9 | * a higher level. While dealing with the user | ||
10 | * (display/resolving) | ||
11 | */ | ||
7 | struct arch_hw_breakpoint { | 12 | struct arch_hw_breakpoint { |
8 | char *name; /* Contains name of the symbol to set bkpt */ | 13 | char *name; /* Contains name of the symbol to set bkpt */ |
9 | unsigned long address; | 14 | unsigned long address; |
@@ -12,44 +17,57 @@ struct arch_hw_breakpoint { | |||
12 | }; | 17 | }; |
13 | 18 | ||
14 | #include <linux/kdebug.h> | 19 | #include <linux/kdebug.h> |
15 | #include <linux/hw_breakpoint.h> | 20 | #include <linux/percpu.h> |
21 | #include <linux/list.h> | ||
16 | 22 | ||
17 | /* Available HW breakpoint length encodings */ | 23 | /* Available HW breakpoint length encodings */ |
18 | #define HW_BREAKPOINT_LEN_1 0x40 | 24 | #define X86_BREAKPOINT_LEN_1 0x40 |
19 | #define HW_BREAKPOINT_LEN_2 0x44 | 25 | #define X86_BREAKPOINT_LEN_2 0x44 |
20 | #define HW_BREAKPOINT_LEN_4 0x4c | 26 | #define X86_BREAKPOINT_LEN_4 0x4c |
21 | #define HW_BREAKPOINT_LEN_EXECUTE 0x40 | 27 | #define X86_BREAKPOINT_LEN_EXECUTE 0x40 |
22 | 28 | ||
23 | #ifdef CONFIG_X86_64 | 29 | #ifdef CONFIG_X86_64 |
24 | #define HW_BREAKPOINT_LEN_8 0x48 | 30 | #define X86_BREAKPOINT_LEN_8 0x48 |
25 | #endif | 31 | #endif |
26 | 32 | ||
27 | /* Available HW breakpoint type encodings */ | 33 | /* Available HW breakpoint type encodings */ |
28 | 34 | ||
29 | /* trigger on instruction execute */ | 35 | /* trigger on instruction execute */ |
30 | #define HW_BREAKPOINT_EXECUTE 0x80 | 36 | #define X86_BREAKPOINT_EXECUTE 0x80 |
31 | /* trigger on memory write */ | 37 | /* trigger on memory write */ |
32 | #define HW_BREAKPOINT_WRITE 0x81 | 38 | #define X86_BREAKPOINT_WRITE 0x81 |
33 | /* trigger on memory read or write */ | 39 | /* trigger on memory read or write */ |
34 | #define HW_BREAKPOINT_RW 0x83 | 40 | #define X86_BREAKPOINT_RW 0x83 |
35 | 41 | ||
36 | /* Total number of available HW breakpoint registers */ | 42 | /* Total number of available HW breakpoint registers */ |
37 | #define HBP_NUM 4 | 43 | #define HBP_NUM 4 |
38 | 44 | ||
39 | extern struct hw_breakpoint *hbp_kernel[HBP_NUM]; | 45 | struct perf_event; |
40 | DECLARE_PER_CPU(struct hw_breakpoint*, this_hbp_kernel[HBP_NUM]); | 46 | struct pmu; |
41 | extern unsigned int hbp_user_refcount[HBP_NUM]; | ||
42 | 47 | ||
43 | extern void arch_install_thread_hw_breakpoint(struct task_struct *tsk); | ||
44 | extern void arch_uninstall_thread_hw_breakpoint(void); | ||
45 | extern int arch_check_va_in_userspace(unsigned long va, u8 hbp_len); | 48 | extern int arch_check_va_in_userspace(unsigned long va, u8 hbp_len); |
46 | extern int arch_validate_hwbkpt_settings(struct hw_breakpoint *bp, | 49 | extern int arch_validate_hwbkpt_settings(struct perf_event *bp, |
47 | struct task_struct *tsk); | 50 | struct task_struct *tsk); |
48 | extern void arch_update_user_hw_breakpoint(int pos, struct task_struct *tsk); | ||
49 | extern void arch_flush_thread_hw_breakpoint(struct task_struct *tsk); | ||
50 | extern void arch_update_kernel_hw_breakpoint(void *); | ||
51 | extern int hw_breakpoint_exceptions_notify(struct notifier_block *unused, | 51 | extern int hw_breakpoint_exceptions_notify(struct notifier_block *unused, |
52 | unsigned long val, void *data); | 52 | unsigned long val, void *data); |
53 | |||
54 | |||
55 | int arch_install_hw_breakpoint(struct perf_event *bp); | ||
56 | void arch_uninstall_hw_breakpoint(struct perf_event *bp); | ||
57 | void hw_breakpoint_pmu_read(struct perf_event *bp); | ||
58 | void hw_breakpoint_pmu_unthrottle(struct perf_event *bp); | ||
59 | |||
60 | extern void | ||
61 | arch_fill_perf_breakpoint(struct perf_event *bp); | ||
62 | |||
63 | unsigned long encode_dr7(int drnum, unsigned int len, unsigned int type); | ||
64 | int decode_dr7(unsigned long dr7, int bpnum, unsigned *len, unsigned *type); | ||
65 | |||
66 | extern int arch_bp_generic_fields(int x86_len, int x86_type, | ||
67 | int *gen_len, int *gen_type); | ||
68 | |||
69 | extern struct pmu perf_ops_bp; | ||
70 | |||
53 | #endif /* __KERNEL__ */ | 71 | #endif /* __KERNEL__ */ |
54 | #endif /* _I386_HW_BREAKPOINT_H */ | 72 | #endif /* _I386_HW_BREAKPOINT_H */ |
55 | 73 | ||
diff --git a/arch/x86/include/asm/processor.h b/arch/x86/include/asm/processor.h index 61aafb71c7ef..820f3000f736 100644 --- a/arch/x86/include/asm/processor.h +++ b/arch/x86/include/asm/processor.h | |||
@@ -423,6 +423,8 @@ extern unsigned int xstate_size; | |||
423 | extern void free_thread_xstate(struct task_struct *); | 423 | extern void free_thread_xstate(struct task_struct *); |
424 | extern struct kmem_cache *task_xstate_cachep; | 424 | extern struct kmem_cache *task_xstate_cachep; |
425 | 425 | ||
426 | struct perf_event; | ||
427 | |||
426 | struct thread_struct { | 428 | struct thread_struct { |
427 | /* Cached TLS descriptors: */ | 429 | /* Cached TLS descriptors: */ |
428 | struct desc_struct tls_array[GDT_ENTRY_TLS_ENTRIES]; | 430 | struct desc_struct tls_array[GDT_ENTRY_TLS_ENTRIES]; |
@@ -444,12 +446,10 @@ struct thread_struct { | |||
444 | unsigned long fs; | 446 | unsigned long fs; |
445 | #endif | 447 | #endif |
446 | unsigned long gs; | 448 | unsigned long gs; |
447 | /* Hardware debugging registers: */ | 449 | /* Save middle states of ptrace breakpoints */ |
448 | unsigned long debugreg[HBP_NUM]; | 450 | struct perf_event *ptrace_bps[HBP_NUM]; |
449 | unsigned long debugreg6; | 451 | /* Debug status used for traps, single steps, etc... */ |
450 | unsigned long debugreg7; | 452 | unsigned long debugreg6; |
451 | /* Hardware breakpoint info */ | ||
452 | struct hw_breakpoint *hbp[HBP_NUM]; | ||
453 | /* Fault info: */ | 453 | /* Fault info: */ |
454 | unsigned long cr2; | 454 | unsigned long cr2; |
455 | unsigned long trap_no; | 455 | unsigned long trap_no; |