GL/ContextWGL: Fix running surfaceless

This commit is contained in:
Connor McLaughlin 2022-10-22 21:15:45 +10:00 committed by refractionpcsx2
parent f5e64232c9
commit bf2575c3c9
2 changed files with 49 additions and 11 deletions

View File

@ -28,6 +28,17 @@ static void* GetProcAddressCallback(const char* name)
return ::GetProcAddress(GetModuleHandleA("opengl32.dll"), name); return ::GetProcAddress(GetModuleHandleA("opengl32.dll"), name);
} }
static bool ReloadWGL(HDC dc)
{
if (!gladLoadWGLLoader([](const char* name) -> void* { return wglGetProcAddress(name); }, dc))
{
Console.Error("Loading GLAD WGL functions failed");
return false;
}
return true;
}
namespace GL namespace GL
{ {
ContextWGL::ContextWGL(const WindowInfo& wi) ContextWGL::ContextWGL(const WindowInfo& wi)
@ -64,8 +75,8 @@ namespace GL
} }
else else
{ {
Console.Error("ContextWGL must always start with a valid surface."); if (!CreatePBuffer())
return false; return false;
} }
// Everything including core/ES requires a dummy profile to load the WGL extensions. // Everything including core/ES requires a dummy profile to load the WGL extensions.
@ -161,8 +172,8 @@ namespace GL
} }
else else
{ {
Console.Error("PBuffer not implemented"); if (!context->CreatePBuffer())
return nullptr; return nullptr;
} }
if (m_version.profile == Profile::NoProfile) if (m_version.profile == Profile::NoProfile)
@ -317,6 +328,29 @@ namespace GL
static constexpr const int pb_attribs[] = {0, 0}; static constexpr const int pb_attribs[] = {0, 0};
HGLRC temp_rc = nullptr;
ScopedGuard temp_rc_guard([&temp_rc, hdc]() { if (temp_rc) {
wglMakeCurrent(hdc, nullptr);
wglDeleteContext(temp_rc);
} });
if (!GLAD_WGL_ARB_pbuffer)
{
// we're probably running completely surfaceless... need a temporary context.
temp_rc = wglCreateContext(hdc);
if (!temp_rc || !wglMakeCurrent(hdc, temp_rc))
{
Console.Error("Failed to create temporary context to load WGL for pbuffer.");
return false;
}
if (!ReloadWGL(hdc) || !GLAD_WGL_ARB_pbuffer)
{
Console.Error("Missing WGL_ARB_pbuffer");
return false;
}
}
pxAssertRel(m_pixel_format.has_value(), "Has pixel format for pbuffer"); pxAssertRel(m_pixel_format.has_value(), "Has pixel format for pbuffer");
HPBUFFERARB pbuffer = wglCreatePbufferARB(hdc, m_pixel_format.value(), 1, 1, pb_attribs); HPBUFFERARB pbuffer = wglCreatePbufferARB(hdc, m_pixel_format.value(), 1, 1, pb_attribs);
if (!pbuffer) if (!pbuffer)
@ -338,6 +372,7 @@ namespace GL
m_dummy_dc = hdc; m_dummy_dc = hdc;
m_pbuffer = pbuffer; m_pbuffer = pbuffer;
temp_rc_guard.Run();
pbuffer_guard.Cancel(); pbuffer_guard.Cancel();
hdc_guard.Cancel(); hdc_guard.Cancel();
hwnd_guard.Cancel(); hwnd_guard.Cancel();
@ -440,11 +475,8 @@ namespace GL
} }
// re-init glad-wgl // re-init glad-wgl
if (make_current && !gladLoadWGLLoader([](const char* name) -> void* { return wglGetProcAddress(name); }, m_dc)) if (make_current && !ReloadWGL(m_dc))
{
Console.Error("Loading GLAD WGL functions failed");
return false; return false;
}
wglDeleteContext(m_rc); wglDeleteContext(m_rc);
} }

View File

@ -36,6 +36,15 @@ public:
} }
__fi ~ScopedGuard() __fi ~ScopedGuard()
{
Run();
}
ScopedGuard(const ScopedGuard&) = delete;
void operator=(const ScopedGuard&) = delete;
/// Runs the destructor function now instead of when we go out of scope.
__fi void Run()
{ {
if (!m_func.has_value()) if (!m_func.has_value())
return; return;
@ -44,9 +53,6 @@ public:
m_func.reset(); m_func.reset();
} }
ScopedGuard(const ScopedGuard&) = delete;
void operator=(const ScopedGuard&) = delete;
/// Prevents the function from being invoked when we go out of scope. /// Prevents the function from being invoked when we go out of scope.
__fi void Cancel() __fi void Cancel()
{ {