diff options
| -rw-r--r-- | arch/sh/kernel/process.c | 10 | ||||
| -rw-r--r-- | arch/sh/kernel/traps.c | 35 | ||||
| -rw-r--r-- | include/asm-sh/bug.h | 53 |
3 files changed, 89 insertions, 9 deletions
diff --git a/arch/sh/kernel/process.c b/arch/sh/kernel/process.c index f3e2631be144..7347f6afa030 100644 --- a/arch/sh/kernel/process.c +++ b/arch/sh/kernel/process.c | |||
| @@ -498,6 +498,16 @@ asmlinkage void break_point_trap_software(unsigned long r4, unsigned long r5, | |||
| 498 | { | 498 | { |
| 499 | struct pt_regs *regs = RELOC_HIDE(&__regs, 0); | 499 | struct pt_regs *regs = RELOC_HIDE(&__regs, 0); |
| 500 | 500 | ||
| 501 | /* Rewind */ | ||
| 501 | regs->pc -= 2; | 502 | regs->pc -= 2; |
| 503 | |||
| 504 | #ifdef CONFIG_BUG | ||
| 505 | if (__kernel_text_address(instruction_pointer(regs))) { | ||
| 506 | u16 insn = *(u16 *)instruction_pointer(regs); | ||
| 507 | if (insn == TRAPA_BUG_OPCODE) | ||
| 508 | handle_BUG(regs); | ||
| 509 | } | ||
| 510 | #endif | ||
| 511 | |||
| 502 | force_sig(SIGTRAP, current); | 512 | force_sig(SIGTRAP, current); |
| 503 | } | 513 | } |
diff --git a/arch/sh/kernel/traps.c b/arch/sh/kernel/traps.c index 3762d9dc2046..ec110157992d 100644 --- a/arch/sh/kernel/traps.c +++ b/arch/sh/kernel/traps.c | |||
| @@ -19,6 +19,7 @@ | |||
| 19 | #include <linux/kallsyms.h> | 19 | #include <linux/kallsyms.h> |
| 20 | #include <linux/io.h> | 20 | #include <linux/io.h> |
| 21 | #include <linux/debug_locks.h> | 21 | #include <linux/debug_locks.h> |
| 22 | #include <linux/limits.h> | ||
| 22 | #include <asm/system.h> | 23 | #include <asm/system.h> |
| 23 | #include <asm/uaccess.h> | 24 | #include <asm/uaccess.h> |
| 24 | 25 | ||
| @@ -129,6 +130,40 @@ static int die_if_no_fixup(const char * str, struct pt_regs * regs, long err) | |||
| 129 | return -EFAULT; | 130 | return -EFAULT; |
| 130 | } | 131 | } |
| 131 | 132 | ||
| 133 | #ifdef CONFIG_BUG | ||
| 134 | #ifdef CONFIG_DEBUG_BUGVERBOSE | ||
| 135 | static inline void do_bug_verbose(struct pt_regs *regs) | ||
| 136 | { | ||
| 137 | struct bug_frame f; | ||
| 138 | long len; | ||
| 139 | |||
| 140 | if (__copy_from_user(&f, (const void __user *)regs->pc, | ||
| 141 | sizeof(struct bug_frame))) | ||
| 142 | return; | ||
| 143 | |||
| 144 | len = __strnlen_user(f.file, PATH_MAX) - 1; | ||
| 145 | if (unlikely(len < 0 || len >= PATH_MAX)) | ||
| 146 | f.file = "<bad filename>"; | ||
| 147 | len = __strnlen_user(f.func, PATH_MAX) - 1; | ||
| 148 | if (unlikely(len < 0 || len >= PATH_MAX)) | ||
| 149 | f.func = "<bad function>"; | ||
| 150 | |||
| 151 | printk(KERN_ALERT "kernel BUG in %s() at %s:%d!\n", | ||
| 152 | f.func, f.file, f.line); | ||
| 153 | } | ||
| 154 | #else | ||
| 155 | static inline void do_bug_verbose(struct pt_regs *regs) | ||
| 156 | { | ||
| 157 | } | ||
| 158 | #endif /* CONFIG_DEBUG_BUGVERBOSE */ | ||
| 159 | #endif /* CONFIG_BUG */ | ||
| 160 | |||
| 161 | void handle_BUG(struct pt_regs *regs) | ||
| 162 | { | ||
| 163 | do_bug_verbose(regs); | ||
| 164 | die("Kernel BUG", regs, TRAPA_BUG_OPCODE & 0xff); | ||
| 165 | } | ||
| 166 | |||
| 132 | /* | 167 | /* |
| 133 | * handle an instruction that does an unaligned memory access by emulating the | 168 | * handle an instruction that does an unaligned memory access by emulating the |
| 134 | * desired behaviour | 169 | * desired behaviour |
diff --git a/include/asm-sh/bug.h b/include/asm-sh/bug.h index 1b4fc52a59e8..2f89dd06d0cd 100644 --- a/include/asm-sh/bug.h +++ b/include/asm-sh/bug.h | |||
| @@ -1,19 +1,54 @@ | |||
| 1 | #ifndef __ASM_SH_BUG_H | 1 | #ifndef __ASM_SH_BUG_H |
| 2 | #define __ASM_SH_BUG_H | 2 | #define __ASM_SH_BUG_H |
| 3 | 3 | ||
| 4 | |||
| 5 | #ifdef CONFIG_BUG | 4 | #ifdef CONFIG_BUG |
| 6 | /* | 5 | |
| 7 | * Tell the user there is some problem. | 6 | struct bug_frame { |
| 8 | */ | 7 | unsigned short opcode; |
| 9 | #define BUG() do { \ | 8 | unsigned short line; |
| 10 | printk("kernel BUG at %s:%d!\n", __FILE__, __LINE__); \ | 9 | const char *file; |
| 11 | *(volatile int *)0 = 0; \ | 10 | const char *func; |
| 11 | }; | ||
| 12 | |||
| 13 | struct pt_regs; | ||
| 14 | |||
| 15 | extern void handle_BUG(struct pt_regs *); | ||
| 16 | |||
| 17 | #define TRAPA_BUG_OPCODE 0xc33e /* trapa #0x3e */ | ||
| 18 | |||
| 19 | #ifdef CONFIG_DEBUG_BUGVERBOSE | ||
| 20 | |||
| 21 | #define BUG() \ | ||
| 22 | do { \ | ||
| 23 | __asm__ __volatile__ ( \ | ||
| 24 | ".align 2\n\t" \ | ||
| 25 | ".short %O0\n\t" \ | ||
| 26 | ".short %O1\n\t" \ | ||
| 27 | ".long %O2\n\t" \ | ||
| 28 | ".long %O3\n\t" \ | ||
| 29 | : \ | ||
| 30 | : "n" (TRAPA_BUG_OPCODE), \ | ||
| 31 | "i" (__LINE__), "X" (__FILE__), \ | ||
| 32 | "X" (__FUNCTION__)); \ | ||
| 33 | } while (0) | ||
| 34 | |||
| 35 | #else | ||
| 36 | |||
| 37 | #define BUG() \ | ||
| 38 | do { \ | ||
| 39 | __asm__ __volatile__ ( \ | ||
| 40 | ".align 2\n\t" \ | ||
| 41 | ".short %O0\n\t" \ | ||
| 42 | : \ | ||
| 43 | : "n" (TRAPA_BUG_OPCODE)); \ | ||
| 12 | } while (0) | 44 | } while (0) |
| 13 | 45 | ||
| 46 | #endif /* CONFIG_DEBUG_BUGVERBOSE */ | ||
| 47 | |||
| 14 | #define HAVE_ARCH_BUG | 48 | #define HAVE_ARCH_BUG |
| 15 | #endif | 49 | |
| 50 | #endif /* CONFIG_BUG */ | ||
| 16 | 51 | ||
| 17 | #include <asm-generic/bug.h> | 52 | #include <asm-generic/bug.h> |
| 18 | 53 | ||
| 19 | #endif | 54 | #endif /* __ASM_SH_BUG_H */ |
