Commit c3e54f09 authored by Russ Cox's avatar Russ Cox

runtime: better trace for fault due to nil pointer call

R=r
CC=golang-dev
https://golang.org/cl/854048
parent 03526597
...@@ -66,10 +66,18 @@ sighandler(int32 sig, Siginfo *info, void *context) ...@@ -66,10 +66,18 @@ sighandler(int32 sig, Siginfo *info, void *context)
gp->sigcode0 = info->si_code; gp->sigcode0 = info->si_code;
gp->sigcode1 = (uintptr)info->si_addr; gp->sigcode1 = (uintptr)info->si_addr;
sp = (uintptr*)r->esp; // Only push sigpanic if r->eip != 0.
*--sp = r->eip; // If r->eip == 0, probably panicked because of a
// call to a nil func. Not pushing that onto sp will
// make the trace look like a call to sigpanic instead.
// (Otherwise the trace will end at sigpanic and we
// won't get to see who faulted.)
if(r->eip != 0) {
sp = (uintptr*)r->esp;
*--sp = r->eip;
r->esp = (uintptr)sp;
}
r->eip = (uintptr)sigpanic; r->eip = (uintptr)sigpanic;
r->esp = (uintptr)sp;
return; return;
} }
......
...@@ -74,11 +74,19 @@ sighandler(int32 sig, Siginfo *info, void *context) ...@@ -74,11 +74,19 @@ sighandler(int32 sig, Siginfo *info, void *context)
gp->sig = sig; gp->sig = sig;
gp->sigcode0 = info->si_code; gp->sigcode0 = info->si_code;
gp->sigcode1 = (uintptr)info->si_addr; gp->sigcode1 = (uintptr)info->si_addr;
sp = (uintptr*)r->rsp; // Only push sigpanic if r->rip != 0.
*--sp = r->rip; // If r->rip == 0, probably panicked because of a
// call to a nil func. Not pushing that onto sp will
// make the trace look like a call to sigpanic instead.
// (Otherwise the trace will end at sigpanic and we
// won't get to see who faulted.)
if(r->rip != 0) {
sp = (uintptr*)r->rsp;
*--sp = r->rip;
r->rsp = (uintptr)sp;
}
r->rip = (uintptr)sigpanic; r->rip = (uintptr)sigpanic;
r->rsp = (uintptr)sp;
return; return;
} }
......
...@@ -64,10 +64,18 @@ sighandler(int32 sig, Siginfo* info, void* context) ...@@ -64,10 +64,18 @@ sighandler(int32 sig, Siginfo* info, void* context)
gp->sigcode0 = info->si_code; gp->sigcode0 = info->si_code;
gp->sigcode1 = (uintptr)info->si_addr; gp->sigcode1 = (uintptr)info->si_addr;
sp = (uintptr*)r->mc_esp; // Only push sigpanic if r->mc_eip != 0.
*--sp = r->mc_eip; // If r->mc_eip == 0, probably panicked because of a
// call to a nil func. Not pushing that onto sp will
// make the trace look like a call to sigpanic instead.
// (Otherwise the trace will end at sigpanic and we
// won't get to see who faulted.)
if(r->mc_eip != 0) {
sp = (uintptr*)r->mc_esp;
*--sp = r->mc_eip;
r->mc_esp = (uintptr)sp;
}
r->mc_eip = (uintptr)sigpanic; r->mc_eip = (uintptr)sigpanic;
r->mc_esp = (uintptr)sp;
return; return;
} }
......
...@@ -72,10 +72,18 @@ sighandler(int32 sig, Siginfo* info, void* context) ...@@ -72,10 +72,18 @@ sighandler(int32 sig, Siginfo* info, void* context)
gp->sigcode0 = info->si_code; gp->sigcode0 = info->si_code;
gp->sigcode1 = (uintptr)info->si_addr; gp->sigcode1 = (uintptr)info->si_addr;
sp = (uintptr*)r->mc_rsp; // Only push sigpanic if r->mc_rip != 0.
*--sp = r->mc_rip; // If r->mc_rip == 0, probably panicked because of a
// call to a nil func. Not pushing that onto sp will
// make the trace look like a call to sigpanic instead.
// (Otherwise the trace will end at sigpanic and we
// won't get to see who faulted.)
if(r->mc_rip != 0) {
sp = (uintptr*)r->mc_rsp;
*--sp = r->mc_rip;
r->mc_rsp = (uintptr)sp;
}
r->mc_rip = (uintptr)sigpanic; r->mc_rip = (uintptr)sigpanic;
r->mc_rsp = (uintptr)sp;
return; return;
} }
......
...@@ -61,10 +61,18 @@ sighandler(int32 sig, Siginfo* info, void* context) ...@@ -61,10 +61,18 @@ sighandler(int32 sig, Siginfo* info, void* context)
gp->sigcode0 = info->si_code; gp->sigcode0 = info->si_code;
gp->sigcode1 = ((uintptr*)info)[3]; gp->sigcode1 = ((uintptr*)info)[3];
sp = (uintptr*)r->esp; // Only push sigpanic if r->eip != 0.
*--sp = r->eip; // If r->eip == 0, probably panicked because of a
// call to a nil func. Not pushing that onto sp will
// make the trace look like a call to sigpanic instead.
// (Otherwise the trace will end at sigpanic and we
// won't get to see who faulted.)
if(r->eip != 0) {
sp = (uintptr*)r->esp;
*--sp = r->eip;
r->esp = (uintptr)sp;
}
r->eip = (uintptr)sigpanic; r->eip = (uintptr)sigpanic;
r->esp = (uintptr)sp;
return; return;
} }
......
...@@ -71,10 +71,18 @@ sighandler(int32 sig, Siginfo* info, void* context) ...@@ -71,10 +71,18 @@ sighandler(int32 sig, Siginfo* info, void* context)
gp->sigcode0 = info->si_code; gp->sigcode0 = info->si_code;
gp->sigcode1 = ((uintptr*)info)[2]; gp->sigcode1 = ((uintptr*)info)[2];
sp = (uintptr*)r->rsp; // Only push sigpanic if r->rip != 0.
*--sp = r->rip; // If r->rip == 0, probably panicked because of a
// call to a nil func. Not pushing that onto sp will
// make the trace look like a call to sigpanic instead.
// (Otherwise the trace will end at sigpanic and we
// won't get to see who faulted.)
if(r->rip != 0) {
sp = (uintptr*)r->rsp;
*--sp = r->rip;
r->rsp = (uintptr)sp;
}
r->rip = (uintptr)sigpanic; r->rip = (uintptr)sigpanic;
r->rsp = (uintptr)sp;
return; return;
} }
......
...@@ -70,7 +70,11 @@ sighandler(int32 sig, Siginfo *info, void *context) ...@@ -70,7 +70,11 @@ sighandler(int32 sig, Siginfo *info, void *context)
// If this is a leaf function, we do smash LR, // If this is a leaf function, we do smash LR,
// but we're not going back there anyway. // but we're not going back there anyway.
r->arm_lr = r->arm_pc; // Don't bother smashing if r->arm_pc is 0,
// which is probably a call to a nil func: the
// old link register is more useful in the stack trace.
if(r->arm_pc != 0)
r->arm_lr = r->arm_pc;
r->arm_pc = (uintptr)sigpanic; r->arm_pc = (uintptr)sigpanic;
return; return;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment