This patch is against the x86-git which included the kgdb core. Per review request from the x86-git maintainers, the kgdb-core has had the kgdb_handle_exception() separated into individual functions. This patch does nothing other than re-organize the code. There are no functional kgdb changes.
kgdb_handle_exception now calls gdb_serial_stub() to handle the debugger communications. The gdb_serial_stub() has call out to sub handlers for all the major gdb serial packet types. Signed-off-by: Jason Wessel <[EMAIL PROTECTED]> --- kernel/kgdb.c | 1144 ++++++++++++++++++++++++++++++++-------------------------- 1 file changed, 636 insertions(+), 508 deletions(-) --- a/kernel/kgdb.c +++ b/kernel/kgdb.c @@ -147,6 +147,18 @@ struct debuggerinfo_struct { struct task_struct *task; } kgdb_info[NR_CPUS]; +struct kgdb_state { + int all_cpus_synced; + int ex_vector; + int signo; + int err_code; + int processor; + int pass_exception; + long threadid; + long kgdb_usethreadid; + struct pt_regs *linux_regs; +}; + /* to keep track of the CPU which is doing the single stepping*/ atomic_t cpu_doing_single_step = ATOMIC_INIT(-1); int kgdb_softlock_skip[NR_CPUS]; @@ -932,9 +944,588 @@ int kgdb_io_ready(int print_wait) return 1; } +/* All the functions that start with gdb_cmd are the various + * operations to implement the handlers for the gdbserial protocol + * where KGDB is communicating with an external debugger + */ +/* Handle the '?' status packets */ +static void gdb_cmd_status(struct kgdb_state *ks) +{ + /* We know that this packet is only sent + * during initial connect. So to be safe, + * we clear out our breakpoints now in case + * GDB is reconnecting. */ + remove_all_break(); + /* Also, if we haven't been able to roundup all + * CPUs, send an 'O' packet informing the user + * as much. Only need to do this once. */ + if (!ks->all_cpus_synced) + kgdb_msg_write("Not all CPUs have been " + "synced for KGDB\n", 39); + remcom_out_buffer[0] = 'S'; + remcom_out_buffer[1] = hexchars[ks->signo >> 4]; + remcom_out_buffer[2] = hexchars[ks->signo % 16]; +} + +/* Handle the 'g' get registers request */ +static void gdb_cmd_getregs(struct kgdb_state *ks) +{ + struct pt_regs *shadowregs; + struct task_struct *thread; + void *local_debuggerinfo; + int i; + + thread = kgdb_usethread; + if (!thread) { + thread = kgdb_info[ks->processor].task; + local_debuggerinfo = + kgdb_info[ks->processor].debuggerinfo; + } else { + local_debuggerinfo = NULL; + for (i = 0; i < NR_CPUS; i++) { + /* Try to find the task on some other + * or possibly this node if we do not + * find the matching task then we try + * to approximate the results. + */ + if (thread == kgdb_info[i].task) + local_debuggerinfo = + kgdb_info[i].debuggerinfo; + } + } + + /* All threads that don't have debuggerinfo should be + * in __schedule() sleeping, since all other CPUs + * are in kgdb_wait, and thus have debuggerinfo. */ + if (kgdb_ops->shadowth && + ks->kgdb_usethreadid >= pid_max + num_online_cpus()) { + shadowregs = kgdb_shadow_regs(ks->linux_regs, + ks->kgdb_usethreadid - + pid_max - + num_online_cpus()); + if (!shadowregs) { + error_packet(remcom_out_buffer, + -EINVAL); + return; + } + regs_to_gdb_regs(gdb_regs, shadowregs); + } else if (local_debuggerinfo) + regs_to_gdb_regs(gdb_regs, local_debuggerinfo); + else { + /* Pull stuff saved during + * switch_to; nothing else is + * accessible (or even particularly relevant). + * This should be enough for a stack trace. */ + sleeping_thread_to_gdb_regs(gdb_regs, thread); + } + kgdb_mem2hex((char *)gdb_regs, remcom_out_buffer, + NUMREGBYTES); +} + +/* Handle the 'G' set registers request */ +static void gdb_cmd_setregs(struct kgdb_state *ks) +{ + kgdb_hex2mem(&remcom_in_buffer[1], (char *)gdb_regs, + NUMREGBYTES); + + if (kgdb_usethread && kgdb_usethread != current) + error_packet(remcom_out_buffer, -EINVAL); + else { + gdb_regs_to_regs(gdb_regs, ks->linux_regs); + strcpy(remcom_out_buffer, "OK"); + } +} + +/* Handle the 'm' memory read bytes */ +static void gdb_cmd_memread(struct kgdb_state *ks) +{ + unsigned long addr; + unsigned long length; + char *ptr = &remcom_in_buffer[1]; + + if (kgdb_hex2long(&ptr, &addr) > 0 && *ptr++ == ',' && + kgdb_hex2long(&ptr, &length) > 0) { + ptr = kgdb_mem2hex((char *)addr, + remcom_out_buffer, + length); + if (IS_ERR(ptr)) + error_packet(remcom_out_buffer, PTR_ERR(ptr)); + } else + error_packet(remcom_out_buffer, -EINVAL); +} + +/* Handle the 'M' memory write bytes */ +static void gdb_cmd_memwrite(struct kgdb_state *ks) +{ + char *ptr = write_mem_msg(0); + + if (IS_ERR(ptr)) + error_packet(remcom_out_buffer, PTR_ERR(ptr)); + else + strcpy(remcom_out_buffer, "OK"); +} + +/* Handle the 'X' memory binary write bytes */ +static void gdb_cmd_binwrite(struct kgdb_state *ks) +{ + char *ptr = write_mem_msg(1); + + if (IS_ERR(ptr)) + error_packet(remcom_out_buffer, PTR_ERR(ptr)); + else + strcpy(remcom_out_buffer, "OK"); +} + +/* Handle the 'D' or 'k', detach or kill packets */ +static void gdb_cmd_detachkill(struct kgdb_state *ks) +{ + int error; + + /* The detach case */ + if (remcom_in_buffer[0] == 'D') { + error = remove_all_break(); + if (error < 0) + error_packet(remcom_out_buffer, error); + else { + strcpy(remcom_out_buffer, "OK"); + kgdb_connected = 0; + } + put_packet(remcom_out_buffer); + } else { + /* Assume the kill case, with no exit code checking, trying to + * force detach the debugger + */ + remove_all_break(); + kgdb_connected = 0; + } +} + + +/* Handle the 'R' reboot packets */ +static int gdb_cmd_reboot(struct kgdb_state *ks) +{ + /* For now, only honor R0 */ + if (strcmp(remcom_in_buffer, "R0") == 0) { + printk(KERN_CRIT "Executing reboot\n"); + strcpy(remcom_out_buffer, "OK"); + put_packet(remcom_out_buffer); + emergency_sync(); + /* Execution should not return from + * machine_restart() + */ + machine_restart(NULL); + kgdb_connected = 0; + return 1; + } + + return 0; +} + + +/* Handle the 'q' query packets */ +static void gdb_cmd_query(struct kgdb_state *ks) +{ + char *ptr; + int i; + struct task_struct *thread; + int numshadowth = num_online_cpus() + kgdb_ops->shadowth; + unsigned char thref[8]; + + switch (remcom_in_buffer[1]) { + case 's': + case 'f': + if (memcmp(remcom_in_buffer + 2, "ThreadInfo", 10)) { + error_packet(remcom_out_buffer, -EINVAL); + break; + } + + /* + * If we have not yet completed in + * pidhash_init() there isn't much we + * can give back. + */ + if (init_pid_ns.last_pid == 0) { + if (remcom_in_buffer[1] == 'f') + strcpy(remcom_out_buffer, + "m0000000000000001"); + break; + } + + if (remcom_in_buffer[1] == 'f') + ks->threadid = 1; + + remcom_out_buffer[0] = 'm'; + ptr = remcom_out_buffer + 1; + for (i = 0; i < 17 && ks->threadid < pid_max + numshadowth; + ks->threadid++) { + thread = getthread(ks->linux_regs, ks->threadid); + if (thread) { + int_to_threadref(thref, ks->threadid); + pack_threadid(ptr, thref); + ptr += 16; + *(ptr++) = ','; + i++; + } + } + *(--ptr) = '\0'; + break; + + case 'C': + /* Current thread id */ + strcpy(remcom_out_buffer, "QC"); + ks->threadid = shadow_pid(current->pid); + int_to_threadref(thref, ks->threadid); + pack_threadid(remcom_out_buffer + 2, thref); + break; + case 'T': + if (memcmp(remcom_in_buffer + 1, + "ThreadExtraInfo,", 16)) { + error_packet(remcom_out_buffer, + -EINVAL); + break; + } + ks->threadid = 0; + ptr = remcom_in_buffer + 17; + kgdb_hex2long(&ptr, &ks->threadid); + if (!getthread(ks->linux_regs, ks->threadid)) { + error_packet(remcom_out_buffer, + -EINVAL); + break; + } + if (ks->threadid < pid_max) + kgdb_mem2hex(getthread(ks->linux_regs, + ks->threadid)->comm, + remcom_out_buffer, 16); + else if (ks->threadid >= pid_max + num_online_cpus()) + kgdb_shadowinfo(ks->linux_regs, + remcom_out_buffer, + ks->threadid - pid_max - + num_online_cpus()); + else { + static char tmpstr[23 + BUF_THREAD_ID_SIZE]; + sprintf(tmpstr, "Shadow task %d for pid 0", + (int)(ks->threadid - pid_max)); + kgdb_mem2hex(tmpstr, remcom_out_buffer, + strlen(tmpstr)); + } + break; + } +} + +/* Handle the 'H' task query packets */ +static void gdb_cmd_task(struct kgdb_state *ks) +{ + struct task_struct *thread; + char *ptr; + + switch (remcom_in_buffer[1]) { + case 'g': + ptr = &remcom_in_buffer[2]; + kgdb_hex2long(&ptr, &ks->threadid); + thread = getthread(ks->linux_regs, ks->threadid); + if (!thread && ks->threadid > 0) { + error_packet(remcom_out_buffer, -EINVAL); + break; + } + kgdb_usethread = thread; + ks->kgdb_usethreadid = ks->threadid; + strcpy(remcom_out_buffer, "OK"); + break; + case 'c': + ptr = &remcom_in_buffer[2]; + kgdb_hex2long(&ptr, &ks->threadid); + if (!ks->threadid) + kgdb_contthread = NULL; + else { + thread = getthread(ks->linux_regs, + ks->threadid); + if (!thread && ks->threadid > 0) { + error_packet(remcom_out_buffer, -EINVAL); + break; + } + kgdb_contthread = thread; + } + strcpy(remcom_out_buffer, "OK"); + break; + } +} + +/* Handle the 'T' thread query packets */ +static void gdb_cmd_thread(struct kgdb_state *ks) +{ + char *ptr = &remcom_in_buffer[1]; + struct task_struct *thread; + + kgdb_hex2long(&ptr, &ks->threadid); + thread = getthread(ks->linux_regs, ks->threadid); + if (thread) + strcpy(remcom_out_buffer, "OK"); + else + error_packet(remcom_out_buffer, -EINVAL); +} + +/* Handle the 'z' or 'Z' breakpoint remove or set packets */ +static void gdb_cmd_break(struct kgdb_state *ks) +{ + /* Since GDB-5.3, it's been drafted that '0' is a software + * breakpoint, '1' is a hardware breakpoint, so let's do that. + */ + char *bpt_type = &remcom_in_buffer[1]; + char *ptr = &remcom_in_buffer[2]; + unsigned long addr; + unsigned long length; + int error = 0; + + if (kgdb_ops->set_hw_breakpoint && *bpt_type >= '1') { + /* Unsupported */ + if (*bpt_type > '4') + return; + } else if (*bpt_type != '0' && *bpt_type != '1') + /* Unsupported. */ + return; + /* Test if this is a hardware breakpoint, and + * if we support it. */ + if (*bpt_type == '1' && + !(kgdb_ops->flags & KGDB_HW_BREAKPOINT)) + /* Unsupported. */ + return; + + if (*(ptr++) != ',') { + error_packet(remcom_out_buffer, -EINVAL); + return; + } else if (kgdb_hex2long(&ptr, &addr)) { + if (*(ptr++) != ',' || + !kgdb_hex2long(&ptr, &length)) { + error_packet(remcom_out_buffer, -EINVAL); + return; + } + } else { + error_packet(remcom_out_buffer, -EINVAL); + return; + } + + if (remcom_in_buffer[0] == 'Z' && *bpt_type == '0') + error = kgdb_set_sw_break(addr); + else if (remcom_in_buffer[0] == 'z' && *bpt_type == '0') + error = kgdb_remove_sw_break(addr); + else if (remcom_in_buffer[0] == 'Z') + error = kgdb_ops->set_hw_breakpoint(addr, + (int)length, *bpt_type); + else if (remcom_in_buffer[0] == 'z') + error = kgdb_ops->remove_hw_breakpoint(addr, + (int) length, *bpt_type); + + if (error == 0) + strcpy(remcom_out_buffer, "OK"); + else + error_packet(remcom_out_buffer, error); +} + +/* Handle the 'C' signal / exception passing packets */ +static int gdb_cmd_exception_pass(struct kgdb_state *ks) +{ + /* C09 == pass exception + * C15 == detach kgdb, pass exception + * C30 == detach kgdb, stop attachwait, pass exception + */ + if (remcom_in_buffer[1] == '0' && + remcom_in_buffer[2] == '9') { + ks->pass_exception = 1; + remcom_in_buffer[0] = 'c'; + } else if (remcom_in_buffer[1] == '1' && + remcom_in_buffer[2] == '5') { + ks->pass_exception = 1; + remcom_in_buffer[0] = 'D'; + remove_all_break(); + kgdb_connected = 0; + return 1; + } else if (remcom_in_buffer[1] == '3' && + remcom_in_buffer[2] == '0') { + ks->pass_exception = 1; + attachwait = 0; + remcom_in_buffer[0] = 'D'; + remove_all_break(); + kgdb_connected = 0; + return 1; + } else { + error_packet(remcom_out_buffer, -EINVAL); + return 0; + } + + /* Indicate fall through */ + return -1; +} /* - * This function does all command procesing for interfacing to gdb. + * This function performs all gdbserial command procesing + */ +static int gdb_serial_stub(struct kgdb_state *ks) +{ + int error = 0; + int tmp; + + /* Clear the out buffer. */ + memset(remcom_out_buffer, 0, sizeof(remcom_out_buffer)); + + if (kgdb_connected) { + unsigned char thref[8]; + char *ptr; + /* Warn debugger if the CPUs are not synced with an 'O' + * packet */ + if (!ks->all_cpus_synced) + kgdb_msg_write("Not all CPUs have been synced for " + "KGDB\n", 39); + /* Reply to host that an exception has occurred */ + ptr = remcom_out_buffer; + *ptr++ = 'T'; + *ptr++ = hexchars[(ks->signo >> 4) % 16]; + *ptr++ = hexchars[ks->signo % 16]; + ptr += strlen(strcpy(ptr, "thread:")); + int_to_threadref(thref, shadow_pid(current->pid)); + ptr = pack_threadid(ptr, thref); + *ptr++ = ';'; + put_packet(remcom_out_buffer); + } + + kgdb_usethread = kgdb_info[ks->processor].task; + ks->kgdb_usethreadid = shadow_pid(kgdb_info[ks->processor].task->pid); + ks->pass_exception = 0; + + while (kgdb_io_ops.read_char) { + error = 0; + + /* Clear the out buffer. */ + memset(remcom_out_buffer, 0, sizeof(remcom_out_buffer)); + + get_packet(remcom_in_buffer); + + switch (remcom_in_buffer[0]) { + case '?': /* gdbserial status */ + gdb_cmd_status(ks); + break; + case 'g': /* return the value of the CPU registers */ + gdb_cmd_getregs(ks); + break; + case 'G': /* set the value of the CPU registers - return OK */ + gdb_cmd_setregs(ks); + break; + case 'm': /* mAA..AA,LLLL Read LLLL bytes at address AA..AA */ + gdb_cmd_memread(ks); + break; + case 'M': /* MAA..AA,LLLL: Write LLLL bytes at address AA..AA */ + gdb_cmd_memwrite(ks); + break; + case 'X': /* XAA..AA,LLLL: Write LLLL bytes at address AA..AA */ + gdb_cmd_binwrite(ks); + break; + /* kill or detach. KGDB should treat this like a + * continue. + */ + case 'D': /* Debugger detach */ + case 'k': /* Debugger detach via kill */ + gdb_cmd_detachkill(ks); + goto default_handle; + case 'R': /* Reboot */ + if (gdb_cmd_reboot(ks)) + goto default_handle; + break; + case 'q': /* query command */ + gdb_cmd_query(ks); + break; + case 'H': /* task related */ + gdb_cmd_task(ks); + break; + case 'T': /* Query thread status */ + gdb_cmd_thread(ks); + break; + case 'z': /* Break point remove */ + case 'Z': /* Break point set */ + gdb_cmd_break(ks); + break; + case 'C': /* Exception passing */ + tmp = gdb_cmd_exception_pass(ks); + if (tmp > 0) + goto default_handle; + if (tmp == 0) + break; + /* Fall through on tmp < 0 */ + case 'c': /* Continue packet */ + case 's': /* Single step packet */ + if (kgdb_contthread && kgdb_contthread != current) { + /* Can't switch threads in kgdb */ + error_packet(remcom_out_buffer, -EINVAL); + break; + } + kgdb_activate_sw_breakpoints(); + /* Fall through to default processing */ + default: +default_handle: + error = kgdb_arch_handle_exception(ks->ex_vector, + ks->signo, + ks->err_code, + remcom_in_buffer, + remcom_out_buffer, + ks->linux_regs); + /* Leave cmd processing on error, detach, + * kill, continue, or single step. + */ + if (error >= 0 || remcom_in_buffer[0] == 'D' || + remcom_in_buffer[0] == 'k') { + error = 0; + goto kgdb_exit; + } + + } /* switch */ + + /* reply to the request */ + put_packet(remcom_out_buffer); + } + +kgdb_exit: + if (ks->pass_exception) + error = 1; + return error; +} + +static int kgdb_reenter_check(struct kgdb_state *ks) +{ + unsigned long addr; + if (atomic_read(&debugger_active) != raw_smp_processor_id() + 1) + return 0; + + /* Panic on recursive debugger calls. */ + exception_level++; + addr = kgdb_arch_pc(ks->ex_vector, ks->linux_regs); + kgdb_deactivate_sw_breakpoints(); + /* If the break point removed ok at the place exception + * occurred, try to recover and print a warning to the end + * user because the user planted a breakpoint in a place that + * KGDB needs in order to function. + */ + if (kgdb_remove_sw_break(addr) == 0) { + exception_level = 0; + kgdb_skipexception(ks->ex_vector, ks->linux_regs); + kgdb_activate_sw_breakpoints(); + printk(KERN_CRIT + "KGDB: re-enter error: breakpoint removed\n"); + WARN_ON(1); + return 1; + } + remove_all_break(); + kgdb_skipexception(ks->ex_vector, ks->linux_regs); + if (exception_level > 1) { + dump_stack(); + panic("Recursive entry to debugger"); + } + + printk(KERN_CRIT + "KGDB: re-enter exception: ALL breakpoints killed\n"); + dump_stack(); + panic("Recursive entry to debugger"); + return 1; +} + +/* kgdb_handle_exception() - main entry point from a kernel exception * * Locking hierarchy: * interface locks, if any (begin_session) @@ -945,56 +1536,27 @@ int kgdb_io_ready(int print_wait) * of a for_each_online_cpu. * */ -int kgdb_handle_exception(int ex_vector, int signo, int err_code, - struct pt_regs *linux_regs) +int kgdb_handle_exception(int evector, int signo, int ecode, + struct pt_regs *regs) { - unsigned long length; - unsigned long addr; - char *ptr; unsigned long flags; unsigned i; - long threadid; - unsigned char thref[8]; - struct task_struct *thread = NULL; unsigned procid; - int numshadowth = num_online_cpus() + kgdb_ops->shadowth; - long kgdb_usethreadid = 0; int error = 0; - int all_cpus_synced = 0; - struct pt_regs *shadowregs; - int processor = raw_smp_processor_id(); - void *local_debuggerinfo; - int pass_exception = 0; + struct kgdb_state kgdb_var; + struct kgdb_state *ks = &kgdb_var; - /* Panic on recursive debugger calls. */ - if (atomic_read(&debugger_active) == raw_smp_processor_id() + 1) { - exception_level++; - addr = kgdb_arch_pc(ex_vector, linux_regs); - kgdb_deactivate_sw_breakpoints(); - /* If the break point removed ok at the place exception - * occurred, try to recover and print a warning to the end - * user because the user planted a breakpoint in a place that - * KGDB needs in order to function. - */ - if (kgdb_remove_sw_break(addr) == 0) { - exception_level = 0; - kgdb_skipexception(ex_vector, linux_regs); - kgdb_activate_sw_breakpoints(); - printk(KERN_CRIT - "KGDB: re-enter error: breakpoint removed\n"); - WARN_ON(1); - return 0; - } - remove_all_break(); - kgdb_skipexception(ex_vector, linux_regs); - if (exception_level > 1) - panic("Recursive entry to debugger"); + ks->processor = raw_smp_processor_id(); + ks->all_cpus_synced = 0; + ks->ex_vector = evector; + ks->signo = signo; + ks->ex_vector = evector; + ks->err_code = ecode; + ks->kgdb_usethreadid = 0; + ks->linux_regs = regs; - printk(KERN_CRIT - "KGDB: re-enter exception: ALL breakpoints killed\n"); - panic("Recursive entry to debugger"); - return 0; - } + if (kgdb_reenter_check(ks)) + return 0; /* Ouch, double exception ! */ acquirelock: @@ -1004,9 +1566,13 @@ int kgdb_handle_exception(int ex_vector, */ local_irq_save(flags); - /* Hold debugger_active */ procid = raw_smp_processor_id(); + /* Being the process of declaring a master debug processor, the + * goal is to have only one single processor set debugger_active + * to the number of the cpu + 1. The atomic variable kgdb_sync is + * used to control the selection. + */ while (1) { int i = 25; /* an arbitrary number */ if (atomic_read(&kgdb_sync) < 0 && @@ -1024,8 +1590,9 @@ int kgdb_handle_exception(int ex_vector, } /* - * Don't enter if the last instance of the exception handler wanted to - * come into the debugger again. + * Do not start the debugger connection on this CPU if the last + * instance of the exception handler wanted to come into the + * debugger on a different CPU via a single step */ if (atomic_read(&cpu_doing_single_step) != -1 && atomic_read(&cpu_doing_single_step) != procid) { @@ -1039,40 +1606,40 @@ int kgdb_handle_exception(int ex_vector, if (!kgdb_io_ready(1)) { error = 1; - goto kgdb_restore; + goto kgdb_restore; /* No I/O connection, so resume the system */ } /* - * Don't enter if we have hit a removed breakpoint. - */ - if (kgdb_skipexception(ex_vector, linux_regs)) + * Don't enter if we have hit a removed breakpoint. + */ + if (kgdb_skipexception(ks->ex_vector, ks->linux_regs)) goto kgdb_restore; - /* - * Call the I/O drivers pre_exception routine - * if the I/O driver defined one - */ + /* Call the I/O driver's pre_exception routine */ if (kgdb_io_ops.pre_exception) kgdb_io_ops.pre_exception(); - kgdb_info[processor].debuggerinfo = linux_regs; - kgdb_info[processor].task = current; + kgdb_info[ks->processor].debuggerinfo = ks->linux_regs; + kgdb_info[ks->processor].task = current; - kgdb_disable_hw_debug(linux_regs); + kgdb_disable_hw_debug(ks->linux_regs); + /* Get the slave CPU lock which will hold all the non-master + * processors in a spin state while the debugger is active + */ if (!debugger_step || !kgdb_contthread) for (i = 0; i < NR_CPUS; i++) spin_lock(&slavecpulocks[i]); #ifdef CONFIG_SMP - /* Make sure we get the other CPUs */ + /* Signal the other CPUs to enter kgdb_wait() */ if (!debugger_step || !kgdb_contthread) kgdb_roundup_cpus(flags); #endif /* spin_lock code is good enough as a barrier so we don't * need one here */ - atomic_set(&procindebug[processor], 1); + atomic_set(&procindebug[ks->processor], 1); /* Wait a reasonable time for the other CPUs to be notified and * be waiting for us. Very early on this could be imperfect @@ -1085,467 +1652,28 @@ int kgdb_handle_exception(int ex_vector, num++; } if (num >= num_online_cpus()) { - all_cpus_synced = 1; + ks->all_cpus_synced = 1; break; } } - /* Clear the out buffer. */ - memset(remcom_out_buffer, 0, sizeof(remcom_out_buffer)); - /* Master processor is completely in the debugger */ - kgdb_post_master_code(linux_regs, ex_vector, err_code); + kgdb_post_master_code(ks->linux_regs, ks->ex_vector, ks->err_code); kgdb_deactivate_sw_breakpoints(); debugger_step = 0; kgdb_contthread = NULL; exception_level = 0; - if (kgdb_connected) { - /* If we're still unable to roundup all of the CPUs, - * send an 'O' packet informing the user again. */ - if (!all_cpus_synced) - kgdb_msg_write("Not all CPUs have been synced for " - "KGDB\n", 39); - /* Reply to host that an exception has occurred */ - ptr = remcom_out_buffer; - *ptr++ = 'T'; - *ptr++ = hexchars[(signo >> 4) % 16]; - *ptr++ = hexchars[signo % 16]; - ptr += strlen(strcpy(ptr, "thread:")); - int_to_threadref(thref, shadow_pid(current->pid)); - ptr = pack_threadid(ptr, thref); - *ptr++ = ';'; - - put_packet(remcom_out_buffer); - } - - kgdb_usethread = kgdb_info[processor].task; - kgdb_usethreadid = shadow_pid(kgdb_info[processor].task->pid); - - while (kgdb_io_ops.read_char) { - char *bpt_type; - error = 0; - - /* Clear the out buffer. */ - memset(remcom_out_buffer, 0, sizeof(remcom_out_buffer)); - - get_packet(remcom_in_buffer); - - switch (remcom_in_buffer[0]) { - case '?': - /* We know that this packet is only sent - * during initial connect. So to be safe, - * we clear out our breakpoints now incase - * GDB is reconnecting. */ - remove_all_break(); - /* Also, if we haven't been able to roundup all - * CPUs, send an 'O' packet informing the user - * as much. Only need to do this once. */ - if (!all_cpus_synced) - kgdb_msg_write("Not all CPUs have been " - "synced for KGDB\n", 39); - remcom_out_buffer[0] = 'S'; - remcom_out_buffer[1] = hexchars[signo >> 4]; - remcom_out_buffer[2] = hexchars[signo % 16]; - break; - - case 'g': /* return the value of the CPU registers */ - thread = kgdb_usethread; - - if (!thread) { - thread = kgdb_info[processor].task; - local_debuggerinfo = - kgdb_info[processor].debuggerinfo; - } else { - local_debuggerinfo = NULL; - for (i = 0; i < NR_CPUS; i++) { - /* Try to find the task on some other - * or possibly this node if we do not - * find the matching task then we try - * to approximate the results. - */ - if (thread == kgdb_info[i].task) - local_debuggerinfo = - kgdb_info[i].debuggerinfo; - } - } - - /* All threads that don't have debuggerinfo should be - * in __schedule() sleeping, since all other CPUs - * are in kgdb_wait, and thus have debuggerinfo. */ - if (kgdb_ops->shadowth && - kgdb_usethreadid >= pid_max + num_online_cpus()) { - shadowregs = kgdb_shadow_regs(linux_regs, - kgdb_usethreadid - - pid_max - - num_online_cpus - ()); - if (!shadowregs) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - regs_to_gdb_regs(gdb_regs, shadowregs); - } else if (local_debuggerinfo) - regs_to_gdb_regs(gdb_regs, local_debuggerinfo); - else { - /* Pull stuff saved during - * switch_to; nothing else is - * accessible (or even particularly relevant). - * This should be enough for a stack trace. */ - sleeping_thread_to_gdb_regs(gdb_regs, thread); - } - kgdb_mem2hex((char *)gdb_regs, remcom_out_buffer, - NUMREGBYTES); - break; - - /* set the value of the CPU registers - return OK */ - case 'G': - kgdb_hex2mem(&remcom_in_buffer[1], (char *)gdb_regs, - NUMREGBYTES); - - if (kgdb_usethread && kgdb_usethread != current) - error_packet(remcom_out_buffer, -EINVAL); - else { - gdb_regs_to_regs(gdb_regs, linux_regs); - strcpy(remcom_out_buffer, "OK"); - } - break; - - /* mAA..AA,LLLL Read LLLL bytes at address AA..AA */ - case 'm': - ptr = &remcom_in_buffer[1]; - if (kgdb_hex2long(&ptr, &addr) > 0 && *ptr++ == ',' && - kgdb_hex2long(&ptr, &length) > 0) { - ptr = kgdb_mem2hex((char *)addr, - remcom_out_buffer, - length); - if (IS_ERR(ptr)) - error_packet(remcom_out_buffer, - PTR_ERR(ptr)); - } else - error_packet(remcom_out_buffer, -EINVAL); - break; - - /* MAA..AA,LLLL: Write LLLL bytes at address AA..AA */ - case 'M': - ptr = write_mem_msg(0); - if (IS_ERR(ptr)) - error_packet(remcom_out_buffer, PTR_ERR(ptr)); - else - strcpy(remcom_out_buffer, "OK"); - break; - /* XAA..AA,LLLL: Write LLLL bytes at address AA..AA */ - case 'X': - ptr = write_mem_msg(1); - if (IS_ERR(ptr)) - error_packet(remcom_out_buffer, PTR_ERR(ptr)); - else - strcpy(remcom_out_buffer, "OK"); - break; - - /* kill or detach. KGDB should treat this like a - * continue. - */ - case 'D': - error = remove_all_break(); - if (error < 0) - error_packet(remcom_out_buffer, error); - else { - strcpy(remcom_out_buffer, "OK"); - kgdb_connected = 0; - } - put_packet(remcom_out_buffer); - goto default_handle; - - case 'k': - /* Don't care about error from remove_all_break */ - remove_all_break(); - kgdb_connected = 0; - goto default_handle; - - /* Reboot */ - case 'R': - /* For now, only honor R0 */ - if (strcmp(remcom_in_buffer, "R0") == 0) { - printk(KERN_CRIT "Executing reboot\n"); - strcpy(remcom_out_buffer, "OK"); - put_packet(remcom_out_buffer); - emergency_sync(); - /* Execution should not return from - * machine_restart() - */ - machine_restart(NULL); - kgdb_connected = 0; - goto default_handle; - } - - /* query */ - case 'q': - switch (remcom_in_buffer[1]) { - case 's': - case 'f': - if (memcmp(remcom_in_buffer + 2, "ThreadInfo", - 10)) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - - /* - * If we have not yet completed in - * pidhash_init() there isn't much we - * can give back. - */ - if (init_pid_ns.last_pid == 0) { - if (remcom_in_buffer[1] == 'f') - strcpy(remcom_out_buffer, - "m0000000000000001"); - break; - } - - if (remcom_in_buffer[1] == 'f') - threadid = 1; - - remcom_out_buffer[0] = 'm'; - ptr = remcom_out_buffer + 1; - for (i = 0; i < 17 && threadid < pid_max + - numshadowth; threadid++) { - thread = getthread(linux_regs, - threadid); - if (thread) { - int_to_threadref(thref, - threadid); - pack_threadid(ptr, thref); - ptr += 16; - *(ptr++) = ','; - i++; - } - } - *(--ptr) = '\0'; - break; - - case 'C': - /* Current thread id */ - strcpy(remcom_out_buffer, "QC"); - - threadid = shadow_pid(current->pid); - - int_to_threadref(thref, threadid); - pack_threadid(remcom_out_buffer + 2, thref); - break; - case 'T': - if (memcmp(remcom_in_buffer + 1, - "ThreadExtraInfo,", 16)) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - threadid = 0; - ptr = remcom_in_buffer + 17; - kgdb_hex2long(&ptr, &threadid); - if (!getthread(linux_regs, threadid)) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - if (threadid < pid_max) - kgdb_mem2hex(getthread(linux_regs, - threadid)->comm, - remcom_out_buffer, 16); - else if (threadid >= pid_max + - num_online_cpus()) - kgdb_shadowinfo(linux_regs, - remcom_out_buffer, - threadid - pid_max - - num_online_cpus()); - else { - static char tmpstr[23 + - BUF_THREAD_ID_SIZE]; - sprintf(tmpstr, "Shadow task %d" - " for pid 0", - (int)(threadid - pid_max)); - kgdb_mem2hex(tmpstr, remcom_out_buffer, - strlen(tmpstr)); - } - break; - } - break; - - /* task related */ - case 'H': - switch (remcom_in_buffer[1]) { - case 'g': - ptr = &remcom_in_buffer[2]; - kgdb_hex2long(&ptr, &threadid); - thread = getthread(linux_regs, threadid); - if (!thread && threadid > 0) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - kgdb_usethread = thread; - kgdb_usethreadid = threadid; - strcpy(remcom_out_buffer, "OK"); - break; - - case 'c': - ptr = &remcom_in_buffer[2]; - kgdb_hex2long(&ptr, &threadid); - if (!threadid) - kgdb_contthread = NULL; - else { - thread = getthread(linux_regs, - threadid); - if (!thread && threadid > 0) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - kgdb_contthread = thread; - } - strcpy(remcom_out_buffer, "OK"); - break; - } - break; - - /* Query thread status */ - case 'T': - ptr = &remcom_in_buffer[1]; - kgdb_hex2long(&ptr, &threadid); - thread = getthread(linux_regs, threadid); - if (thread) - strcpy(remcom_out_buffer, "OK"); - else - error_packet(remcom_out_buffer, -EINVAL); - break; - /* Since GDB-5.3, it's been drafted that '0' is a software - * breakpoint, '1' is a hardware breakpoint, so let's do - * that. - */ - case 'z': - case 'Z': - bpt_type = &remcom_in_buffer[1]; - ptr = &remcom_in_buffer[2]; - - if (kgdb_ops->set_hw_breakpoint && *bpt_type >= '1') { - /* Unsupported */ - if (*bpt_type > '4') - break; - } else if (*bpt_type != '0' && *bpt_type != '1') - /* Unsupported. */ - break; - /* Test if this is a hardware breakpoint, and - * if we support it. */ - if (*bpt_type == '1' && - !(kgdb_ops->flags & KGDB_HW_BREAKPOINT)) - /* Unsupported. */ - break; - - if (*(ptr++) != ',') { - error_packet(remcom_out_buffer, -EINVAL); - break; - } else if (kgdb_hex2long(&ptr, &addr)) { - if (*(ptr++) != ',' || - !kgdb_hex2long(&ptr, &length)) { - error_packet(remcom_out_buffer, - -EINVAL); - break; - } - } else { - error_packet(remcom_out_buffer, -EINVAL); - break; - } + /* Talk to debugger with gdbserial protocol */ + error = gdb_serial_stub(ks); - if (remcom_in_buffer[0] == 'Z' && *bpt_type == '0') - error = kgdb_set_sw_break(addr); - else if (remcom_in_buffer[0] == 'z' && *bpt_type == '0') - error = kgdb_remove_sw_break(addr); - else if (remcom_in_buffer[0] == 'Z') - error = kgdb_ops->set_hw_breakpoint(addr, - (int)length, - *bpt_type); - else if (remcom_in_buffer[0] == 'z') - error = kgdb_ops->remove_hw_breakpoint(addr, - (int) - length, - *bpt_type); - - if (error == 0) - strcpy(remcom_out_buffer, "OK"); - else - error_packet(remcom_out_buffer, error); - - break; - case 'C': - /* C09 == pass exception - * C15 == detach kgdb, pass exception - * C30 == detach kgdb, stop attachwait, pass exception - */ - if (remcom_in_buffer[1] == '0' && - remcom_in_buffer[2] == '9') { - pass_exception = 1; - remcom_in_buffer[0] = 'c'; - } else if (remcom_in_buffer[1] == '1' && - remcom_in_buffer[2] == '5') { - pass_exception = 1; - remcom_in_buffer[0] = 'D'; - remove_all_break(); - kgdb_connected = 0; - goto default_handle; - } else if (remcom_in_buffer[1] == '3' && - remcom_in_buffer[2] == '0') { - pass_exception = 1; - attachwait = 0; - remcom_in_buffer[0] = 'D'; - remove_all_break(); - kgdb_connected = 0; - goto default_handle; - } else { - error_packet(remcom_out_buffer, error); - break; - } - case 'c': - case 's': - if (kgdb_contthread && kgdb_contthread != current) { - /* Can't switch threads in kgdb */ - error_packet(remcom_out_buffer, -EINVAL); - break; - } - kgdb_activate_sw_breakpoints(); - /* Followthrough to default processing */ - default: -default_handle: - error = kgdb_arch_handle_exception(ex_vector, signo, - err_code, - remcom_in_buffer, - remcom_out_buffer, - linux_regs); - if (error >= 0 || remcom_in_buffer[0] == 'D' || - remcom_in_buffer[0] == 'k') { - error = 0; - goto kgdb_exit; - } - - } /* switch */ - - /* reply to the request */ - put_packet(remcom_out_buffer); - } - - kgdb_exit: - if (pass_exception) - error = 1; - /* - * Call the I/O driver's post_exception routine - * if the I/O driver defined one. - */ + /* Call the I/O driver's post_exception routine */ if (kgdb_io_ops.post_exception) kgdb_io_ops.post_exception(); - kgdb_info[processor].debuggerinfo = NULL; - kgdb_info[processor].task = NULL; - atomic_set(&procindebug[processor], 0); + kgdb_info[ks->processor].debuggerinfo = NULL; + kgdb_info[ks->processor].task = NULL; + atomic_set(&procindebug[ks->processor], 0); if (!debugger_step || !kgdb_contthread) { for (i = 0; i < NR_CPUS; i++) @@ -1581,7 +1709,7 @@ default_handle: atomic_set(&debugger_active, 0); atomic_set(&kgdb_sync, -1); clocksource_touch_watchdog(); - kgdb_softlock_skip[processor] = 1; + kgdb_softlock_skip[ks->processor] = 1; local_irq_restore(flags); return error; -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to [EMAIL PROTECTED] More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/