-
Notifications
You must be signed in to change notification settings - Fork 17.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
runtime: network connection timeout reported too late in a GUI application #50356
Comments
Thanks for the report. What is it that brings you to the conclusion that this is a Go problem, as opposed to a problem with glfw? |
my own code doesn't uses glfw, but directly cocoa api, someday i find this problem, and reproduce it with glfw. i could only think that the problem may be come from go runtime currently, as it stuck at the read call and doesn't wake up even deadline reached. there may be some conflicts with glfw's waitevents and go runtime, but i couldn't find and solve it |
What happens if you simply omit the glfw calls? They don't seem to have anything to do with the rest of your program. |
It works well if i remove glfw related code and replace the demo i provided is just to reproduce this issue. it's actually extracted from my bidirectional rpc library, when connection established, both client and server conn could receive or send rpc call. the demo only contains the heartbeat checking |
NSEvent *event = [NSApp nextEventMatchingMask:NSEventMaskAny
untilDate:[NSDate distantFuture]
inMode:NSDefaultRunLoopMode
dequeue:YES]; both glfw and my code contains this code to block on event loop, does it conflict with go runtime? |
I have just port this demo to gioui and also found this issue, which internally uses I have imagine that the bug may come from my code, but as i said before, it works well if i remove GUI related code. |
If glfw blocks in the event loop in a way that Go does not understand, that will indeed break timeouts and scheduling in general. That said, it's not obvious to me why that is happening here. |
I have make a more simple demo: package main
import (
"github.com/go-gl/glfw/v3.3/glfw"
"log"
"runtime"
"time"
)
func main() {
runtime.LockOSThread()
err := glfw.Init()
if err != nil {
log.Fatalln(err)
}
defer glfw.Terminate()
w, err := glfw.CreateWindow(300, 300, "Test", nil, nil)
if err != nil {
log.Fatalln(err)
}
defer w.Destroy()
go func() {
ticker := time.NewTicker(time.Second * 3)
for range ticker.C {
log.Println("timer")
}
}()
for !w.ShouldClose() {
glfw.WaitEvents()
}
} it just contains a timer to print then i run this program and put the opened window into background,
the runtime scheduling are breaking:
|
there seems no issue if i keeping the window on foreground or wake it up from background. |
I'm sorry, I have no idea. It appears that the problem is related to glfw, so perhaps this bug should move to the glfw bug tracker so that the glfw maintainers can take a look at it. |
it's also occurred with gioui(https://github.com/gioui/gio), which doesn't uses glfw. package main
// A simple Gio program. See https://gioui.org for more information.
import (
"log"
"os"
"time"
"gioui.org/app"
"gioui.org/io/system"
)
func main() {
go func() {
ticker := time.NewTicker(time.Second * 3)
for range ticker.C {
log.Println("timer")
}
}()
go func() {
w := app.NewWindow()
if err := loop(w); err != nil {
log.Fatal(err)
}
os.Exit(0)
}()
app.Main()
}
func loop(w *app.Window) error {
for {
e := <-w.Events()
switch e := e.(type) {
case system.DestroyEvent:
return e.Err
case system.FrameEvent:
}
}
} the problem may come from Apple's Cocoa framework or macOS itself. |
A new demo directly calls Cocoa api: package main
/*
#include <AppKit/AppKit.h>
#cgo CFLAGS: -x objective-c
#cgo LDFLAGS: -framework AppKit
@interface AppDelegate: NSObject <NSApplicationDelegate>
@end
@implementation AppDelegate
- (void)applicationDidFinishLaunching:(NSNotification *)notification {
NSWindow *w = [[NSWindow alloc] initWithContentRect:NSMakeRect(0, 0, 300,300)
styleMask:NSWindowStyleMaskTitled|NSWindowStyleMaskClosable|NSWindowStyleMaskMiniaturizable|NSWindowStyleMaskResizable
backing:NSBackingStoreBuffered
defer:FALSE];
[w center];
[w makeKeyAndOrderFront:w];
[NSApp setActivationPolicy:NSApplicationActivationPolicyRegular];
[NSApp activateIgnoringOtherApps:TRUE];
}
@end
void runApp() {
[NSApplication sharedApplication];
AppDelegate *delegate = [[AppDelegate alloc] init];
[NSApp setDelegate:delegate];
[NSApp run];
}
*/
import "C"
import (
"log"
"runtime"
"time"
)
func main() {
runtime.LockOSThread()
go func() {
ticker := time.NewTicker(time.Second * 3)
for range ticker.C {
log.Println("timer")
}
}()
C.runApp()
} |
I added some logs around this line in go runtime Line 232 in 91e7821
start:=nanotime()
println("sleepStart",ns)
res:= semasleep(ns)
println("sleepDone",nanotime()-start) the logs shows that it may sleep extra 10s:
|
The problem is that you are not using lockosthread correctly. To run the GUI on the main thread you need to do it slightly different, like by using this library: https://github.com/faiface/mainthread. Otherwise the goroutine you start will be on the same main OS thread as the GUI library, which leads to the Goroutine getting blocked due to GUI library blocking that thread. Edit: also the is thread needs to be locked in an init function, and not in main, where we might already not be on the main thread anymore. |
@ianlancetaylor thank you, i have found the answer here https://stackoverflow.com/a/49681111/4023435, it's because of App Nap. @beoran it should be ok to I'm going close this issue, thanks all for help. |
@zhuah It depends on the operating and windowing system. For example, on Linux with X11 it is ok to run the GUI on the same OS thread that started the GUI, but on other OS or windowing systems all graphics must be run from the first main thread only, and then you need the library above. But yes, in this case the problem was due to app nap. Similar problems can arise on mobile systems that agressively attempt to save power. |
@beoran Got it, thank you ! |
What version of Go are you using (
go version
)?Does this issue reproduce with the latest release?
Yes
What operating system and processor architecture are you using (
go env
)?go env
OutputWhat did you do?
I created a gui app, which embedded a server, but the network connection will accidentally timeout after running a while.
i have create a simple demo to reproduce it:
Steps to reproduce: running the program, then put the glfw window to background, wait for a while(1min ~2min),
the program will exited with error:
if i remove this block:
then the connection will timeout twice, which also cause it to be closed.
It seems only occurred in GUI app, i can't reproduce it in a normal server program without GUI.
What did you expect to see?
What did you see instead?
The text was updated successfully, but these errors were encountered: