1
0
mirror of https://github.com/mpv-player/mpv synced 2025-01-11 17:39:38 +00:00

win32: create OpenGL context on the GUI thread

This fixes the fullscreen issues on Intel for me. I'm baffled by it and
don't understand why this suddenly works. Intel drivers being shit?
Windows being shit? HWND or HDC being only 97% thread-safe instead of
98%? Me missing something subtle that is not documented anywhere?

Who knows.

Now instead of creating the HDC and OpenGL context on the renderer
thread, they're created on the GUI thread. The renderer thread will
then only call wglMakeCurrent, SwapBuffers, and OpenGL standard
functions.

Probably fixes github issue #968.
This commit is contained in:
wm4 2014-08-05 19:40:57 +02:00
parent afcbee8305
commit 210c83ab31
3 changed files with 40 additions and 12 deletions

View File

@ -27,6 +27,7 @@
struct w32_context { struct w32_context {
HGLRC context; HGLRC context;
HDC hdc; HDC hdc;
int flags;
}; };
static bool create_dc(struct MPGLContext *ctx, int flags) static bool create_dc(struct MPGLContext *ctx, int flags)
@ -193,6 +194,21 @@ out:
return false; return false;
} }
static void create_ctx(void *ptr)
{
struct MPGLContext *ctx = ptr;
struct w32_context *w32_ctx = ctx->priv;
if (!create_dc(ctx, w32_ctx->flags))
return;
if (ctx->requested_gl_version >= MPGL_VER(3, 0))
create_context_w32_gl3(ctx);
if (!w32_ctx->context)
create_context_w32_old(ctx);
wglMakeCurrent(w32_ctx->hdc, NULL);
}
static bool config_window_w32(struct MPGLContext *ctx, int flags) static bool config_window_w32(struct MPGLContext *ctx, int flags)
{ {
struct w32_context *w32_ctx = ctx->priv; struct w32_context *w32_ctx = ctx->priv;
@ -202,30 +218,35 @@ static bool config_window_w32(struct MPGLContext *ctx, int flags)
if (w32_ctx->context) // reuse existing context if (w32_ctx->context) // reuse existing context
return true; return true;
if (!create_dc(ctx, flags)) w32_ctx->flags = flags;
return false; vo_w32_run_on_thread(ctx->vo, create_ctx, ctx);
bool success = false; if (w32_ctx->context)
if (ctx->requested_gl_version >= MPGL_VER(3, 0)) wglMakeCurrent(w32_ctx->hdc, w32_ctx->context);
success = create_context_w32_gl3(ctx); return !!w32_ctx->context;
if (!success)
success = create_context_w32_old(ctx);
return success;
} }
static void releaseGlContext_w32(MPGLContext *ctx) static void destroy_gl(void *ptr)
{ {
struct MPGLContext *ctx = ptr;
struct w32_context *w32_ctx = ctx->priv; struct w32_context *w32_ctx = ctx->priv;
if (w32_ctx->context) { if (w32_ctx->context)
wglMakeCurrent(w32_ctx->hdc, 0);
wglDeleteContext(w32_ctx->context); wglDeleteContext(w32_ctx->context);
}
w32_ctx->context = 0; w32_ctx->context = 0;
if (w32_ctx->hdc) if (w32_ctx->hdc)
ReleaseDC(vo_w32_hwnd(ctx->vo), w32_ctx->hdc); ReleaseDC(vo_w32_hwnd(ctx->vo), w32_ctx->hdc);
w32_ctx->hdc = NULL; w32_ctx->hdc = NULL;
} }
static void releaseGlContext_w32(MPGLContext *ctx)
{
struct MPGLContext *ctx = ptr;
struct w32_context *w32_ctx = ctx->priv;
if (w32_ctx->context)
wglMakeCurrent(w32_ctx->hdc, 0);
vo_w32_run_on_thread(ctx->vo, destroy_gl, ctx);
}
static void swapGlBuffers_w32(MPGLContext *ctx) static void swapGlBuffers_w32(MPGLContext *ctx)
{ {
struct w32_context *w32_ctx = ctx->priv; struct w32_context *w32_ctx = ctx->priv;

View File

@ -1202,3 +1202,9 @@ HWND vo_w32_hwnd(struct vo *vo)
struct vo_w32_state *w32 = vo->w32; struct vo_w32_state *w32 = vo->w32;
return w32->window; // immutable, so no synchronization needed return w32->window; // immutable, so no synchronization needed
} }
void vo_w32_run_on_thread(struct vo *vo, void (*cb)(void *ctx), void *ctx)
{
struct vo_w32_state *w32 = vo->w32;
mp_dispatch_run(w32->dispatch, cb, ctx);
}

View File

@ -32,5 +32,6 @@ void vo_w32_uninit(struct vo *vo);
int vo_w32_control(struct vo *vo, int *events, int request, void *arg); int vo_w32_control(struct vo *vo, int *events, int request, void *arg);
int vo_w32_config(struct vo *vo, uint32_t); int vo_w32_config(struct vo *vo, uint32_t);
HWND vo_w32_hwnd(struct vo *vo); HWND vo_w32_hwnd(struct vo *vo);
void vo_w32_run_on_thread(struct vo *vo, void (*cb)(void *ctx), void *ctx);
#endif /* MPLAYER_W32_COMMON_H */ #endif /* MPLAYER_W32_COMMON_H */