diff options
Diffstat (limited to 'drivers/misc/kgdbts.c')
-rw-r--r-- | drivers/misc/kgdbts.c | 50 |
1 files changed, 17 insertions, 33 deletions
diff --git a/drivers/misc/kgdbts.c b/drivers/misc/kgdbts.c index 72450237a0f4..8cebec5e85ee 100644 --- a/drivers/misc/kgdbts.c +++ b/drivers/misc/kgdbts.c | |||
@@ -285,32 +285,28 @@ static void hw_break_val_write(void) | |||
285 | static int check_and_rewind_pc(char *put_str, char *arg) | 285 | static int check_and_rewind_pc(char *put_str, char *arg) |
286 | { | 286 | { |
287 | unsigned long addr = lookup_addr(arg); | 287 | unsigned long addr = lookup_addr(arg); |
288 | unsigned long ip; | ||
288 | int offset = 0; | 289 | int offset = 0; |
289 | 290 | ||
290 | kgdb_hex2mem(&put_str[1], (char *)kgdbts_gdb_regs, | 291 | kgdb_hex2mem(&put_str[1], (char *)kgdbts_gdb_regs, |
291 | NUMREGBYTES); | 292 | NUMREGBYTES); |
292 | gdb_regs_to_pt_regs(kgdbts_gdb_regs, &kgdbts_regs); | 293 | gdb_regs_to_pt_regs(kgdbts_gdb_regs, &kgdbts_regs); |
293 | v2printk("Stopped at IP: %lx\n", instruction_pointer(&kgdbts_regs)); | 294 | ip = instruction_pointer(&kgdbts_regs); |
294 | #ifdef CONFIG_X86 | 295 | v2printk("Stopped at IP: %lx\n", ip); |
295 | /* On x86 a breakpoint stop requires it to be decremented */ | 296 | #ifdef GDB_ADJUSTS_BREAK_OFFSET |
296 | if (addr + 1 == kgdbts_regs.ip) | 297 | /* On some arches, a breakpoint stop requires it to be decremented */ |
297 | offset = -1; | 298 | if (addr + BREAK_INSTR_SIZE == ip) |
298 | #elif defined(CONFIG_SUPERH) | 299 | offset = -BREAK_INSTR_SIZE; |
299 | /* On SUPERH a breakpoint stop requires it to be decremented */ | ||
300 | if (addr + 2 == kgdbts_regs.pc) | ||
301 | offset = -2; | ||
302 | #endif | 300 | #endif |
303 | if (strcmp(arg, "silent") && | 301 | if (strcmp(arg, "silent") && ip + offset != addr) { |
304 | instruction_pointer(&kgdbts_regs) + offset != addr) { | ||
305 | eprintk("kgdbts: BP mismatch %lx expected %lx\n", | 302 | eprintk("kgdbts: BP mismatch %lx expected %lx\n", |
306 | instruction_pointer(&kgdbts_regs) + offset, addr); | 303 | ip + offset, addr); |
307 | return 1; | 304 | return 1; |
308 | } | 305 | } |
309 | #ifdef CONFIG_X86 | 306 | /* Readjust the instruction pointer if needed */ |
310 | /* On x86 adjust the instruction pointer if needed */ | 307 | ip += offset; |
311 | kgdbts_regs.ip += offset; | 308 | #ifdef GDB_ADJUSTS_BREAK_OFFSET |
312 | #elif defined(CONFIG_SUPERH) | 309 | instruction_pointer_set(&kgdbts_regs, ip); |
313 | kgdbts_regs.pc += offset; | ||
314 | #endif | 310 | #endif |
315 | return 0; | 311 | return 0; |
316 | } | 312 | } |
@@ -645,7 +641,7 @@ static int validate_simple_test(char *put_str) | |||
645 | 641 | ||
646 | while (*chk_str != '\0' && *put_str != '\0') { | 642 | while (*chk_str != '\0' && *put_str != '\0') { |
647 | /* If someone does a * to match the rest of the string, allow | 643 | /* If someone does a * to match the rest of the string, allow |
648 | * it, or stop if the recieved string is complete. | 644 | * it, or stop if the received string is complete. |
649 | */ | 645 | */ |
650 | if (*put_str == '#' || *chk_str == '*') | 646 | if (*put_str == '#' || *chk_str == '*') |
651 | return 0; | 647 | return 0; |
@@ -988,7 +984,7 @@ static void kgdbts_run_tests(void) | |||
988 | 984 | ||
989 | static int kgdbts_option_setup(char *opt) | 985 | static int kgdbts_option_setup(char *opt) |
990 | { | 986 | { |
991 | if (strlen(opt) > MAX_CONFIG_LEN) { | 987 | if (strlen(opt) >= MAX_CONFIG_LEN) { |
992 | printk(KERN_ERR "kgdbts: config string too long\n"); | 988 | printk(KERN_ERR "kgdbts: config string too long\n"); |
993 | return -ENOSPC; | 989 | return -ENOSPC; |
994 | } | 990 | } |
@@ -1044,12 +1040,6 @@ static int __init init_kgdbts(void) | |||
1044 | return configure_kgdbts(); | 1040 | return configure_kgdbts(); |
1045 | } | 1041 | } |
1046 | 1042 | ||
1047 | static void cleanup_kgdbts(void) | ||
1048 | { | ||
1049 | if (configured == 1) | ||
1050 | kgdb_unregister_io_module(&kgdbts_io_ops); | ||
1051 | } | ||
1052 | |||
1053 | static int kgdbts_get_char(void) | 1043 | static int kgdbts_get_char(void) |
1054 | { | 1044 | { |
1055 | int val = 0; | 1045 | int val = 0; |
@@ -1081,10 +1071,8 @@ static int param_set_kgdbts_var(const char *kmessage, struct kernel_param *kp) | |||
1081 | return 0; | 1071 | return 0; |
1082 | } | 1072 | } |
1083 | 1073 | ||
1084 | if (kgdb_connected) { | 1074 | if (configured == 1) { |
1085 | printk(KERN_ERR | 1075 | printk(KERN_ERR "kgdbts: ERROR: Already configured and running.\n"); |
1086 | "kgdbts: Cannot reconfigure while KGDB is connected.\n"); | ||
1087 | |||
1088 | return -EBUSY; | 1076 | return -EBUSY; |
1089 | } | 1077 | } |
1090 | 1078 | ||
@@ -1093,9 +1081,6 @@ static int param_set_kgdbts_var(const char *kmessage, struct kernel_param *kp) | |||
1093 | if (config[len - 1] == '\n') | 1081 | if (config[len - 1] == '\n') |
1094 | config[len - 1] = '\0'; | 1082 | config[len - 1] = '\0'; |
1095 | 1083 | ||
1096 | if (configured == 1) | ||
1097 | cleanup_kgdbts(); | ||
1098 | |||
1099 | /* Go and configure with the new params. */ | 1084 | /* Go and configure with the new params. */ |
1100 | return configure_kgdbts(); | 1085 | return configure_kgdbts(); |
1101 | } | 1086 | } |
@@ -1123,7 +1108,6 @@ static struct kgdb_io kgdbts_io_ops = { | |||
1123 | }; | 1108 | }; |
1124 | 1109 | ||
1125 | module_init(init_kgdbts); | 1110 | module_init(init_kgdbts); |
1126 | module_exit(cleanup_kgdbts); | ||
1127 | module_param_call(kgdbts, param_set_kgdbts_var, param_get_string, &kps, 0644); | 1111 | module_param_call(kgdbts, param_set_kgdbts_var, param_get_string, &kps, 0644); |
1128 | MODULE_PARM_DESC(kgdbts, "<A|V1|V2>[F#|S#][N#]"); | 1112 | MODULE_PARM_DESC(kgdbts, "<A|V1|V2>[F#|S#][N#]"); |
1129 | MODULE_DESCRIPTION("KGDB Test Suite"); | 1113 | MODULE_DESCRIPTION("KGDB Test Suite"); |