Commit 1c8e6077 authored by Richard Musiol's avatar Richard Musiol Committed by Richard Musiol

runtime: do not omit stack trace of goroutine that handles async events

On wasm there is a special goroutine that handles asynchronous events.
Blocking this goroutine often causes a deadlock. However, the stack
trace of this goroutine was omitted when printing the deadlock error.

This change adds an exception so the goroutine is not considered as
an internal system goroutine and the stack trace gets printed, which
helps with debugging the deadlock.

Updates #32764

Change-Id: Icc8f5ba3ca5a485d557b7bdd76bf2f1ffb92eb3e
Reviewed-on: https://go-review.googlesource.com/c/go/+/199537
Run-TryBot: Richard Musiol <neelance@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarCherry Zhang <cherryyz@google.com>
parent 30521d51
...@@ -37,6 +37,7 @@ const ( ...@@ -37,6 +37,7 @@ const (
FuncID_debugCallV1 FuncID_debugCallV1
FuncID_gopanic FuncID_gopanic
FuncID_panicwrap FuncID_panicwrap
FuncID_handleAsyncEvents
FuncID_wrapper // any autogenerated code (hash/eq algorithms, method wrappers, etc.) FuncID_wrapper // any autogenerated code (hash/eq algorithms, method wrappers, etc.)
) )
...@@ -82,6 +83,8 @@ func GetFuncID(name, file string) FuncID { ...@@ -82,6 +83,8 @@ func GetFuncID(name, file string) FuncID {
return FuncID_gopanic return FuncID_gopanic
case "runtime.panicwrap": case "runtime.panicwrap":
return FuncID_panicwrap return FuncID_panicwrap
case "runtime.handleAsyncEvents":
return FuncID_handleAsyncEvents
} }
if file == "<autogenerated>" { if file == "<autogenerated>" {
return FuncID_wrapper return FuncID_wrapper
......
...@@ -149,7 +149,11 @@ var returnedEventHandler *g ...@@ -149,7 +149,11 @@ var returnedEventHandler *g
func init() { func init() {
// At the toplevel we need an extra goroutine that handles asynchronous events. // At the toplevel we need an extra goroutine that handles asynchronous events.
initg := getg() initg := getg()
go func() { go handleAsyncEvents(initg)
gopark(nil, nil, waitReasonZero, traceEvNone, 1)
}
func handleAsyncEvents(initg *g) {
returnedEventHandler = getg() returnedEventHandler = getg()
goready(initg, 1) goready(initg, 1)
...@@ -157,8 +161,6 @@ func init() { ...@@ -157,8 +161,6 @@ func init() {
returnedEventHandler = nil returnedEventHandler = nil
pause(getcallersp() - 16) pause(getcallersp() - 16)
}()
gopark(nil, nil, waitReasonZero, traceEvNone, 1)
} }
// beforeIdle gets called by the scheduler if no goroutine is awake. // beforeIdle gets called by the scheduler if no goroutine is awake.
......
...@@ -253,6 +253,7 @@ const ( ...@@ -253,6 +253,7 @@ const (
funcID_debugCallV1 funcID_debugCallV1
funcID_gopanic funcID_gopanic
funcID_panicwrap funcID_panicwrap
funcID_handleAsyncEvents
funcID_wrapper // any autogenerated code (hash/eq algorithms, method wrappers, etc.) funcID_wrapper // any autogenerated code (hash/eq algorithms, method wrappers, etc.)
) )
......
...@@ -997,8 +997,8 @@ func topofstack(f funcInfo, g0 bool) bool { ...@@ -997,8 +997,8 @@ func topofstack(f funcInfo, g0 bool) bool {
// isSystemGoroutine reports whether the goroutine g must be omitted // isSystemGoroutine reports whether the goroutine g must be omitted
// in stack dumps and deadlock detector. This is any goroutine that // in stack dumps and deadlock detector. This is any goroutine that
// starts at a runtime.* entry point, except for runtime.main and // starts at a runtime.* entry point, except for runtime.main,
// sometimes runtime.runfinq. // runtime.handleAsyncEvents (wasm only) and sometimes runtime.runfinq.
// //
// If fixed is true, any goroutine that can vary between user and // If fixed is true, any goroutine that can vary between user and
// system (that is, the finalizer goroutine) is considered a user // system (that is, the finalizer goroutine) is considered a user
...@@ -1009,7 +1009,7 @@ func isSystemGoroutine(gp *g, fixed bool) bool { ...@@ -1009,7 +1009,7 @@ func isSystemGoroutine(gp *g, fixed bool) bool {
if !f.valid() { if !f.valid() {
return false return false
} }
if f.funcID == funcID_runtime_main { if f.funcID == funcID_runtime_main || f.funcID == funcID_handleAsyncEvents {
return false return false
} }
if f.funcID == funcID_runfinq { if f.funcID == funcID_runfinq {
......
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