|
@@ -161,37 +161,39 @@ early_param("nokgdbroundup", opt_nokgdbroundup);
|
|
* Weak aliases for breakpoint management,
|
|
* Weak aliases for breakpoint management,
|
|
* can be overriden by architectures when needed:
|
|
* can be overriden by architectures when needed:
|
|
*/
|
|
*/
|
|
-int __weak kgdb_arch_set_breakpoint(unsigned long addr, char *saved_instr)
|
|
|
|
|
|
+int __weak kgdb_arch_set_breakpoint(struct kgdb_bkpt *bpt)
|
|
{
|
|
{
|
|
int err;
|
|
int err;
|
|
|
|
|
|
- err = probe_kernel_read(saved_instr, (char *)addr, BREAK_INSTR_SIZE);
|
|
|
|
|
|
+ err = probe_kernel_read(bpt->saved_instr, (char *)bpt->bpt_addr,
|
|
|
|
+ BREAK_INSTR_SIZE);
|
|
if (err)
|
|
if (err)
|
|
return err;
|
|
return err;
|
|
-
|
|
|
|
- return probe_kernel_write((char *)addr, arch_kgdb_ops.gdb_bpt_instr,
|
|
|
|
- BREAK_INSTR_SIZE);
|
|
|
|
|
|
+ err = probe_kernel_write((char *)bpt->bpt_addr,
|
|
|
|
+ arch_kgdb_ops.gdb_bpt_instr, BREAK_INSTR_SIZE);
|
|
|
|
+ return err;
|
|
}
|
|
}
|
|
|
|
|
|
-int __weak kgdb_arch_remove_breakpoint(unsigned long addr, char *bundle)
|
|
|
|
|
|
+int __weak kgdb_arch_remove_breakpoint(struct kgdb_bkpt *bpt)
|
|
{
|
|
{
|
|
- return probe_kernel_write((char *)addr,
|
|
|
|
- (char *)bundle, BREAK_INSTR_SIZE);
|
|
|
|
|
|
+ return probe_kernel_write((char *)bpt->bpt_addr,
|
|
|
|
+ (char *)bpt->saved_instr, BREAK_INSTR_SIZE);
|
|
}
|
|
}
|
|
|
|
|
|
int __weak kgdb_validate_break_address(unsigned long addr)
|
|
int __weak kgdb_validate_break_address(unsigned long addr)
|
|
{
|
|
{
|
|
- char tmp_variable[BREAK_INSTR_SIZE];
|
|
|
|
|
|
+ struct kgdb_bkpt tmp;
|
|
int err;
|
|
int err;
|
|
- /* Validate setting the breakpoint and then removing it. In the
|
|
|
|
|
|
+ /* Validate setting the breakpoint and then removing it. If the
|
|
* remove fails, the kernel needs to emit a bad message because we
|
|
* remove fails, the kernel needs to emit a bad message because we
|
|
* are deep trouble not being able to put things back the way we
|
|
* are deep trouble not being able to put things back the way we
|
|
* found them.
|
|
* found them.
|
|
*/
|
|
*/
|
|
- err = kgdb_arch_set_breakpoint(addr, tmp_variable);
|
|
|
|
|
|
+ tmp.bpt_addr = addr;
|
|
|
|
+ err = kgdb_arch_set_breakpoint(&tmp);
|
|
if (err)
|
|
if (err)
|
|
return err;
|
|
return err;
|
|
- err = kgdb_arch_remove_breakpoint(addr, tmp_variable);
|
|
|
|
|
|
+ err = kgdb_arch_remove_breakpoint(&tmp);
|
|
if (err)
|
|
if (err)
|
|
printk(KERN_ERR "KGDB: Critical breakpoint error, kernel "
|
|
printk(KERN_ERR "KGDB: Critical breakpoint error, kernel "
|
|
"memory destroyed at: %lx", addr);
|
|
"memory destroyed at: %lx", addr);
|
|
@@ -235,7 +237,6 @@ static void kgdb_flush_swbreak_addr(unsigned long addr)
|
|
*/
|
|
*/
|
|
int dbg_activate_sw_breakpoints(void)
|
|
int dbg_activate_sw_breakpoints(void)
|
|
{
|
|
{
|
|
- unsigned long addr;
|
|
|
|
int error;
|
|
int error;
|
|
int ret = 0;
|
|
int ret = 0;
|
|
int i;
|
|
int i;
|
|
@@ -244,16 +245,15 @@ int dbg_activate_sw_breakpoints(void)
|
|
if (kgdb_break[i].state != BP_SET)
|
|
if (kgdb_break[i].state != BP_SET)
|
|
continue;
|
|
continue;
|
|
|
|
|
|
- addr = kgdb_break[i].bpt_addr;
|
|
|
|
- error = kgdb_arch_set_breakpoint(addr,
|
|
|
|
- kgdb_break[i].saved_instr);
|
|
|
|
|
|
+ error = kgdb_arch_set_breakpoint(&kgdb_break[i]);
|
|
if (error) {
|
|
if (error) {
|
|
ret = error;
|
|
ret = error;
|
|
- printk(KERN_INFO "KGDB: BP install failed: %lx", addr);
|
|
|
|
|
|
+ printk(KERN_INFO "KGDB: BP install failed: %lx",
|
|
|
|
+ kgdb_break[i].bpt_addr);
|
|
continue;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
|
|
- kgdb_flush_swbreak_addr(addr);
|
|
|
|
|
|
+ kgdb_flush_swbreak_addr(kgdb_break[i].bpt_addr);
|
|
kgdb_break[i].state = BP_ACTIVE;
|
|
kgdb_break[i].state = BP_ACTIVE;
|
|
}
|
|
}
|
|
return ret;
|
|
return ret;
|
|
@@ -302,7 +302,6 @@ int dbg_set_sw_break(unsigned long addr)
|
|
|
|
|
|
int dbg_deactivate_sw_breakpoints(void)
|
|
int dbg_deactivate_sw_breakpoints(void)
|
|
{
|
|
{
|
|
- unsigned long addr;
|
|
|
|
int error;
|
|
int error;
|
|
int ret = 0;
|
|
int ret = 0;
|
|
int i;
|
|
int i;
|
|
@@ -310,15 +309,14 @@ int dbg_deactivate_sw_breakpoints(void)
|
|
for (i = 0; i < KGDB_MAX_BREAKPOINTS; i++) {
|
|
for (i = 0; i < KGDB_MAX_BREAKPOINTS; i++) {
|
|
if (kgdb_break[i].state != BP_ACTIVE)
|
|
if (kgdb_break[i].state != BP_ACTIVE)
|
|
continue;
|
|
continue;
|
|
- addr = kgdb_break[i].bpt_addr;
|
|
|
|
- error = kgdb_arch_remove_breakpoint(addr,
|
|
|
|
- kgdb_break[i].saved_instr);
|
|
|
|
|
|
+ error = kgdb_arch_remove_breakpoint(&kgdb_break[i]);
|
|
if (error) {
|
|
if (error) {
|
|
- printk(KERN_INFO "KGDB: BP remove failed: %lx\n", addr);
|
|
|
|
|
|
+ printk(KERN_INFO "KGDB: BP remove failed: %lx\n",
|
|
|
|
+ kgdb_break[i].bpt_addr);
|
|
ret = error;
|
|
ret = error;
|
|
}
|
|
}
|
|
|
|
|
|
- kgdb_flush_swbreak_addr(addr);
|
|
|
|
|
|
+ kgdb_flush_swbreak_addr(kgdb_break[i].bpt_addr);
|
|
kgdb_break[i].state = BP_SET;
|
|
kgdb_break[i].state = BP_SET;
|
|
}
|
|
}
|
|
return ret;
|
|
return ret;
|
|
@@ -352,7 +350,6 @@ int kgdb_isremovedbreak(unsigned long addr)
|
|
|
|
|
|
int dbg_remove_all_break(void)
|
|
int dbg_remove_all_break(void)
|
|
{
|
|
{
|
|
- unsigned long addr;
|
|
|
|
int error;
|
|
int error;
|
|
int i;
|
|
int i;
|
|
|
|
|
|
@@ -360,12 +357,10 @@ int dbg_remove_all_break(void)
|
|
for (i = 0; i < KGDB_MAX_BREAKPOINTS; i++) {
|
|
for (i = 0; i < KGDB_MAX_BREAKPOINTS; i++) {
|
|
if (kgdb_break[i].state != BP_ACTIVE)
|
|
if (kgdb_break[i].state != BP_ACTIVE)
|
|
goto setundefined;
|
|
goto setundefined;
|
|
- addr = kgdb_break[i].bpt_addr;
|
|
|
|
- error = kgdb_arch_remove_breakpoint(addr,
|
|
|
|
- kgdb_break[i].saved_instr);
|
|
|
|
|
|
+ error = kgdb_arch_remove_breakpoint(&kgdb_break[i]);
|
|
if (error)
|
|
if (error)
|
|
printk(KERN_ERR "KGDB: breakpoint remove failed: %lx\n",
|
|
printk(KERN_ERR "KGDB: breakpoint remove failed: %lx\n",
|
|
- addr);
|
|
|
|
|
|
+ kgdb_break[i].bpt_addr);
|
|
setundefined:
|
|
setundefined:
|
|
kgdb_break[i].state = BP_UNDEFINED;
|
|
kgdb_break[i].state = BP_UNDEFINED;
|
|
}
|
|
}
|