diff options
Diffstat (limited to 'arch/powerpc/kernel/kgdb.c')
-rw-r--r-- | arch/powerpc/kernel/kgdb.c | 27 |
1 files changed, 24 insertions, 3 deletions
diff --git a/arch/powerpc/kernel/kgdb.c b/arch/powerpc/kernel/kgdb.c index 782bd0a3c2f0..c470a40b29f5 100644 --- a/arch/powerpc/kernel/kgdb.c +++ b/arch/powerpc/kernel/kgdb.c | |||
@@ -25,6 +25,7 @@ | |||
25 | #include <asm/processor.h> | 25 | #include <asm/processor.h> |
26 | #include <asm/machdep.h> | 26 | #include <asm/machdep.h> |
27 | #include <asm/debug.h> | 27 | #include <asm/debug.h> |
28 | #include <linux/slab.h> | ||
28 | 29 | ||
29 | /* | 30 | /* |
30 | * This table contains the mapping between PowerPC hardware trap types, and | 31 | * This table contains the mapping between PowerPC hardware trap types, and |
@@ -101,6 +102,21 @@ static int computeSignal(unsigned int tt) | |||
101 | return SIGHUP; /* default for things we don't know about */ | 102 | return SIGHUP; /* default for things we don't know about */ |
102 | } | 103 | } |
103 | 104 | ||
105 | /** | ||
106 | * | ||
107 | * kgdb_skipexception - Bail out of KGDB when we've been triggered. | ||
108 | * @exception: Exception vector number | ||
109 | * @regs: Current &struct pt_regs. | ||
110 | * | ||
111 | * On some architectures we need to skip a breakpoint exception when | ||
112 | * it occurs after a breakpoint has been removed. | ||
113 | * | ||
114 | */ | ||
115 | int kgdb_skipexception(int exception, struct pt_regs *regs) | ||
116 | { | ||
117 | return kgdb_isremovedbreak(regs->nip); | ||
118 | } | ||
119 | |||
104 | static int kgdb_call_nmi_hook(struct pt_regs *regs) | 120 | static int kgdb_call_nmi_hook(struct pt_regs *regs) |
105 | { | 121 | { |
106 | kgdb_nmicallback(raw_smp_processor_id(), regs); | 122 | kgdb_nmicallback(raw_smp_processor_id(), regs); |
@@ -138,6 +154,8 @@ static int kgdb_handle_breakpoint(struct pt_regs *regs) | |||
138 | static int kgdb_singlestep(struct pt_regs *regs) | 154 | static int kgdb_singlestep(struct pt_regs *regs) |
139 | { | 155 | { |
140 | struct thread_info *thread_info, *exception_thread_info; | 156 | struct thread_info *thread_info, *exception_thread_info; |
157 | struct thread_info *backup_current_thread_info = \ | ||
158 | (struct thread_info *)kmalloc(sizeof(struct thread_info), GFP_KERNEL); | ||
141 | 159 | ||
142 | if (user_mode(regs)) | 160 | if (user_mode(regs)) |
143 | return 0; | 161 | return 0; |
@@ -155,13 +173,17 @@ static int kgdb_singlestep(struct pt_regs *regs) | |||
155 | thread_info = (struct thread_info *)(regs->gpr[1] & ~(THREAD_SIZE-1)); | 173 | thread_info = (struct thread_info *)(regs->gpr[1] & ~(THREAD_SIZE-1)); |
156 | exception_thread_info = current_thread_info(); | 174 | exception_thread_info = current_thread_info(); |
157 | 175 | ||
158 | if (thread_info != exception_thread_info) | 176 | if (thread_info != exception_thread_info) { |
177 | /* Save the original current_thread_info. */ | ||
178 | memcpy(backup_current_thread_info, exception_thread_info, sizeof *thread_info); | ||
159 | memcpy(exception_thread_info, thread_info, sizeof *thread_info); | 179 | memcpy(exception_thread_info, thread_info, sizeof *thread_info); |
180 | } | ||
160 | 181 | ||
161 | kgdb_handle_exception(0, SIGTRAP, 0, regs); | 182 | kgdb_handle_exception(0, SIGTRAP, 0, regs); |
162 | 183 | ||
163 | if (thread_info != exception_thread_info) | 184 | if (thread_info != exception_thread_info) |
164 | memcpy(thread_info, exception_thread_info, sizeof *thread_info); | 185 | /* Restore current_thread_info lastly. */ |
186 | memcpy(exception_thread_info, backup_current_thread_info, sizeof *thread_info); | ||
165 | 187 | ||
166 | return 1; | 188 | return 1; |
167 | } | 189 | } |
@@ -410,7 +432,6 @@ int kgdb_arch_handle_exception(int vector, int signo, int err_code, | |||
410 | #else | 432 | #else |
411 | linux_regs->msr |= MSR_SE; | 433 | linux_regs->msr |= MSR_SE; |
412 | #endif | 434 | #endif |
413 | kgdb_single_step = 1; | ||
414 | atomic_set(&kgdb_cpu_doing_single_step, | 435 | atomic_set(&kgdb_cpu_doing_single_step, |
415 | raw_smp_processor_id()); | 436 | raw_smp_processor_id()); |
416 | } | 437 | } |