diff options
Diffstat (limited to 'arch/blackfin/kernel')
-rw-r--r-- | arch/blackfin/kernel/Makefile | 1 | ||||
-rw-r--r-- | arch/blackfin/kernel/pseudodbg.c | 73 | ||||
-rw-r--r-- | arch/blackfin/kernel/traps.c | 15 |
3 files changed, 89 insertions, 0 deletions
diff --git a/arch/blackfin/kernel/Makefile b/arch/blackfin/kernel/Makefile index 2fc7f32ae32f..30d0d1f01dc7 100644 --- a/arch/blackfin/kernel/Makefile +++ b/arch/blackfin/kernel/Makefile | |||
@@ -31,6 +31,7 @@ obj-$(CONFIG_EARLY_PRINTK) += early_printk.o | |||
31 | obj-$(CONFIG_EARLY_PRINTK) += shadow_console.o | 31 | obj-$(CONFIG_EARLY_PRINTK) += shadow_console.o |
32 | obj-$(CONFIG_STACKTRACE) += stacktrace.o | 32 | obj-$(CONFIG_STACKTRACE) += stacktrace.o |
33 | obj-$(CONFIG_DEBUG_VERBOSE) += trace.o | 33 | obj-$(CONFIG_DEBUG_VERBOSE) += trace.o |
34 | obj-$(CONFIG_BFIN_PSEUDODBG_INSNS) += pseudodbg.o | ||
34 | 35 | ||
35 | # the kgdb test puts code into L2 and without linker | 36 | # the kgdb test puts code into L2 and without linker |
36 | # relaxation, we need to force long calls to/from it | 37 | # relaxation, we need to force long calls to/from it |
diff --git a/arch/blackfin/kernel/pseudodbg.c b/arch/blackfin/kernel/pseudodbg.c new file mode 100644 index 000000000000..4474b8db3501 --- /dev/null +++ b/arch/blackfin/kernel/pseudodbg.c | |||
@@ -0,0 +1,73 @@ | |||
1 | /* The fake debug assert instructions | ||
2 | * | ||
3 | * Copyright 2010 Analog Devices Inc. | ||
4 | * | ||
5 | * Licensed under the GPL-2 or later | ||
6 | */ | ||
7 | |||
8 | #include <linux/types.h> | ||
9 | #include <linux/kernel.h> | ||
10 | #include <linux/ptrace.h> | ||
11 | |||
12 | #define PseudoDbg_Assert_opcode 0xf0000000 | ||
13 | #define PseudoDbg_Assert_expected_bits 0 | ||
14 | #define PseudoDbg_Assert_expected_mask 0xffff | ||
15 | #define PseudoDbg_Assert_regtest_bits 16 | ||
16 | #define PseudoDbg_Assert_regtest_mask 0x7 | ||
17 | #define PseudoDbg_Assert_grp_bits 19 | ||
18 | #define PseudoDbg_Assert_grp_mask 0x7 | ||
19 | #define PseudoDbg_Assert_dbgop_bits 22 | ||
20 | #define PseudoDbg_Assert_dbgop_mask 0x3 | ||
21 | #define PseudoDbg_Assert_dontcare_bits 24 | ||
22 | #define PseudoDbg_Assert_dontcare_mask 0x7 | ||
23 | #define PseudoDbg_Assert_code_bits 27 | ||
24 | #define PseudoDbg_Assert_code_mask 0x1f | ||
25 | |||
26 | bool execute_pseudodbg_assert(struct pt_regs *fp, unsigned int opcode) | ||
27 | { | ||
28 | int expected = ((opcode >> PseudoDbg_Assert_expected_bits) & PseudoDbg_Assert_expected_mask); | ||
29 | int dbgop = ((opcode >> (PseudoDbg_Assert_dbgop_bits)) & PseudoDbg_Assert_dbgop_mask); | ||
30 | int grp = ((opcode >> (PseudoDbg_Assert_grp_bits)) & PseudoDbg_Assert_grp_mask); | ||
31 | int regtest = ((opcode >> (PseudoDbg_Assert_regtest_bits)) & PseudoDbg_Assert_regtest_mask); | ||
32 | long *value = &fp->r0; | ||
33 | |||
34 | if ((opcode & 0xFF000000) != PseudoDbg_Assert_opcode) | ||
35 | return false; | ||
36 | |||
37 | /* Only do Dregs and Pregs for now */ | ||
38 | if (grp > 1) | ||
39 | return false; | ||
40 | |||
41 | /* | ||
42 | * Unfortunately, the pt_regs structure is not laid out the same way as the | ||
43 | * hardware register file, so we need to do some fix ups. | ||
44 | */ | ||
45 | if (grp == 0 || (grp == 1 && regtest < 6)) | ||
46 | value -= (regtest + 8 * grp); | ||
47 | else if (grp == 1 && regtest == 6) | ||
48 | value = &fp->usp; | ||
49 | else if (grp == 1 && regtest == 7) | ||
50 | value = &fp->fp; | ||
51 | |||
52 | if (dbgop == 0 || dbgop == 2) { | ||
53 | /* DBGA ( regs_lo , uimm16 ) */ | ||
54 | /* DBGAL ( regs , uimm16 ) */ | ||
55 | if (expected != (*value & 0xFFFF)) { | ||
56 | pr_notice("DBGA (%s%i.L,0x%x) failure, got 0x%x\n", grp ? "P" : "R", | ||
57 | regtest, expected, (unsigned int)(*value & 0xFFFF)); | ||
58 | return false; | ||
59 | } | ||
60 | |||
61 | } else if (dbgop == 1 || dbgop == 3) { | ||
62 | /* DBGA ( regs_hi , uimm16 ) */ | ||
63 | /* DBGAH ( regs , uimm16 ) */ | ||
64 | if (expected != ((*value >> 16) & 0xFFFF)) { | ||
65 | pr_notice("DBGA (%s%i.H,0x%x) failure, got 0x%x\n", grp ? "P" : "R", | ||
66 | regtest, expected, (unsigned int)((*value >> 16) & 0xFFFF)); | ||
67 | return false; | ||
68 | } | ||
69 | } | ||
70 | |||
71 | fp->pc += 4; | ||
72 | return true; | ||
73 | } | ||
diff --git a/arch/blackfin/kernel/traps.c b/arch/blackfin/kernel/traps.c index fffcf8a516bf..9369836365bb 100644 --- a/arch/blackfin/kernel/traps.c +++ b/arch/blackfin/kernel/traps.c | |||
@@ -16,6 +16,7 @@ | |||
16 | #include <linux/irq.h> | 16 | #include <linux/irq.h> |
17 | #include <asm/trace.h> | 17 | #include <asm/trace.h> |
18 | #include <asm/fixed_code.h> | 18 | #include <asm/fixed_code.h> |
19 | #include <asm/pseudo_instructions.h> | ||
19 | 20 | ||
20 | #ifdef CONFIG_KGDB | 21 | #ifdef CONFIG_KGDB |
21 | # include <linux/kgdb.h> | 22 | # include <linux/kgdb.h> |
@@ -68,6 +69,9 @@ asmlinkage notrace void trap_c(struct pt_regs *fp) | |||
68 | #ifdef CONFIG_DEBUG_BFIN_HWTRACE_ON | 69 | #ifdef CONFIG_DEBUG_BFIN_HWTRACE_ON |
69 | int j; | 70 | int j; |
70 | #endif | 71 | #endif |
72 | #ifdef CONFIG_BFIN_PSEUDODBG_INSNS | ||
73 | int opcode; | ||
74 | #endif | ||
71 | unsigned int cpu = raw_smp_processor_id(); | 75 | unsigned int cpu = raw_smp_processor_id(); |
72 | const char *strerror = NULL; | 76 | const char *strerror = NULL; |
73 | int sig = 0; | 77 | int sig = 0; |
@@ -200,6 +204,17 @@ asmlinkage notrace void trap_c(struct pt_regs *fp) | |||
200 | } | 204 | } |
201 | } | 205 | } |
202 | #endif | 206 | #endif |
207 | #ifdef CONFIG_BFIN_PSEUDODBG_INSNS | ||
208 | /* | ||
209 | * Support for the fake instructions, if the instruction fails, | ||
210 | * then just execute a illegal opcode failure (like normal). | ||
211 | * Don't support these instructions inside the kernel | ||
212 | */ | ||
213 | if (!kernel_mode_regs(fp) && get_instruction(&opcode, (unsigned short *)fp->pc)) { | ||
214 | if (execute_pseudodbg_assert(fp, opcode)) | ||
215 | goto traps_done; | ||
216 | } | ||
217 | #endif | ||
203 | info.si_code = ILL_ILLOPC; | 218 | info.si_code = ILL_ILLOPC; |
204 | sig = SIGILL; | 219 | sig = SIGILL; |
205 | strerror = KERN_NOTICE EXC_0x21(KERN_NOTICE); | 220 | strerror = KERN_NOTICE EXC_0x21(KERN_NOTICE); |