2012-10-03 16:25:41 +00:00
|
|
|
/*
|
|
|
|
* This file is part of mplayer.
|
|
|
|
*
|
|
|
|
* mplayer is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* mplayer is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with mplayer. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <libavutil/common.h>
|
|
|
|
|
|
|
|
#include "bitmap_packer.h"
|
|
|
|
|
|
|
|
#include "gl_osd.h"
|
|
|
|
|
|
|
|
struct osd_fmt_entry {
|
|
|
|
GLint internal_format;
|
|
|
|
GLint format;
|
|
|
|
GLenum type;
|
|
|
|
};
|
|
|
|
|
vo_opengl: fix alpha values written to the framebuffer
When blending OSD and subtitles onto the video, we write bogus alpha
values. This doesn't normally matter, because these values are normally
unused and discarded. But at least on Wayland, the alpha values are used
by the compositor and leads to transparent windows even with opaque
video on places where the OSD happens to use transparency.
(Also see github issue #338.)
Until now, the alpha basically contained garbage. The source factor
GL_SRC_ALPHA meant that alpha was multiplied with itself. Use GL_ONE
instead (which is why we have to use glBlendFuncSeparate()). This should
give correct results, even with video that has alpha. (Or at least it's
something close to correct, I haven't thought too hard how the
compositor will blend it, and in fact I couldn't manage to test it.)
If glBlendFuncSeparate() is not available, fall back to glBlendFunc(),
which does the same as the code did before this commit. Technically, we
support GL 1.1, but glBlendFuncSeparate is 1.4, and I guess we should
try not to crash if vo_opengl_old runs on a system with GL 1.1 drivers
only.
2013-11-10 02:14:13 +00:00
|
|
|
// glBlendFuncSeparate() arguments
|
|
|
|
static const int blend_factors[SUBBITMAP_COUNT][4] = {
|
|
|
|
[SUBBITMAP_LIBASS] = {GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA,
|
|
|
|
GL_ONE, GL_ONE_MINUS_SRC_ALPHA},
|
|
|
|
[SUBBITMAP_RGBA] = {GL_ONE, GL_ONE_MINUS_SRC_ALPHA,
|
|
|
|
GL_ONE, GL_ONE_MINUS_SRC_ALPHA},
|
2012-10-03 16:25:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static const struct osd_fmt_entry osd_to_gl3_formats[SUBBITMAP_COUNT] = {
|
|
|
|
[SUBBITMAP_LIBASS] = {GL_RED, GL_RED, GL_UNSIGNED_BYTE},
|
2014-12-20 18:23:17 +00:00
|
|
|
[SUBBITMAP_RGBA] = {GL_RGBA, GL_RGBA, GL_UNSIGNED_BYTE},
|
2012-10-03 16:25:41 +00:00
|
|
|
};
|
|
|
|
|
2014-12-17 20:48:23 +00:00
|
|
|
static const struct osd_fmt_entry osd_to_gles3_formats[SUBBITMAP_COUNT] = {
|
|
|
|
[SUBBITMAP_LIBASS] = {GL_R8, GL_RED, GL_UNSIGNED_BYTE},
|
2014-12-20 18:23:17 +00:00
|
|
|
[SUBBITMAP_RGBA] = {GL_RGBA8, GL_RGBA, GL_UNSIGNED_BYTE},
|
2014-12-17 20:48:23 +00:00
|
|
|
};
|
|
|
|
|
2014-12-20 17:43:11 +00:00
|
|
|
static const struct osd_fmt_entry osd_to_gl2_formats[SUBBITMAP_COUNT] = {
|
2014-12-18 13:46:19 +00:00
|
|
|
[SUBBITMAP_LIBASS] = {GL_LUMINANCE, GL_LUMINANCE, GL_UNSIGNED_BYTE},
|
2014-12-20 18:23:17 +00:00
|
|
|
[SUBBITMAP_RGBA] = {GL_RGBA, GL_RGBA, GL_UNSIGNED_BYTE},
|
2012-10-03 16:25:41 +00:00
|
|
|
};
|
|
|
|
|
2015-01-21 19:32:42 +00:00
|
|
|
struct mpgl_osd *mpgl_osd_init(GL *gl, struct mp_log *log, struct osd_state *osd)
|
2012-10-03 16:25:41 +00:00
|
|
|
{
|
|
|
|
GLint max_texture_size;
|
|
|
|
gl->GetIntegerv(GL_MAX_TEXTURE_SIZE, &max_texture_size);
|
|
|
|
|
|
|
|
struct mpgl_osd *ctx = talloc_ptrtype(NULL, ctx);
|
|
|
|
*ctx = (struct mpgl_osd) {
|
2013-09-11 23:33:33 +00:00
|
|
|
.log = log,
|
2014-06-15 18:46:57 +00:00
|
|
|
.osd = osd,
|
2012-10-03 16:25:41 +00:00
|
|
|
.gl = gl,
|
2014-06-15 18:46:57 +00:00
|
|
|
.fmt_table = osd_to_gl3_formats,
|
2012-10-03 16:25:41 +00:00
|
|
|
.scratch = talloc_zero_size(ctx, 1),
|
|
|
|
};
|
|
|
|
|
2015-01-21 19:32:42 +00:00
|
|
|
if (gl->es >= 300) {
|
2014-12-17 20:48:23 +00:00
|
|
|
ctx->fmt_table = osd_to_gles3_formats;
|
|
|
|
} else if (!(gl->mpgl_caps & MPGL_CAP_TEX_RG)) {
|
2014-12-20 17:43:11 +00:00
|
|
|
ctx->fmt_table = osd_to_gl2_formats;
|
2014-12-17 20:48:23 +00:00
|
|
|
}
|
2014-12-16 17:55:02 +00:00
|
|
|
|
2012-10-03 16:25:41 +00:00
|
|
|
for (int n = 0; n < MAX_OSD_PARTS; n++) {
|
|
|
|
struct mpgl_osd_part *p = talloc_ptrtype(ctx, p);
|
|
|
|
*p = (struct mpgl_osd_part) {
|
|
|
|
.packer = talloc_struct(p, struct bitmap_packer, {
|
|
|
|
.w_max = max_texture_size,
|
|
|
|
.h_max = max_texture_size,
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
ctx->parts[n] = p;
|
|
|
|
}
|
|
|
|
|
VO, sub: refactor
Remove VFCTRL_DRAW_OSD, VFCAP_EOSD_FILTER, VFCAP_EOSD_RGBA, VFCAP_EOSD,
VOCTRL_DRAW_EOSD, VOCTRL_GET_EOSD_RES, VOCTRL_QUERY_EOSD_FORMAT.
Remove draw_osd_with_eosd(), which rendered the OSD by calling
VOCTRL_DRAW_EOSD. Change VOs to call osd_draw() directly, which takes
a callback as argument. (This basically works like the old OSD API,
except multiple OSD bitmap formats are supported and caching is
possible.)
Remove all mentions of "eosd". It's simply "osd" now.
Make OSD size per-OSD-object, as they can be different when using
vf_sub. Include display_par/video_par in resolution change detection.
Fix the issue with margin borders in vo_corevideo.
2012-10-19 17:25:18 +00:00
|
|
|
for (int n = 0; n < SUBBITMAP_COUNT; n++)
|
|
|
|
ctx->formats[n] = ctx->fmt_table[n].type != 0;
|
|
|
|
|
2012-10-03 16:25:41 +00:00
|
|
|
return ctx;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mpgl_osd_destroy(struct mpgl_osd *ctx)
|
|
|
|
{
|
2014-06-15 18:46:57 +00:00
|
|
|
if (!ctx)
|
|
|
|
return;
|
|
|
|
|
2012-10-03 16:25:41 +00:00
|
|
|
GL *gl = ctx->gl;
|
|
|
|
|
|
|
|
for (int n = 0; n < MAX_OSD_PARTS; n++) {
|
|
|
|
struct mpgl_osd_part *p = ctx->parts[n];
|
|
|
|
gl->DeleteTextures(1, &p->texture);
|
|
|
|
if (gl->DeleteBuffers)
|
|
|
|
gl->DeleteBuffers(1, &p->buffer);
|
|
|
|
}
|
|
|
|
talloc_free(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool upload_pbo(struct mpgl_osd *ctx, struct mpgl_osd_part *osd,
|
|
|
|
struct sub_bitmaps *imgs)
|
|
|
|
{
|
|
|
|
GL *gl = ctx->gl;
|
|
|
|
bool success = true;
|
|
|
|
struct osd_fmt_entry fmt = ctx->fmt_table[imgs->format];
|
|
|
|
int pix_stride = glFmt2bpp(fmt.format, fmt.type);
|
|
|
|
|
|
|
|
if (!osd->buffer) {
|
|
|
|
gl->GenBuffers(1, &osd->buffer);
|
|
|
|
gl->BindBuffer(GL_PIXEL_UNPACK_BUFFER, osd->buffer);
|
|
|
|
gl->BufferData(GL_PIXEL_UNPACK_BUFFER, osd->w * osd->h * pix_stride,
|
|
|
|
NULL, GL_DYNAMIC_COPY);
|
|
|
|
gl->BindBuffer(GL_PIXEL_UNPACK_BUFFER, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
gl->BindBuffer(GL_PIXEL_UNPACK_BUFFER, osd->buffer);
|
|
|
|
char *data = gl->MapBuffer(GL_PIXEL_UNPACK_BUFFER, GL_WRITE_ONLY);
|
|
|
|
if (!data) {
|
|
|
|
success = false;
|
|
|
|
} else {
|
|
|
|
struct pos bb[2];
|
|
|
|
packer_get_bb(osd->packer, bb);
|
|
|
|
size_t stride = osd->w * pix_stride;
|
|
|
|
packer_copy_subbitmaps(osd->packer, imgs, data, pix_stride, stride);
|
|
|
|
if (!gl->UnmapBuffer(GL_PIXEL_UNPACK_BUFFER))
|
|
|
|
success = false;
|
|
|
|
glUploadTex(gl, GL_TEXTURE_2D, fmt.format, fmt.type, NULL, stride,
|
|
|
|
bb[0].x, bb[0].y, bb[1].x - bb[0].x, bb[1].y - bb[0].y,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
gl->BindBuffer(GL_PIXEL_UNPACK_BUFFER, 0);
|
|
|
|
|
|
|
|
if (!success) {
|
2013-09-11 23:33:33 +00:00
|
|
|
MP_FATAL(ctx, "Error: can't upload subtitles! "
|
|
|
|
"Remove the 'pbo' suboption.\n");
|
2012-10-03 16:25:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void upload_tex(struct mpgl_osd *ctx, struct mpgl_osd_part *osd,
|
|
|
|
struct sub_bitmaps *imgs)
|
|
|
|
{
|
|
|
|
struct osd_fmt_entry fmt = ctx->fmt_table[imgs->format];
|
|
|
|
if (osd->packer->padding) {
|
|
|
|
struct pos bb[2];
|
|
|
|
packer_get_bb(osd->packer, bb);
|
|
|
|
glClearTex(ctx->gl, GL_TEXTURE_2D, fmt.format, fmt.type,
|
|
|
|
bb[0].x, bb[0].y, bb[1].x - bb[0].y, bb[1].y - bb[0].y,
|
|
|
|
0, &ctx->scratch);
|
|
|
|
}
|
|
|
|
for (int n = 0; n < osd->packer->count; n++) {
|
|
|
|
struct sub_bitmap *s = &imgs->parts[n];
|
|
|
|
struct pos p = osd->packer->result[n];
|
|
|
|
|
|
|
|
glUploadTex(ctx->gl, GL_TEXTURE_2D, fmt.format, fmt.type,
|
|
|
|
s->bitmap, s->stride, p.x, p.y, s->w, s->h, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool upload_osd(struct mpgl_osd *ctx, struct mpgl_osd_part *osd,
|
|
|
|
struct sub_bitmaps *imgs)
|
|
|
|
{
|
|
|
|
GL *gl = ctx->gl;
|
|
|
|
|
2012-10-04 00:29:54 +00:00
|
|
|
// assume 2x2 filter on scaling
|
|
|
|
osd->packer->padding = ctx->scaled || imgs->scaled;
|
2012-10-03 16:25:41 +00:00
|
|
|
int r = packer_pack_from_subbitmaps(osd->packer, imgs);
|
|
|
|
if (r < 0) {
|
2013-09-11 23:33:33 +00:00
|
|
|
MP_ERR(ctx, "OSD bitmaps do not fit on a surface with the maximum "
|
|
|
|
"supported size %dx%d.\n", osd->packer->w_max, osd->packer->h_max);
|
2012-10-03 16:25:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct osd_fmt_entry fmt = ctx->fmt_table[imgs->format];
|
|
|
|
assert(fmt.type != 0);
|
|
|
|
|
|
|
|
if (!osd->texture)
|
|
|
|
gl->GenTextures(1, &osd->texture);
|
|
|
|
|
|
|
|
gl->BindTexture(GL_TEXTURE_2D, osd->texture);
|
|
|
|
|
|
|
|
if (osd->packer->w > osd->w || osd->packer->h > osd->h
|
|
|
|
|| osd->format != imgs->format)
|
|
|
|
{
|
|
|
|
osd->format = imgs->format;
|
|
|
|
osd->w = FFMAX(32, osd->packer->w);
|
|
|
|
osd->h = FFMAX(32, osd->packer->h);
|
|
|
|
|
|
|
|
gl->TexImage2D(GL_TEXTURE_2D, 0, fmt.internal_format, osd->w, osd->h,
|
|
|
|
0, fmt.format, fmt.type, NULL);
|
|
|
|
|
|
|
|
gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
|
|
|
|
if (gl->DeleteBuffers)
|
|
|
|
gl->DeleteBuffers(1, &osd->buffer);
|
|
|
|
osd->buffer = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool uploaded = false;
|
|
|
|
if (ctx->use_pbo)
|
|
|
|
uploaded = upload_pbo(ctx, osd, imgs);
|
|
|
|
if (!uploaded)
|
|
|
|
upload_tex(ctx, osd, imgs);
|
|
|
|
|
|
|
|
gl->BindTexture(GL_TEXTURE_2D, 0);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-06-15 21:58:33 +00:00
|
|
|
struct mpgl_osd_part *mpgl_osd_generate(struct mpgl_osd *ctx,
|
|
|
|
struct sub_bitmaps *imgs)
|
2012-10-03 16:25:41 +00:00
|
|
|
{
|
VO, sub: refactor
Remove VFCTRL_DRAW_OSD, VFCAP_EOSD_FILTER, VFCAP_EOSD_RGBA, VFCAP_EOSD,
VOCTRL_DRAW_EOSD, VOCTRL_GET_EOSD_RES, VOCTRL_QUERY_EOSD_FORMAT.
Remove draw_osd_with_eosd(), which rendered the OSD by calling
VOCTRL_DRAW_EOSD. Change VOs to call osd_draw() directly, which takes
a callback as argument. (This basically works like the old OSD API,
except multiple OSD bitmap formats are supported and caching is
possible.)
Remove all mentions of "eosd". It's simply "osd" now.
Make OSD size per-OSD-object, as they can be different when using
vf_sub. Include display_par/video_par in resolution change detection.
Fix the issue with margin borders in vo_corevideo.
2012-10-19 17:25:18 +00:00
|
|
|
if (imgs->num_parts == 0 || !ctx->formats[imgs->format])
|
2012-10-03 16:25:41 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
struct mpgl_osd_part *osd = ctx->parts[imgs->render_index];
|
|
|
|
|
|
|
|
if (imgs->bitmap_pos_id != osd->bitmap_pos_id) {
|
|
|
|
if (imgs->bitmap_id != osd->bitmap_id) {
|
|
|
|
if (!upload_osd(ctx, osd, imgs))
|
|
|
|
osd->packer->count = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
osd->bitmap_id = imgs->bitmap_id;
|
|
|
|
osd->bitmap_pos_id = imgs->bitmap_pos_id;
|
|
|
|
osd->num_vertices = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return osd->packer->count ? osd : NULL;
|
|
|
|
}
|
|
|
|
|
VO, sub: refactor
Remove VFCTRL_DRAW_OSD, VFCAP_EOSD_FILTER, VFCAP_EOSD_RGBA, VFCAP_EOSD,
VOCTRL_DRAW_EOSD, VOCTRL_GET_EOSD_RES, VOCTRL_QUERY_EOSD_FORMAT.
Remove draw_osd_with_eosd(), which rendered the OSD by calling
VOCTRL_DRAW_EOSD. Change VOs to call osd_draw() directly, which takes
a callback as argument. (This basically works like the old OSD API,
except multiple OSD bitmap formats are supported and caching is
possible.)
Remove all mentions of "eosd". It's simply "osd" now.
Make OSD size per-OSD-object, as they can be different when using
vf_sub. Include display_par/video_par in resolution change detection.
Fix the issue with margin borders in vo_corevideo.
2012-10-19 17:25:18 +00:00
|
|
|
void mpgl_osd_set_gl_state(struct mpgl_osd *ctx, struct mpgl_osd_part *p)
|
2012-10-03 16:25:41 +00:00
|
|
|
{
|
|
|
|
GL *gl = ctx->gl;
|
|
|
|
|
|
|
|
gl->BindTexture(GL_TEXTURE_2D, p->texture);
|
|
|
|
gl->Enable(GL_BLEND);
|
vo_opengl: fix alpha values written to the framebuffer
When blending OSD and subtitles onto the video, we write bogus alpha
values. This doesn't normally matter, because these values are normally
unused and discarded. But at least on Wayland, the alpha values are used
by the compositor and leads to transparent windows even with opaque
video on places where the OSD happens to use transparency.
(Also see github issue #338.)
Until now, the alpha basically contained garbage. The source factor
GL_SRC_ALPHA meant that alpha was multiplied with itself. Use GL_ONE
instead (which is why we have to use glBlendFuncSeparate()). This should
give correct results, even with video that has alpha. (Or at least it's
something close to correct, I haven't thought too hard how the
compositor will blend it, and in fact I couldn't manage to test it.)
If glBlendFuncSeparate() is not available, fall back to glBlendFunc(),
which does the same as the code did before this commit. Technically, we
support GL 1.1, but glBlendFuncSeparate is 1.4, and I guess we should
try not to crash if vo_opengl_old runs on a system with GL 1.1 drivers
only.
2013-11-10 02:14:13 +00:00
|
|
|
|
|
|
|
const int *factors = &blend_factors[p->format][0];
|
2015-01-21 19:32:42 +00:00
|
|
|
gl->BlendFuncSeparate(factors[0], factors[1], factors[2], factors[3]);
|
2012-10-03 16:25:41 +00:00
|
|
|
}
|
|
|
|
|
VO, sub: refactor
Remove VFCTRL_DRAW_OSD, VFCAP_EOSD_FILTER, VFCAP_EOSD_RGBA, VFCAP_EOSD,
VOCTRL_DRAW_EOSD, VOCTRL_GET_EOSD_RES, VOCTRL_QUERY_EOSD_FORMAT.
Remove draw_osd_with_eosd(), which rendered the OSD by calling
VOCTRL_DRAW_EOSD. Change VOs to call osd_draw() directly, which takes
a callback as argument. (This basically works like the old OSD API,
except multiple OSD bitmap formats are supported and caching is
possible.)
Remove all mentions of "eosd". It's simply "osd" now.
Make OSD size per-OSD-object, as they can be different when using
vf_sub. Include display_par/video_par in resolution change detection.
Fix the issue with margin borders in vo_corevideo.
2012-10-19 17:25:18 +00:00
|
|
|
void mpgl_osd_unset_gl_state(struct mpgl_osd *ctx, struct mpgl_osd_part *p)
|
2012-10-03 16:25:41 +00:00
|
|
|
{
|
|
|
|
GL *gl = ctx->gl;
|
|
|
|
|
|
|
|
gl->Disable(GL_BLEND);
|
|
|
|
gl->BindTexture(GL_TEXTURE_2D, 0);
|
|
|
|
}
|