diff options
author | Steven Rostedt <srostedt@redhat.com> | 2008-05-12 15:20:43 -0400 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2008-05-23 14:33:28 -0400 |
commit | dfa60aba04dae7833d75b2e2be124bb7cfb8239f (patch) | |
tree | 6aebf657355b5e2132e94420712f55e52eb35c62 | |
parent | b0fc494fae96a7089f3651cb451f461c7291244c (diff) |
ftrace: use nops instead of jmp
This patch patches the call to mcount with nops instead
of a jmp over the mcount call.
Signed-off-by: Steven Rostedt <srostedt@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r-- | arch/x86/kernel/alternative.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/ftrace.c | 40 | ||||
-rw-r--r-- | include/asm-x86/alternative.h | 2 |
3 files changed, 28 insertions, 18 deletions
diff --git a/arch/x86/kernel/alternative.c b/arch/x86/kernel/alternative.c index 65c7857a90dd..de240ba2e288 100644 --- a/arch/x86/kernel/alternative.c +++ b/arch/x86/kernel/alternative.c | |||
@@ -143,7 +143,7 @@ static const unsigned char *const p6_nops[ASM_NOP_MAX+1] = { | |||
143 | #ifdef CONFIG_X86_64 | 143 | #ifdef CONFIG_X86_64 |
144 | 144 | ||
145 | extern char __vsyscall_0; | 145 | extern char __vsyscall_0; |
146 | static inline const unsigned char*const * find_nop_table(void) | 146 | const unsigned char *const *find_nop_table(void) |
147 | { | 147 | { |
148 | return boot_cpu_data.x86_vendor != X86_VENDOR_INTEL || | 148 | return boot_cpu_data.x86_vendor != X86_VENDOR_INTEL || |
149 | boot_cpu_data.x86 < 6 ? k8_nops : p6_nops; | 149 | boot_cpu_data.x86 < 6 ? k8_nops : p6_nops; |
@@ -162,7 +162,7 @@ static const struct nop { | |||
162 | { -1, NULL } | 162 | { -1, NULL } |
163 | }; | 163 | }; |
164 | 164 | ||
165 | static const unsigned char*const * find_nop_table(void) | 165 | const unsigned char *const *find_nop_table(void) |
166 | { | 166 | { |
167 | const unsigned char *const *noptable = intel_nops; | 167 | const unsigned char *const *noptable = intel_nops; |
168 | int i; | 168 | int i; |
diff --git a/arch/x86/kernel/ftrace.c b/arch/x86/kernel/ftrace.c index 5dd58136ef02..2e060c58b860 100644 --- a/arch/x86/kernel/ftrace.c +++ b/arch/x86/kernel/ftrace.c | |||
@@ -16,11 +16,12 @@ | |||
16 | #include <linux/init.h> | 16 | #include <linux/init.h> |
17 | #include <linux/list.h> | 17 | #include <linux/list.h> |
18 | 18 | ||
19 | #define CALL_BACK 5 | 19 | #include <asm/alternative.h> |
20 | 20 | ||
21 | #define JMPFWD 0x03eb | 21 | #define CALL_BACK 5 |
22 | 22 | ||
23 | static unsigned short ftrace_jmp = JMPFWD; | 23 | /* Long is fine, even if it is only 4 bytes ;-) */ |
24 | static long *ftrace_nop; | ||
24 | 25 | ||
25 | struct ftrace_record { | 26 | struct ftrace_record { |
26 | struct dyn_ftrace rec; | 27 | struct dyn_ftrace rec; |
@@ -55,13 +56,13 @@ static struct ftrace_page *ftrace_pages; | |||
55 | notrace struct dyn_ftrace *ftrace_alloc_shutdown_node(unsigned long ip) | 56 | notrace struct dyn_ftrace *ftrace_alloc_shutdown_node(unsigned long ip) |
56 | { | 57 | { |
57 | struct ftrace_record *rec; | 58 | struct ftrace_record *rec; |
58 | unsigned short save; | 59 | unsigned long save; |
59 | 60 | ||
60 | ip -= CALL_BACK; | 61 | ip -= CALL_BACK; |
61 | save = *(short *)ip; | 62 | save = *(long *)ip; |
62 | 63 | ||
63 | /* If this was already converted, skip it */ | 64 | /* If this was already converted, skip it */ |
64 | if (save == JMPFWD) | 65 | if (save == *ftrace_nop) |
65 | return NULL; | 66 | return NULL; |
66 | 67 | ||
67 | if (ftrace_pages->index == ENTRIES_PER_PAGE) { | 68 | if (ftrace_pages->index == ENTRIES_PER_PAGE) { |
@@ -79,9 +80,10 @@ static int notrace | |||
79 | ftrace_modify_code(unsigned long ip, unsigned char *old_code, | 80 | ftrace_modify_code(unsigned long ip, unsigned char *old_code, |
80 | unsigned char *new_code) | 81 | unsigned char *new_code) |
81 | { | 82 | { |
82 | unsigned short old = *(unsigned short *)old_code; | 83 | unsigned replaced; |
83 | unsigned short new = *(unsigned short *)new_code; | 84 | unsigned old = *(unsigned *)old_code; /* 4 bytes */ |
84 | unsigned short replaced; | 85 | unsigned new = *(unsigned *)new_code; /* 4 bytes */ |
86 | unsigned char newch = new_code[4]; | ||
85 | int faulted = 0; | 87 | int faulted = 0; |
86 | 88 | ||
87 | /* | 89 | /* |
@@ -94,7 +96,9 @@ ftrace_modify_code(unsigned long ip, unsigned char *old_code, | |||
94 | */ | 96 | */ |
95 | asm volatile ( | 97 | asm volatile ( |
96 | "1: lock\n" | 98 | "1: lock\n" |
97 | " cmpxchg %w3, (%2)\n" | 99 | " cmpxchg %3, (%2)\n" |
100 | " jnz 2f\n" | ||
101 | " movb %b4, 4(%2)\n" | ||
98 | "2:\n" | 102 | "2:\n" |
99 | ".section .fixup, \"ax\"\n" | 103 | ".section .fixup, \"ax\"\n" |
100 | " movl $1, %0\n" | 104 | " movl $1, %0\n" |
@@ -102,11 +106,12 @@ ftrace_modify_code(unsigned long ip, unsigned char *old_code, | |||
102 | ".previous\n" | 106 | ".previous\n" |
103 | _ASM_EXTABLE(1b, 3b) | 107 | _ASM_EXTABLE(1b, 3b) |
104 | : "=r"(faulted), "=a"(replaced) | 108 | : "=r"(faulted), "=a"(replaced) |
105 | : "r"(ip), "r"(new), "0"(faulted), "a"(old) | 109 | : "r"(ip), "r"(new), "r"(newch), |
110 | "0"(faulted), "a"(old) | ||
106 | : "memory"); | 111 | : "memory"); |
107 | sync_core(); | 112 | sync_core(); |
108 | 113 | ||
109 | if (replaced != old) | 114 | if (replaced != old && replaced != new) |
110 | faulted = 2; | 115 | faulted = 2; |
111 | 116 | ||
112 | return faulted; | 117 | return faulted; |
@@ -132,7 +137,7 @@ notrace void ftrace_code_disable(struct dyn_ftrace *rec) | |||
132 | /* move the IP back to the start of the call */ | 137 | /* move the IP back to the start of the call */ |
133 | ip -= CALL_BACK; | 138 | ip -= CALL_BACK; |
134 | 139 | ||
135 | r->failed = ftrace_modify_code(ip, save.code, (char *)&ftrace_jmp); | 140 | r->failed = ftrace_modify_code(ip, save.code, (char *)ftrace_nop); |
136 | } | 141 | } |
137 | 142 | ||
138 | static void notrace ftrace_replace_code(int saved) | 143 | static void notrace ftrace_replace_code(int saved) |
@@ -144,9 +149,9 @@ static void notrace ftrace_replace_code(int saved) | |||
144 | int i; | 149 | int i; |
145 | 150 | ||
146 | if (saved) | 151 | if (saved) |
147 | old = (char *)&ftrace_jmp; | 152 | old = (char *)ftrace_nop; |
148 | else | 153 | else |
149 | new = (char *)&ftrace_jmp; | 154 | new = (char *)ftrace_nop; |
150 | 155 | ||
151 | for (pg = ftrace_pages_start; pg; pg = pg->next) { | 156 | for (pg = ftrace_pages_start; pg; pg = pg->next) { |
152 | for (i = 0; i < pg->index; i++) { | 157 | for (i = 0; i < pg->index; i++) { |
@@ -194,12 +199,15 @@ notrace void ftrace_shutdown_replenish(void) | |||
194 | ftrace_pages->next = (void *)get_zeroed_page(GFP_KERNEL); | 199 | ftrace_pages->next = (void *)get_zeroed_page(GFP_KERNEL); |
195 | } | 200 | } |
196 | 201 | ||
197 | notrace int ftrace_shutdown_arch_init(void) | 202 | notrace int __init ftrace_shutdown_arch_init(void) |
198 | { | 203 | { |
204 | const unsigned char *const *noptable = find_nop_table(); | ||
199 | struct ftrace_page *pg; | 205 | struct ftrace_page *pg; |
200 | int cnt; | 206 | int cnt; |
201 | int i; | 207 | int i; |
202 | 208 | ||
209 | ftrace_nop = (unsigned long *)noptable[CALL_BACK]; | ||
210 | |||
203 | /* allocate a few pages */ | 211 | /* allocate a few pages */ |
204 | ftrace_pages_start = (void *)get_zeroed_page(GFP_KERNEL); | 212 | ftrace_pages_start = (void *)get_zeroed_page(GFP_KERNEL); |
205 | if (!ftrace_pages_start) | 213 | if (!ftrace_pages_start) |
diff --git a/include/asm-x86/alternative.h b/include/asm-x86/alternative.h index 1f6a9ca10126..f6aa18eadf71 100644 --- a/include/asm-x86/alternative.h +++ b/include/asm-x86/alternative.h | |||
@@ -72,6 +72,8 @@ static inline void alternatives_smp_module_del(struct module *mod) {} | |||
72 | static inline void alternatives_smp_switch(int smp) {} | 72 | static inline void alternatives_smp_switch(int smp) {} |
73 | #endif /* CONFIG_SMP */ | 73 | #endif /* CONFIG_SMP */ |
74 | 74 | ||
75 | const unsigned char *const *find_nop_table(void); | ||
76 | |||
75 | /* | 77 | /* |
76 | * Alternative instructions for different CPU types or capabilities. | 78 | * Alternative instructions for different CPU types or capabilities. |
77 | * | 79 | * |