mirror of
https://github.com/mpv-player/mpv
synced 2025-01-10 00:49:32 +00:00
b037121430
The purpose of the new API is to make it useable with other APIs than OpenGL, especially D3D11 and vulkan. In theory it's now possible to support other vo_gpu backends, as well as backends that don't use the vo_gpu code at all. This also aims to get rid of the dumb mpv_get_sub_api() function. The life cycle of the new mpv_render_context is a bit different from mpv_opengl_cb_context, and you explicitly create/destroy the new context, instead of calling init/uninit on an object returned by mpv_get_sub_api(). In other to make the render API generic, it's annoyingly EGL style, and requires you to pass in API-specific objects to generic functions. This is to avoid explicit objects like the internal ra API has, because that sounds more complicated and annoying for an API that's supposed to never change. The opengl_cb API will continue to exist for a bit longer, but internally there are already a few tradeoffs, like reduced thread-safety. Mostly untested. Seems to work fine with mpc-qt.
114 lines
3.4 KiB
C
114 lines
3.4 KiB
C
#include "common.h"
|
|
#include "context.h"
|
|
#include "ra_gl.h"
|
|
#include "options/m_config.h"
|
|
#include "libmpv/render_gl.h"
|
|
#include "video/out/gpu/libmpv_gpu.h"
|
|
#include "video/out/gpu/ra.h"
|
|
|
|
struct priv {
|
|
GL *gl;
|
|
struct ra_ctx *ra_ctx;
|
|
};
|
|
|
|
static int init(struct libmpv_gpu_context *ctx, mpv_render_param *params)
|
|
{
|
|
ctx->priv = talloc_zero(NULL, struct priv);
|
|
struct priv *p = ctx->priv;
|
|
|
|
mpv_opengl_init_params *init_params =
|
|
get_mpv_render_param(params, MPV_RENDER_PARAM_OPENGL_INIT_PARAMS, NULL);
|
|
if (!init_params)
|
|
return MPV_ERROR_INVALID_PARAMETER;
|
|
|
|
p->gl = talloc_zero(p, GL);
|
|
|
|
mpgl_load_functions2(p->gl, init_params->get_proc_address,
|
|
init_params->get_proc_address_ctx,
|
|
init_params->extra_exts, ctx->log);
|
|
if (!p->gl->version && !p->gl->es) {
|
|
MP_FATAL(ctx, "OpenGL not initialized.\n");
|
|
return MPV_ERROR_UNSUPPORTED;
|
|
}
|
|
|
|
// initialize a blank ra_ctx to reuse ra_gl_ctx
|
|
p->ra_ctx = talloc_zero(p, struct ra_ctx);
|
|
p->ra_ctx->log = ctx->log;
|
|
p->ra_ctx->global = ctx->global;
|
|
p->ra_ctx->opts = (struct ra_ctx_opts) {
|
|
.probing = false,
|
|
.allow_sw = true,
|
|
};
|
|
|
|
static const struct ra_swapchain_fns empty_swapchain_fns = {0};
|
|
struct ra_gl_ctx_params gl_params = {
|
|
// vo_opengl_cb is essentially like a gigantic external swapchain where
|
|
// the user is in charge of presentation / swapping etc. But we don't
|
|
// actually need to provide any of these functions, since we can just
|
|
// not call them to begin with - so just set it to an empty object to
|
|
// signal to ra_gl_p that we don't care about its latency emulation
|
|
// functionality
|
|
.external_swapchain = &empty_swapchain_fns
|
|
};
|
|
|
|
p->gl->SwapInterval = NULL; // we shouldn't randomly change this, so lock it
|
|
if (!ra_gl_ctx_init(p->ra_ctx, p->gl, gl_params))
|
|
return MPV_ERROR_UNSUPPORTED;
|
|
|
|
int debug;
|
|
mp_read_option_raw(ctx->global, "gpu-debug", &m_option_type_flag, &debug);
|
|
p->ra_ctx->opts.debug = debug;
|
|
p->gl->debug_context = debug;
|
|
ra_gl_set_debug(p->ra_ctx->ra, debug);
|
|
|
|
ctx->ra = p->ra_ctx->ra;
|
|
return 0;
|
|
}
|
|
|
|
static int wrap_fbo(struct libmpv_gpu_context *ctx, mpv_render_param *params,
|
|
struct ra_tex **out)
|
|
{
|
|
struct priv *p = ctx->priv;
|
|
|
|
mpv_opengl_fbo *fbo =
|
|
get_mpv_render_param(params, MPV_RENDER_PARAM_OPENGL_FBO, NULL);
|
|
if (!fbo)
|
|
return MPV_ERROR_INVALID_PARAMETER;
|
|
|
|
if (fbo->fbo && !(p->gl->mpgl_caps & MPGL_CAP_FB)) {
|
|
MP_FATAL(ctx, "Rendering to FBO requested, but no FBO extension found!\n");
|
|
return MPV_ERROR_UNSUPPORTED;
|
|
}
|
|
|
|
struct ra_swapchain *sw = p->ra_ctx->swapchain;
|
|
struct ra_fbo target;
|
|
ra_gl_ctx_resize(sw, fbo->w, fbo->h, fbo->fbo);
|
|
ra_gl_ctx_start_frame(sw, &target);
|
|
*out = target.tex;
|
|
return 0;
|
|
}
|
|
|
|
static void done_frame(struct libmpv_gpu_context *ctx, bool ds)
|
|
{
|
|
struct priv *p = ctx->priv;
|
|
|
|
struct ra_swapchain *sw = p->ra_ctx->swapchain;
|
|
struct vo_frame dummy = {.display_synced = ds};
|
|
ra_gl_ctx_submit_frame(sw, &dummy);
|
|
}
|
|
|
|
static void destroy(struct libmpv_gpu_context *ctx)
|
|
{
|
|
struct priv *p = ctx->priv;
|
|
|
|
ra_gl_ctx_uninit(p->ra_ctx);
|
|
}
|
|
|
|
const struct libmpv_gpu_context_fns libmpv_gpu_context_gl = {
|
|
.api_name = MPV_RENDER_API_TYPE_OPENGL,
|
|
.init = init,
|
|
.wrap_fbo = wrap_fbo,
|
|
.done_frame = done_frame,
|
|
.destroy = destroy,
|
|
};
|