Commit 089d363a authored by Dmitry Vyukov's avatar Dmitry Vyukov

runtime: fix tracing of syscall exit

Fix tracing of syscall exit after:
https://go-review.googlesource.com/#/c/7504/

Change-Id: Idcde2aa826d2b9a05d0a90a80242b6bfa78846ab
Reviewed-on: https://go-review.googlesource.com/8728Reviewed-by: default avatarRick Hudson <rlh@golang.org>
Run-TryBot: Dmitry Vyukov <dvyukov@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent 53a8ee50
...@@ -235,6 +235,12 @@ func parseEvents(rawEvents []rawEvent) (events []*Event, err error) { ...@@ -235,6 +235,12 @@ func parseEvents(rawEvents []rawEvent) (events []*Event, err error) {
EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond, EvGoBlockNet, EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond, EvGoBlockNet,
EvGoSysBlock: EvGoSysBlock:
lastG = 0 lastG = 0
case EvGoSysExit:
if e.Args[1] != 0 {
// EvGoSysExit emission is delayed until the thread has a P.
// Give it the real timestamp.
e.Ts = int64(e.Args[1])
}
} }
events = append(events, e) events = append(events, e)
} }
...@@ -423,7 +429,12 @@ func postProcessTrace(events []*Event) error { ...@@ -423,7 +429,12 @@ func postProcessTrace(events []*Event) error {
g1.state = gWaiting g1.state = gWaiting
gs[ev.Args[0]] = g1 gs[ev.Args[0]] = g1
case EvGoInSyscall: case EvGoInSyscall:
// this case is intentionally left blank g1 := gs[ev.Args[0]]
if g1.state != gRunnable {
return fmt.Errorf("g %v is not runnable before EvGoInSyscall (offset %v, time %v)", ev.Args[0], ev.Off, ev.Ts)
}
g1.state = gWaiting
gs[ev.Args[0]] = g1
case EvGoCreate: case EvGoCreate:
if err := checkRunning(p, g, ev); err != nil { if err := checkRunning(p, g, ev); err != nil {
return err return err
...@@ -498,17 +509,18 @@ func postProcessTrace(events []*Event) error { ...@@ -498,17 +509,18 @@ func postProcessTrace(events []*Event) error {
if err := checkRunning(p, g, ev); err != nil { if err := checkRunning(p, g, ev); err != nil {
return err return err
} }
g.state = gRunnable g.state = gWaiting
g.evStart.Link = ev g.evStart.Link = ev
g.evStart = nil g.evStart = nil
p.g = 0 p.g = 0
case EvGoSysExit: case EvGoSysExit:
if g.state != gRunnable { if g.state != gWaiting {
return fmt.Errorf("g %v is not runnable during syscall exit (offset %v, time %v)", ev.G, ev.Off, ev.Ts) return fmt.Errorf("g %v is not waiting during syscall exit (offset %v, time %v)", ev.G, ev.Off, ev.Ts)
} }
if g.ev != nil && g.ev.Type == EvGoSysCall { if g.ev != nil && g.ev.Type == EvGoSysCall {
g.ev.Link = ev g.ev.Link = ev
} }
g.state = gRunnable
g.ev = ev g.ev = ev
case EvGoSleep, EvGoBlock, EvGoBlockSend, EvGoBlockRecv, case EvGoSleep, EvGoBlock, EvGoBlockSend, EvGoBlockRecv,
EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond, EvGoBlockNet: EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond, EvGoBlockNet:
...@@ -638,6 +650,18 @@ func (l eventList) Swap(i, j int) { ...@@ -638,6 +650,18 @@ func (l eventList) Swap(i, j int) {
l[i], l[j] = l[j], l[i] l[i], l[j] = l[j], l[i]
} }
// Print dumps events to stdout. For debugging.
func Print(events []*Event) {
for _, ev := range events {
desc := EventDescriptions[ev.Type]
fmt.Printf("%v %v p=%v g=%v off=%v", ev.Ts, desc.Name, ev.P, ev.G, ev.Off)
for i, a := range desc.Args {
fmt.Printf(" %v=%v", a, ev.Args[i])
}
fmt.Printf("\n")
}
}
// Event types in the trace. // Event types in the trace.
// Verbatim copy from src/runtime/trace.go. // Verbatim copy from src/runtime/trace.go.
const ( const (
...@@ -670,7 +694,7 @@ const ( ...@@ -670,7 +694,7 @@ const (
EvGoBlockCond = 26 // goroutine blocks on Cond [timestamp, stack] EvGoBlockCond = 26 // goroutine blocks on Cond [timestamp, stack]
EvGoBlockNet = 27 // goroutine blocks on network [timestamp, stack] EvGoBlockNet = 27 // goroutine blocks on network [timestamp, stack]
EvGoSysCall = 28 // syscall enter [timestamp, stack] EvGoSysCall = 28 // syscall enter [timestamp, stack]
EvGoSysExit = 29 // syscall exit [timestamp, goroutine id] EvGoSysExit = 29 // syscall exit [timestamp, goroutine id, real timestamp]
EvGoSysBlock = 30 // syscall blocks [timestamp] EvGoSysBlock = 30 // syscall blocks [timestamp]
EvGoWaiting = 31 // denotes that goroutine is blocked when tracing starts [goroutine id] EvGoWaiting = 31 // denotes that goroutine is blocked when tracing starts [goroutine id]
EvGoInSyscall = 32 // denotes that goroutine is in syscall when tracing starts [goroutine id] EvGoInSyscall = 32 // denotes that goroutine is in syscall when tracing starts [goroutine id]
...@@ -715,7 +739,7 @@ var EventDescriptions = [EvCount]struct { ...@@ -715,7 +739,7 @@ var EventDescriptions = [EvCount]struct {
EvGoBlockCond: {"GoBlockCond", true, []string{}}, EvGoBlockCond: {"GoBlockCond", true, []string{}},
EvGoBlockNet: {"GoBlockNet", true, []string{}}, EvGoBlockNet: {"GoBlockNet", true, []string{}},
EvGoSysCall: {"GoSysCall", true, []string{}}, EvGoSysCall: {"GoSysCall", true, []string{}},
EvGoSysExit: {"GoSysExit", false, []string{"g"}}, EvGoSysExit: {"GoSysExit", false, []string{"g", "ts"}},
EvGoSysBlock: {"GoSysBlock", false, []string{}}, EvGoSysBlock: {"GoSysBlock", false, []string{}},
EvGoWaiting: {"GoWaiting", false, []string{"g"}}, EvGoWaiting: {"GoWaiting", false, []string{"g"}},
EvGoInSyscall: {"GoInSyscall", false, []string{"g"}}, EvGoInSyscall: {"GoInSyscall", false, []string{"g"}},
......
...@@ -1813,17 +1813,18 @@ func exitsyscall(dummy int32) { ...@@ -1813,17 +1813,18 @@ func exitsyscall(dummy int32) {
return return
} }
var exitTicks int64
if trace.enabled { if trace.enabled {
// Wait till traceGoSysBlock event is emited. // Wait till traceGoSysBlock event is emited.
// This ensures consistency of the trace (the goroutine is started after it is blocked). // This ensures consistency of the trace (the goroutine is started after it is blocked).
for oldp != nil && oldp.syscalltick == _g_.m.syscalltick { for oldp != nil && oldp.syscalltick == _g_.m.syscalltick {
osyield() osyield()
} }
// This can't be done since the GC may be running and this code // We can't trace syscall exit right now because we don't have a P.
// will invoke write barriers. // Tracing code can invoke write barriers that cannot run without a P.
// TODO: Figure out how to get traceGoSysExit into the trace log or // So instead we remember the syscall exit time and emit the event
// it is likely not to work as expected. // below when we have a P.
// systemstack(traceGoSysExit) exitTicks = cputicks()
} }
_g_.m.locks-- _g_.m.locks--
...@@ -1831,6 +1832,11 @@ func exitsyscall(dummy int32) { ...@@ -1831,6 +1832,11 @@ func exitsyscall(dummy int32) {
// Call the scheduler. // Call the scheduler.
mcall(exitsyscall0) mcall(exitsyscall0)
// The goroutine must not be re-scheduled up to traceGoSysExit.
// Otherwise we can emit GoStart but not GoSysExit, that would lead
// no an inconsistent trace.
_g_.m.locks++
if _g_.m.mcache == nil { if _g_.m.mcache == nil {
throw("lost mcache") throw("lost mcache")
} }
...@@ -1844,6 +1850,13 @@ func exitsyscall(dummy int32) { ...@@ -1844,6 +1850,13 @@ func exitsyscall(dummy int32) {
_g_.syscallsp = 0 _g_.syscallsp = 0
_g_.m.p.syscalltick++ _g_.m.p.syscalltick++
_g_.throwsplit = false _g_.throwsplit = false
if exitTicks != 0 {
systemstack(func() {
traceGoSysExit(exitTicks)
})
}
_g_.m.locks--
} }
//go:nosplit //go:nosplit
...@@ -1871,7 +1884,7 @@ func exitsyscallfast() bool { ...@@ -1871,7 +1884,7 @@ func exitsyscallfast() bool {
// Denote blocking of the new syscall. // Denote blocking of the new syscall.
traceGoSysBlock(_g_.m.p) traceGoSysBlock(_g_.m.p)
// Denote completion of the current syscall. // Denote completion of the current syscall.
traceGoSysExit() traceGoSysExit(0)
}) })
} }
_g_.m.p.syscalltick++ _g_.m.p.syscalltick++
...@@ -1895,7 +1908,7 @@ func exitsyscallfast() bool { ...@@ -1895,7 +1908,7 @@ func exitsyscallfast() bool {
osyield() osyield()
} }
} }
traceGoSysExit() traceGoSysExit(0)
} }
}) })
if ok { if ok {
......
...@@ -45,7 +45,7 @@ const ( ...@@ -45,7 +45,7 @@ const (
traceEvGoBlockCond = 26 // goroutine blocks on Cond [timestamp, stack] traceEvGoBlockCond = 26 // goroutine blocks on Cond [timestamp, stack]
traceEvGoBlockNet = 27 // goroutine blocks on network [timestamp, stack] traceEvGoBlockNet = 27 // goroutine blocks on network [timestamp, stack]
traceEvGoSysCall = 28 // syscall enter [timestamp, stack] traceEvGoSysCall = 28 // syscall enter [timestamp, stack]
traceEvGoSysExit = 29 // syscall exit [timestamp, goroutine id] traceEvGoSysExit = 29 // syscall exit [timestamp, goroutine id, real timestamp]
traceEvGoSysBlock = 30 // syscall blocks [timestamp] traceEvGoSysBlock = 30 // syscall blocks [timestamp]
traceEvGoWaiting = 31 // denotes that goroutine is blocked when tracing starts [goroutine id] traceEvGoWaiting = 31 // denotes that goroutine is blocked when tracing starts [goroutine id]
traceEvGoInSyscall = 32 // denotes that goroutine is in syscall when tracing starts [goroutine id] traceEvGoInSyscall = 32 // denotes that goroutine is in syscall when tracing starts [goroutine id]
...@@ -797,8 +797,8 @@ func traceGoSysCall() { ...@@ -797,8 +797,8 @@ func traceGoSysCall() {
traceEvent(traceEvGoSysCall, 4) traceEvent(traceEvGoSysCall, 4)
} }
func traceGoSysExit() { func traceGoSysExit(ts int64) {
traceEvent(traceEvGoSysExit, -1, uint64(getg().m.curg.goid)) traceEvent(traceEvGoSysExit, -1, uint64(getg().m.curg.goid), uint64(ts)/traceTickDiv)
} }
func traceGoSysBlock(pp *p) { func traceGoSysBlock(pp *p) {
......
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