aboutsummaryrefslogtreecommitdiff
path: root/src/frontend/duckstation/gl/context.cpp
diff options
context:
space:
mode:
authorRSDuck <RSDuck@users.noreply.github.com>2022-10-17 22:55:11 +0200
committerGitHub <noreply@github.com>2022-10-17 22:55:11 +0200
commitac3118cbc54715da55856958b42817fbd1eed666 (patch)
tree717e66ec7df79171821f46e02b47d1357256ace2 /src/frontend/duckstation/gl/context.cpp
parent31ba585d392c9745917a141c72b3349d029586f0 (diff)
No more context mess (#1531)
* WIP: use Duckstation's context code to directly render into QT Widget from separate thread without two OpenGL contexts currently only works on Windows * reenable gay OSD * add back vsync * make it atleast a little more thread safe * linux support * don't segfault on closing * reorganise and cleanup build system it's still not good, but better than before * macos? * try to get it working on Ubuntu CI also update instructions * let's try this * ok how about this * try creating an OGL 4.3 context first (https://i.kym-cdn.com/photos/images/original/001/264/842/220.png) * fix Ubuntu * hm * try again for Windows * let's try this * make the OpenGL renderer work again that was stupid * do OGL surface resizing from the mainthread * Fix small mistake in GL context creation on macOS causing version 3.2 to be considered invalid * C stupidness * cleanup * don't let the emuthread deinit OGL if there's no OGL * reset lastScreenWidth/Height when deiniting OpenGL * disable stencil test while drawing framebuffers * macOS: Link Cocoa framework explicitly when not building with Qt6 Seems to be needed for the classes used by DuckStation's GL context code. * Set ScreenPanelGL's minimum size immediately Fixes GL context creation for OpenGL display on macOS using the wrong size as the underlying window was not resized to the correct size by Qt yet. * don't emit window updates when OGL display is used * stuff Arisotura said Co-authored-by: Nadia Holmquist Pedersen <nadia@nhp.sh>
Diffstat (limited to 'src/frontend/duckstation/gl/context.cpp')
-rw-r--r--src/frontend/duckstation/gl/context.cpp175
1 files changed, 175 insertions, 0 deletions
diff --git a/src/frontend/duckstation/gl/context.cpp b/src/frontend/duckstation/gl/context.cpp
new file mode 100644
index 0000000..98e6bd1
--- /dev/null
+++ b/src/frontend/duckstation/gl/context.cpp
@@ -0,0 +1,175 @@
+#include "context.h"
+#include "../log.h"
+#include "loader.h"
+#include <cstdlib>
+#include <cstring>
+#ifdef __APPLE__
+#include <stdlib.h>
+#else
+#include <malloc.h>
+#endif
+Log_SetChannel(GL::Context);
+
+#if defined(_WIN32) && !defined(_M_ARM64)
+#include "context_wgl.h"
+#elif defined(__APPLE__)
+#include "context_agl.h"
+#else
+#include "context_egl_wayland.h"
+#include "context_egl_x11.h"
+#include "context_glx.h"
+#endif
+
+namespace GL {
+
+static bool ShouldPreferESContext()
+{
+#ifndef _MSC_VER
+ const char* value = std::getenv("PREFER_GLES_CONTEXT");
+ return (value && strcmp(value, "1") == 0);
+#else
+ char buffer[2] = {};
+ size_t buffer_size = sizeof(buffer);
+ getenv_s(&buffer_size, buffer, "PREFER_GLES_CONTEXT");
+ return (std::strcmp(buffer, "1") == 0);
+#endif
+}
+
+Context::Context(const WindowInfo& wi) : m_wi(wi) {}
+
+Context::~Context() = default;
+
+std::vector<Context::FullscreenModeInfo> Context::EnumerateFullscreenModes()
+{
+ return {};
+}
+
+std::unique_ptr<GL::Context> Context::Create(const WindowInfo& wi, const Version* versions_to_try,
+ size_t num_versions_to_try)
+{
+ if (ShouldPreferESContext())
+ {
+ // move ES versions to the front
+ Version* new_versions_to_try = static_cast<Version*>(alloca(sizeof(Version) * num_versions_to_try));
+ size_t count = 0;
+ for (size_t i = 0; i < num_versions_to_try; i++)
+ {
+ if (versions_to_try[i].profile == Profile::ES)
+ new_versions_to_try[count++] = versions_to_try[i];
+ }
+ for (size_t i = 0; i < num_versions_to_try; i++)
+ {
+ if (versions_to_try[i].profile != Profile::ES)
+ new_versions_to_try[count++] = versions_to_try[i];
+ }
+ versions_to_try = new_versions_to_try;
+ }
+
+ std::unique_ptr<Context> context;
+#if defined(_WIN32) && !defined(_M_ARM64)
+ context = ContextWGL::Create(wi, versions_to_try, num_versions_to_try);
+#elif defined(__APPLE__)
+ context = ContextAGL::Create(wi, versions_to_try, num_versions_to_try);
+#else
+ if (wi.type == WindowInfo::Type::X11)
+ {
+ const char* use_egl_x11 = std::getenv("USE_EGL_X11");
+ if (use_egl_x11 && std::strcmp(use_egl_x11, "1") == 0)
+ context = ContextEGLX11::Create(wi, versions_to_try, num_versions_to_try);
+ else
+ context = ContextGLX::Create(wi, versions_to_try, num_versions_to_try);
+ }
+
+ if (wi.type == WindowInfo::Type::Wayland)
+ context = ContextEGLWayland::Create(wi, versions_to_try, num_versions_to_try);
+#endif
+
+ if (!context)
+ return nullptr;
+
+ Log_InfoPrintf("Created a %s context", context->IsGLES() ? "OpenGL ES" : "OpenGL");
+
+ // TODO: Not thread-safe.
+ static Context* context_being_created;
+ context_being_created = context.get();
+
+ if (!gladLoadGLLoader([](const char* name) { return context_being_created->GetProcAddress(name); }))
+ {
+ Log_ErrorPrintf("Failed to load GL functions for GLAD");
+ return nullptr;
+ }
+
+ const char* gl_vendor = reinterpret_cast<const char*>(glGetString(GL_VENDOR));
+ const char* gl_renderer = reinterpret_cast<const char*>(glGetString(GL_RENDERER));
+ const char* gl_version = reinterpret_cast<const char*>(glGetString(GL_VERSION));
+ const char* gl_shading_language_version = reinterpret_cast<const char*>(glGetString(GL_SHADING_LANGUAGE_VERSION));
+ Log_InfoPrintf("GL_VENDOR: %s", gl_vendor);
+ Log_InfoPrintf("GL_RENDERER: %s", gl_renderer);
+ Log_InfoPrintf("GL_VERSION: %s", gl_version);
+ Log_InfoPrintf("GL_SHADING_LANGUAGE_VERSION: %s", gl_shading_language_version);
+
+ return context;
+}
+
+const std::array<Context::Version, 11>& Context::GetAllDesktopVersionsList()
+{
+ static constexpr std::array<Version, 11> vlist = {{{Profile::Core, 4, 6},
+ {Profile::Core, 4, 5},
+ {Profile::Core, 4, 4},
+ {Profile::Core, 4, 3},
+ {Profile::Core, 4, 2},
+ {Profile::Core, 4, 1},
+ {Profile::Core, 4, 0},
+ {Profile::Core, 3, 3},
+ {Profile::Core, 3, 2},
+ {Profile::Core, 3, 1},
+ {Profile::Core, 3, 0}}};
+ return vlist;
+}
+
+const std::array<Context::Version, 12>& Context::GetAllDesktopVersionsListWithFallback()
+{
+ static constexpr std::array<Version, 12> vlist = {{{Profile::Core, 4, 6},
+ {Profile::Core, 4, 5},
+ {Profile::Core, 4, 4},
+ {Profile::Core, 4, 3},
+ {Profile::Core, 4, 2},
+ {Profile::Core, 4, 1},
+ {Profile::Core, 4, 0},
+ {Profile::Core, 3, 3},
+ {Profile::Core, 3, 2},
+ {Profile::Core, 3, 1},
+ {Profile::Core, 3, 0},
+ {Profile::NoProfile, 0, 0}}};
+ return vlist;
+}
+
+const std::array<Context::Version, 4>& Context::GetAllESVersionsList()
+{
+ static constexpr std::array<Version, 4> vlist = {
+ {{Profile::ES, 3, 2}, {Profile::ES, 3, 1}, {Profile::ES, 3, 0}, {Profile::ES, 2, 0}}};
+ return vlist;
+}
+
+const std::array<Context::Version, 16>& Context::GetAllVersionsList()
+{
+ static constexpr std::array<Version, 16> vlist = {{{Profile::Core, 4, 6},
+ {Profile::Core, 4, 5},
+ {Profile::Core, 4, 4},
+ {Profile::Core, 4, 3},
+ {Profile::Core, 4, 2},
+ {Profile::Core, 4, 1},
+ {Profile::Core, 4, 0},
+ {Profile::Core, 3, 3},
+ {Profile::Core, 3, 2},
+ {Profile::Core, 3, 1},
+ {Profile::Core, 3, 0},
+ {Profile::ES, 3, 2},
+ {Profile::ES, 3, 1},
+ {Profile::ES, 3, 0},
+ {Profile::ES, 2, 0},
+ {Profile::NoProfile, 0, 0}}};
+ return vlist;
+}
+
+} // namespace GL