From 1e583e6e5b19160210644905f5cd74687c7457d3 Mon Sep 17 00:00:00 2001 From: Hajime Hoshi Date: Mon, 4 Nov 2024 14:34:08 +0900 Subject: [PATCH] internal/graphicsdriver/opengl: bug fix: use GLX when possible for ES Use GLX when possible. EGL with an X window might not work well on Wayland unfortunately. Closes #3152 --- .../graphicsdriver/opengl/graphics_glfw.go | 56 ++++++++++++++++++- 1 file changed, 54 insertions(+), 2 deletions(-) diff --git a/internal/graphicsdriver/opengl/graphics_glfw.go b/internal/graphicsdriver/opengl/graphics_glfw.go index 3456ad574..63ee3c16e 100644 --- a/internal/graphicsdriver/opengl/graphics_glfw.go +++ b/internal/graphicsdriver/opengl/graphics_glfw.go @@ -17,8 +17,12 @@ package opengl import ( + "bufio" + "bytes" "fmt" + "os/exec" "runtime" + "strings" "github.com/hajimehoshi/ebiten/v2/internal/glfw" "github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver" @@ -26,6 +30,50 @@ import ( "github.com/hajimehoshi/ebiten/v2/internal/microsoftgdk" ) +func isGLXExtensionForGL2Available() bool { + if runtime.GOOS == "windows" || runtime.GOOS == "darwin" { + return false + } + + var buf bytes.Buffer + cmd := exec.Command("glxinfo") + cmd.Stdout = &buf + if err := cmd.Run(); err != nil { + return false + } + + const ( + indent = " " + ext = "GLX_EXT_create_context_es2_profile" + ) + + var listingExtensions bool + s := bufio.NewScanner(&buf) + for s.Scan() { + if !listingExtensions { + if s.Text() == "GLX extensions:" { + listingExtensions = true + } + continue + } + + if !strings.HasPrefix(s.Text(), indent) { + listingExtensions = false + break + } + + line := s.Text() + for len(line) > 0 { + head, tail, _ := strings.Cut(line, ",") + if strings.TrimSpace(head) == ext { + return true + } + line = tail + } + } + return false +} + type graphicsPlatform struct { window *glfw.Window } @@ -60,8 +108,12 @@ func setGLFWClientAPI(isES bool) error { if err := glfw.WindowHint(glfw.ContextVersionMinor, 0); err != nil { return err } - if err := glfw.WindowHint(glfw.ContextCreationAPI, glfw.EGLContextAPI); err != nil { - return err + // Use GLX if the extension allows, or use EGL otherwise. + // Prefer GLX since EGL might not work well on Wayland (#3152). + if !isGLXExtensionForGL2Available() { + if err := glfw.WindowHint(glfw.ContextCreationAPI, glfw.EGLContextAPI); err != nil { + return err + } } return nil }