diff options
author | Arjan van de Ven <arjan@linux.intel.com> | 2008-08-15 18:29:38 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-09-10 03:08:53 -0400 |
commit | ab7476cf76e560f0efda2a631a70aabe93009025 (patch) | |
tree | d4e843760774d7099692950a6ac49ccdbc8b0d55 | |
parent | fb822db465bd9fd4208eef1af4490539b236c54e (diff) |
debug: add notifier chain debugging, v2
- unbreak ia64 (and powerpc) where function pointers dont
point at code but at data (reported by Tony Luck)
[ mingo@elte.hu: various cleanups ]
Signed-off-by: Arjan van de Ven <arjan@linux.intel.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
-rw-r--r-- | include/linux/kernel.h | 3 | ||||
-rw-r--r-- | kernel/extable.c | 16 | ||||
-rw-r--r-- | kernel/notifier.c | 10 | ||||
-rw-r--r-- | lib/vsprintf.c | 2 |
4 files changed, 21 insertions, 10 deletions
diff --git a/include/linux/kernel.h b/include/linux/kernel.h index 2651f805ba6d..4e1366b552ae 100644 --- a/include/linux/kernel.h +++ b/include/linux/kernel.h | |||
@@ -187,6 +187,9 @@ extern unsigned long long memparse(char *ptr, char **retptr); | |||
187 | extern int core_kernel_text(unsigned long addr); | 187 | extern int core_kernel_text(unsigned long addr); |
188 | extern int __kernel_text_address(unsigned long addr); | 188 | extern int __kernel_text_address(unsigned long addr); |
189 | extern int kernel_text_address(unsigned long addr); | 189 | extern int kernel_text_address(unsigned long addr); |
190 | extern int func_ptr_is_kernel_text(void *ptr); | ||
191 | extern void *dereference_function_descriptor(void *ptr); | ||
192 | |||
190 | struct pid; | 193 | struct pid; |
191 | extern struct pid *session_of_pgrp(struct pid *pgrp); | 194 | extern struct pid *session_of_pgrp(struct pid *pgrp); |
192 | 195 | ||
diff --git a/kernel/extable.c b/kernel/extable.c index a26cb2e17023..adf0cc9c02d6 100644 --- a/kernel/extable.c +++ b/kernel/extable.c | |||
@@ -66,3 +66,19 @@ int kernel_text_address(unsigned long addr) | |||
66 | return 1; | 66 | return 1; |
67 | return module_text_address(addr) != NULL; | 67 | return module_text_address(addr) != NULL; |
68 | } | 68 | } |
69 | |||
70 | /* | ||
71 | * On some architectures (PPC64, IA64) function pointers | ||
72 | * are actually only tokens to some data that then holds the | ||
73 | * real function address. As a result, to find if a function | ||
74 | * pointer is part of the kernel text, we need to do some | ||
75 | * special dereferencing first. | ||
76 | */ | ||
77 | int func_ptr_is_kernel_text(void *ptr) | ||
78 | { | ||
79 | unsigned long addr; | ||
80 | addr = (unsigned long) dereference_function_descriptor(ptr); | ||
81 | if (core_kernel_text(addr)) | ||
82 | return 1; | ||
83 | return module_text_address(addr) != NULL; | ||
84 | } | ||
diff --git a/kernel/notifier.c b/kernel/notifier.c index 143fdd77dbf7..0f39e398ef60 100644 --- a/kernel/notifier.c +++ b/kernel/notifier.c | |||
@@ -21,10 +21,6 @@ BLOCKING_NOTIFIER_HEAD(reboot_notifier_list); | |||
21 | static int notifier_chain_register(struct notifier_block **nl, | 21 | static int notifier_chain_register(struct notifier_block **nl, |
22 | struct notifier_block *n) | 22 | struct notifier_block *n) |
23 | { | 23 | { |
24 | if (!kernel_text_address((unsigned long)n->notifier_call)) { | ||
25 | WARN(1, "Invalid notifier registered!"); | ||
26 | return 0; | ||
27 | } | ||
28 | while ((*nl) != NULL) { | 24 | while ((*nl) != NULL) { |
29 | if (n->priority > (*nl)->priority) | 25 | if (n->priority > (*nl)->priority) |
30 | break; | 26 | break; |
@@ -38,10 +34,6 @@ static int notifier_chain_register(struct notifier_block **nl, | |||
38 | static int notifier_chain_cond_register(struct notifier_block **nl, | 34 | static int notifier_chain_cond_register(struct notifier_block **nl, |
39 | struct notifier_block *n) | 35 | struct notifier_block *n) |
40 | { | 36 | { |
41 | if (!kernel_text_address((unsigned long)n->notifier_call)) { | ||
42 | WARN(1, "Invalid notifier registered!"); | ||
43 | return 0; | ||
44 | } | ||
45 | while ((*nl) != NULL) { | 37 | while ((*nl) != NULL) { |
46 | if ((*nl) == n) | 38 | if ((*nl) == n) |
47 | return 0; | 39 | return 0; |
@@ -92,7 +84,7 @@ static int __kprobes notifier_call_chain(struct notifier_block **nl, | |||
92 | next_nb = rcu_dereference(nb->next); | 84 | next_nb = rcu_dereference(nb->next); |
93 | 85 | ||
94 | #ifdef CONFIG_DEBUG_NOTIFIERS | 86 | #ifdef CONFIG_DEBUG_NOTIFIERS |
95 | if (!kernel_text_address((unsigned long)nb->notifier_call)) { | 87 | if (unlikely(!func_ptr_is_kernel_text(nb->notifier_call))) { |
96 | WARN(1, "Invalid notifier called!"); | 88 | WARN(1, "Invalid notifier called!"); |
97 | nb = next_nb; | 89 | nb = next_nb; |
98 | continue; | 90 | continue; |
diff --git a/lib/vsprintf.c b/lib/vsprintf.c index d8d1d1142248..f5e5ffb9942f 100644 --- a/lib/vsprintf.c +++ b/lib/vsprintf.c | |||
@@ -513,7 +513,7 @@ static char *string(char *buf, char *end, char *s, int field_width, int precisio | |||
513 | return buf; | 513 | return buf; |
514 | } | 514 | } |
515 | 515 | ||
516 | static inline void *dereference_function_descriptor(void *ptr) | 516 | void *dereference_function_descriptor(void *ptr) |
517 | { | 517 | { |
518 | #if defined(CONFIG_IA64) || defined(CONFIG_PPC64) | 518 | #if defined(CONFIG_IA64) || defined(CONFIG_PPC64) |
519 | void *p; | 519 | void *p; |