2008-09-07 14:09:51 +00:00
|
|
|
/*
|
|
|
|
* common OpenGL routines
|
2005-09-25 16:41:28 +00:00
|
|
|
*
|
2010-04-25 06:52:13 +00:00
|
|
|
* copyleft (C) 2005-2010 Reimar Döffinger <Reimar.Doeffinger@gmx.de>
|
2005-09-14 22:08:04 +00:00
|
|
|
* Special thanks go to the xine team and Matthias Hopf, whose video_out_opengl.c
|
|
|
|
* gave me lots of good ideas.
|
2008-09-07 14:09:51 +00:00
|
|
|
*
|
2015-04-13 07:36:54 +00:00
|
|
|
* This file is part of mpv.
|
2008-09-07 14:09:51 +00:00
|
|
|
*
|
2016-01-19 17:36:34 +00:00
|
|
|
* mpv is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2008-09-07 14:09:51 +00:00
|
|
|
*
|
2015-04-13 07:36:54 +00:00
|
|
|
* mpv is distributed in the hope that it will be useful,
|
2008-09-07 14:09:51 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2016-01-19 17:36:34 +00:00
|
|
|
* GNU Lesser General Public License for more details.
|
2010-06-13 10:42:32 +00:00
|
|
|
*
|
2016-01-19 17:36:34 +00:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with mpv. If not, see <http://www.gnu.org/licenses/>.
|
2005-09-14 22:08:04 +00:00
|
|
|
*/
|
2008-09-07 14:09:51 +00:00
|
|
|
|
2011-09-25 09:18:07 +00:00
|
|
|
#include <stddef.h>
|
2005-07-27 17:22:24 +00:00
|
|
|
#include <stdlib.h>
|
2005-09-14 22:08:04 +00:00
|
|
|
#include <stdio.h>
|
2005-07-27 17:22:24 +00:00
|
|
|
#include <string.h>
|
2012-03-30 22:33:53 +00:00
|
|
|
#include <stdbool.h>
|
2005-09-14 22:08:04 +00:00
|
|
|
#include <math.h>
|
2012-10-02 23:54:13 +00:00
|
|
|
#include <assert.h>
|
2015-12-19 11:59:07 +00:00
|
|
|
|
2015-08-29 02:12:56 +00:00
|
|
|
#include "common.h"
|
2014-07-13 18:12:13 +00:00
|
|
|
#include "common/common.h"
|
2017-10-02 15:30:27 +00:00
|
|
|
#include "utils.h"
|
2005-09-25 16:41:28 +00:00
|
|
|
|
2012-10-02 23:54:13 +00:00
|
|
|
// This guesses if the current GL context is a suspected software renderer.
|
|
|
|
static bool is_software_gl(GL *gl)
|
|
|
|
{
|
|
|
|
const char *renderer = gl->GetString(GL_RENDERER);
|
|
|
|
const char *vendor = gl->GetString(GL_VENDOR);
|
|
|
|
return !(renderer && vendor) ||
|
|
|
|
strcmp(renderer, "Software Rasterizer") == 0 ||
|
|
|
|
strstr(renderer, "llvmpipe") ||
|
2020-02-25 18:23:06 +00:00
|
|
|
strstr(renderer, "softpipe") ||
|
2012-10-02 23:54:13 +00:00
|
|
|
strcmp(vendor, "Microsoft Corporation") == 0 ||
|
2018-07-22 21:07:32 +00:00
|
|
|
strcmp(renderer, "Mesa X11") == 0 ||
|
|
|
|
strcmp(renderer, "Apple Software Renderer") == 0;
|
2012-10-02 23:54:13 +00:00
|
|
|
}
|
|
|
|
|
2014-12-09 21:36:28 +00:00
|
|
|
static void GLAPIENTRY dummy_glBindFramebuffer(GLenum target, GLuint framebuffer)
|
2014-12-09 21:28:16 +00:00
|
|
|
{
|
|
|
|
assert(framebuffer == 0);
|
|
|
|
}
|
|
|
|
|
2012-10-02 23:54:13 +00:00
|
|
|
#define FN_OFFS(name) offsetof(GL, name)
|
|
|
|
|
2015-03-25 11:40:45 +00:00
|
|
|
#define DEF_FN(name) {FN_OFFS(name), "gl" # name}
|
|
|
|
#define DEF_FN_NAME(name, str) {FN_OFFS(name), str}
|
2012-10-02 23:54:13 +00:00
|
|
|
|
|
|
|
struct gl_function {
|
|
|
|
ptrdiff_t offset;
|
2015-03-25 11:40:45 +00:00
|
|
|
char *name;
|
2006-01-07 19:53:51 +00:00
|
|
|
};
|
|
|
|
|
2012-10-02 23:54:13 +00:00
|
|
|
struct gl_functions {
|
|
|
|
const char *extension; // introduced with this extension in any version
|
|
|
|
int provides; // bitfield of MPGL_CAP_* constants
|
|
|
|
int ver_core; // introduced as required function
|
2014-12-18 22:51:21 +00:00
|
|
|
int ver_es_core; // introduced as required GL ES function
|
2016-05-12 18:08:49 +00:00
|
|
|
int ver_exclude; // not applicable to versions >= ver_exclude
|
|
|
|
int ver_es_exclude; // same for GLES
|
2014-06-10 21:56:05 +00:00
|
|
|
const struct gl_function *functions;
|
2012-10-02 23:54:13 +00:00
|
|
|
};
|
|
|
|
|
2015-01-21 19:32:42 +00:00
|
|
|
#define MAX_FN_COUNT 100 // max functions per gl_functions section
|
2012-10-02 23:54:13 +00:00
|
|
|
|
2014-12-18 22:51:21 +00:00
|
|
|
// Note: to keep the number of sections low, some functions are in multiple
|
|
|
|
// sections (if there are tricky combinations of GL/ES versions)
|
2014-06-10 21:56:05 +00:00
|
|
|
static const struct gl_functions gl_functions[] = {
|
2015-01-21 19:32:42 +00:00
|
|
|
// GL 2.1+ desktop and GLES 2.0+ (anything we support)
|
|
|
|
// Probably all of these are in GL 2.0 too, but we require GLSL 120.
|
2014-12-18 22:51:21 +00:00
|
|
|
{
|
2015-01-21 19:32:42 +00:00
|
|
|
.ver_core = 210,
|
2014-12-18 22:51:21 +00:00
|
|
|
.ver_es_core = 200,
|
2014-06-10 21:56:05 +00:00
|
|
|
.functions = (const struct gl_function[]) {
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(ActiveTexture),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(AttachShader),
|
|
|
|
DEF_FN(BindAttribLocation),
|
|
|
|
DEF_FN(BindBuffer),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(BindTexture),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(BlendFuncSeparate),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(BufferData),
|
2017-07-16 15:17:39 +00:00
|
|
|
DEF_FN(BufferSubData),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(Clear),
|
|
|
|
DEF_FN(ClearColor),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(CompileShader),
|
|
|
|
DEF_FN(CreateProgram),
|
|
|
|
DEF_FN(CreateShader),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(DeleteBuffers),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(DeleteProgram),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(DeleteShader),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(DeleteTextures),
|
|
|
|
DEF_FN(Disable),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(DisableVertexAttribArray),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(DrawArrays),
|
|
|
|
DEF_FN(Enable),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(EnableVertexAttribArray),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(Finish),
|
|
|
|
DEF_FN(Flush),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(GenBuffers),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(GenTextures),
|
|
|
|
DEF_FN(GetAttribLocation),
|
|
|
|
DEF_FN(GetError),
|
|
|
|
DEF_FN(GetIntegerv),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(GetProgramInfoLog),
|
|
|
|
DEF_FN(GetProgramiv),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(GetShaderInfoLog),
|
|
|
|
DEF_FN(GetShaderiv),
|
|
|
|
DEF_FN(GetString),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(GetUniformLocation),
|
|
|
|
DEF_FN(LinkProgram),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(PixelStorei),
|
|
|
|
DEF_FN(ReadPixels),
|
2016-09-12 13:08:38 +00:00
|
|
|
DEF_FN(Scissor),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(ShaderSource),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(TexImage2D),
|
|
|
|
DEF_FN(TexParameteri),
|
|
|
|
DEF_FN(TexSubImage2D),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(Uniform1f),
|
|
|
|
DEF_FN(Uniform2f),
|
|
|
|
DEF_FN(Uniform3f),
|
|
|
|
DEF_FN(Uniform1i),
|
2013-05-25 23:48:39 +00:00
|
|
|
DEF_FN(UniformMatrix2fv),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(UniformMatrix3fv),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(UseProgram),
|
|
|
|
DEF_FN(VertexAttribPointer),
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(Viewport),
|
|
|
|
{0}
|
2012-10-02 23:54:13 +00:00
|
|
|
},
|
|
|
|
},
|
2015-01-21 19:32:42 +00:00
|
|
|
// GL 2.1+ desktop only (and GLSL 120 shaders)
|
2012-10-02 23:54:13 +00:00
|
|
|
{
|
2014-12-18 22:51:21 +00:00
|
|
|
.ver_core = 210,
|
2015-11-19 20:21:04 +00:00
|
|
|
.provides = MPGL_CAP_ROW_LENGTH | MPGL_CAP_1D_TEX,
|
2014-06-10 21:56:05 +00:00
|
|
|
.functions = (const struct gl_function[]) {
|
2015-01-21 19:32:42 +00:00
|
|
|
DEF_FN(GetTexLevelParameteriv),
|
|
|
|
DEF_FN(ReadBuffer),
|
|
|
|
DEF_FN(TexImage1D),
|
2014-12-18 22:51:21 +00:00
|
|
|
DEF_FN(UnmapBuffer),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2016-05-23 16:02:37 +00:00
|
|
|
// GL 2.1 has this as extension only.
|
|
|
|
{
|
|
|
|
.ver_exclude = 300,
|
|
|
|
.ver_es_exclude = 300,
|
|
|
|
.extension = "GL_ARB_map_buffer_range",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(MapBufferRange),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2015-01-21 19:32:42 +00:00
|
|
|
// GL 3.0+ and ES 3.x core only functions.
|
2014-12-18 22:51:21 +00:00
|
|
|
{
|
|
|
|
.ver_core = 300,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.functions = (const struct gl_function[]) {
|
2015-11-06 12:59:33 +00:00
|
|
|
DEF_FN(BindBufferBase),
|
2015-11-15 17:30:54 +00:00
|
|
|
DEF_FN(BlitFramebuffer),
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(GetStringi),
|
2016-05-23 16:02:37 +00:00
|
|
|
DEF_FN(MapBufferRange),
|
2014-12-18 22:51:21 +00:00
|
|
|
// for ES 3.0
|
|
|
|
DEF_FN(ReadBuffer),
|
|
|
|
DEF_FN(UnmapBuffer),
|
2012-10-02 23:54:13 +00:00
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2015-11-19 09:18:39 +00:00
|
|
|
// For ES 3.1 core
|
|
|
|
{
|
|
|
|
.ver_es_core = 310,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(GetTexLevelParameteriv),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2015-11-19 20:21:04 +00:00
|
|
|
{
|
|
|
|
.ver_core = 210,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.provides = MPGL_CAP_3D_TEX,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(TexImage3D),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2014-12-18 23:58:56 +00:00
|
|
|
// Useful for ES 2.0
|
|
|
|
{
|
|
|
|
.ver_core = 110,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.extension = "GL_EXT_unpack_subimage",
|
|
|
|
.provides = MPGL_CAP_ROW_LENGTH,
|
|
|
|
},
|
2012-10-02 23:54:13 +00:00
|
|
|
// Framebuffers, extension in GL 2.x, core in GL 3.x core.
|
|
|
|
{
|
2014-12-18 22:51:21 +00:00
|
|
|
.ver_core = 300,
|
2015-07-27 21:27:49 +00:00
|
|
|
.ver_es_core = 200,
|
2012-10-02 23:54:13 +00:00
|
|
|
.extension = "GL_ARB_framebuffer_object",
|
|
|
|
.provides = MPGL_CAP_FB,
|
2014-06-10 21:56:05 +00:00
|
|
|
.functions = (const struct gl_function[]) {
|
2012-10-02 23:54:13 +00:00
|
|
|
DEF_FN(BindFramebuffer),
|
|
|
|
DEF_FN(GenFramebuffers),
|
|
|
|
DEF_FN(DeleteFramebuffers),
|
|
|
|
DEF_FN(CheckFramebufferStatus),
|
|
|
|
DEF_FN(FramebufferTexture2D),
|
2016-06-14 08:35:43 +00:00
|
|
|
DEF_FN(GetFramebufferAttachmentParameteriv),
|
2012-10-02 23:54:13 +00:00
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2014-12-23 13:32:24 +00:00
|
|
|
// VAOs, extension in GL 2.x, core in GL 3.x core.
|
|
|
|
{
|
|
|
|
.ver_core = 300,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.extension = "GL_ARB_vertex_array_object",
|
|
|
|
.provides = MPGL_CAP_VAO,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(GenVertexArrays),
|
|
|
|
DEF_FN(BindVertexArray),
|
|
|
|
DEF_FN(DeleteVertexArrays),
|
|
|
|
{0}
|
|
|
|
}
|
|
|
|
},
|
2012-10-02 23:54:13 +00:00
|
|
|
// GL_RED / GL_RG textures, extension in GL 2.x, core in GL 3.x core.
|
|
|
|
{
|
2014-12-18 22:51:21 +00:00
|
|
|
.ver_core = 300,
|
|
|
|
.ver_es_core = 300,
|
2012-10-02 23:54:13 +00:00
|
|
|
.extension = "GL_ARB_texture_rg",
|
|
|
|
.provides = MPGL_CAP_TEX_RG,
|
|
|
|
},
|
2016-05-12 18:42:12 +00:00
|
|
|
{
|
|
|
|
.ver_core = 300,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.extension = "GL_EXT_texture_rg",
|
|
|
|
.provides = MPGL_CAP_TEX_RG,
|
|
|
|
},
|
2016-04-27 17:14:10 +00:00
|
|
|
// GL_R16 etc.
|
|
|
|
{
|
|
|
|
.extension = "GL_EXT_texture_norm16",
|
|
|
|
.provides = MPGL_CAP_EXT16,
|
2016-05-12 18:08:49 +00:00
|
|
|
.ver_exclude = 1, // never in desktop GL
|
|
|
|
},
|
|
|
|
// Float texture support for GL 2.x
|
|
|
|
{
|
|
|
|
.extension = "GL_ARB_texture_float",
|
|
|
|
.provides = MPGL_CAP_ARB_FLOAT,
|
|
|
|
.ver_exclude = 300,
|
|
|
|
.ver_es_exclude = 1,
|
|
|
|
},
|
|
|
|
// 16 bit float textures that can be rendered to in GLES
|
|
|
|
{
|
|
|
|
.extension = "GL_EXT_color_buffer_half_float",
|
|
|
|
.provides = MPGL_CAP_EXT_CR_HFLOAT,
|
|
|
|
.ver_exclude = 1,
|
2016-05-23 17:29:08 +00:00
|
|
|
.ver_es_exclude = 320,
|
2016-04-27 17:14:10 +00:00
|
|
|
},
|
2015-10-30 19:26:51 +00:00
|
|
|
{
|
|
|
|
.ver_core = 320,
|
2018-10-16 15:51:47 +00:00
|
|
|
.ver_es_core = 300,
|
2015-10-30 19:26:51 +00:00
|
|
|
.extension = "GL_ARB_sync",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(FenceSync),
|
|
|
|
DEF_FN(ClientWaitSync),
|
|
|
|
DEF_FN(DeleteSync),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2016-06-05 19:55:30 +00:00
|
|
|
{
|
|
|
|
.ver_core = 330,
|
|
|
|
.extension = "GL_ARB_timer_query",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(GenQueries),
|
|
|
|
DEF_FN(DeleteQueries),
|
|
|
|
DEF_FN(BeginQuery),
|
|
|
|
DEF_FN(EndQuery),
|
|
|
|
DEF_FN(QueryCounter),
|
|
|
|
DEF_FN(IsQuery),
|
|
|
|
DEF_FN(GetQueryObjectiv),
|
|
|
|
DEF_FN(GetQueryObjecti64v),
|
|
|
|
DEF_FN(GetQueryObjectuiv),
|
|
|
|
DEF_FN(GetQueryObjectui64v),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2016-06-15 10:32:47 +00:00
|
|
|
{
|
|
|
|
.extension = "GL_EXT_disjoint_timer_query",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN_NAME(GenQueries, "glGenQueriesEXT"),
|
|
|
|
DEF_FN_NAME(DeleteQueries, "glDeleteQueriesEXT"),
|
|
|
|
DEF_FN_NAME(BeginQuery, "glBeginQueryEXT"),
|
|
|
|
DEF_FN_NAME(EndQuery, "glEndQueryEXT"),
|
|
|
|
DEF_FN_NAME(QueryCounter, "glQueryCounterEXT"),
|
|
|
|
DEF_FN_NAME(IsQuery, "glIsQueryEXT"),
|
|
|
|
DEF_FN_NAME(GetQueryObjectiv, "glGetQueryObjectivEXT"),
|
|
|
|
DEF_FN_NAME(GetQueryObjecti64v, "glGetQueryObjecti64vEXT"),
|
|
|
|
DEF_FN_NAME(GetQueryObjectuiv, "glGetQueryObjectuivEXT"),
|
|
|
|
DEF_FN_NAME(GetQueryObjectui64v, "glGetQueryObjectui64vEXT"),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2017-08-16 20:13:51 +00:00
|
|
|
{
|
|
|
|
.ver_core = 430,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(InvalidateTexImage),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2016-05-23 15:31:07 +00:00
|
|
|
{
|
|
|
|
.ver_core = 430,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(InvalidateFramebuffer),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2017-04-08 14:38:56 +00:00
|
|
|
{
|
|
|
|
.ver_core = 410,
|
|
|
|
.ver_es_core = 300,
|
|
|
|
.extension = "GL_ARB_get_program_binary",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(GetProgramBinary),
|
|
|
|
DEF_FN(ProgramBinary),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2017-07-23 07:41:51 +00:00
|
|
|
{
|
|
|
|
.ver_core = 440,
|
|
|
|
.extension = "GL_ARB_buffer_storage",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(BufferStorage),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2017-07-17 16:11:32 +00:00
|
|
|
{
|
|
|
|
.ver_core = 420,
|
|
|
|
.extension = "GL_ARB_shader_image_load_store",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(BindImageTexture),
|
|
|
|
DEF_FN(MemoryBarrier),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2017-08-26 03:48:12 +00:00
|
|
|
{
|
|
|
|
.ver_core = 310,
|
|
|
|
.extension = "GL_ARB_uniform_buffer_object",
|
|
|
|
.provides = MPGL_CAP_UBO,
|
|
|
|
},
|
2017-07-17 19:39:06 +00:00
|
|
|
{
|
|
|
|
.ver_core = 430,
|
|
|
|
.extension = "GL_ARB_shader_storage_buffer_object",
|
|
|
|
.provides = MPGL_CAP_SSBO,
|
|
|
|
},
|
2017-07-17 16:11:32 +00:00
|
|
|
{
|
|
|
|
.ver_core = 430,
|
|
|
|
.extension = "GL_ARB_compute_shader",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(DispatchCompute),
|
|
|
|
{0},
|
|
|
|
},
|
|
|
|
},
|
2017-07-24 19:15:15 +00:00
|
|
|
{
|
|
|
|
.ver_core = 430,
|
|
|
|
.extension = "GL_ARB_arrays_of_arrays",
|
|
|
|
.provides = MPGL_CAP_NESTED_ARRAY,
|
|
|
|
},
|
2012-10-02 23:54:13 +00:00
|
|
|
// Swap control, always an OS specific extension
|
2015-03-25 11:29:19 +00:00
|
|
|
// The OSX code loads this manually.
|
2012-10-02 23:54:13 +00:00
|
|
|
{
|
2015-03-25 11:29:19 +00:00
|
|
|
.extension = "GLX_SGI_swap_control",
|
2014-06-10 21:56:05 +00:00
|
|
|
.functions = (const struct gl_function[]) {
|
2015-03-25 11:40:45 +00:00
|
|
|
DEF_FN_NAME(SwapInterval, "glXSwapIntervalSGI"),
|
2015-03-25 11:29:19 +00:00
|
|
|
{0},
|
|
|
|
},
|
|
|
|
},
|
2017-09-12 04:30:10 +00:00
|
|
|
// This one overrides GLX_SGI_swap_control on platforms using mesa. The
|
|
|
|
// only difference is that it supports glXSwapInterval(0).
|
|
|
|
{
|
|
|
|
.extension = "GLX_MESA_swap_control",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN_NAME(SwapInterval, "glXSwapIntervalMESA"),
|
|
|
|
{0},
|
|
|
|
},
|
|
|
|
},
|
2015-03-25 11:29:19 +00:00
|
|
|
{
|
|
|
|
.extension = "WGL_EXT_swap_control",
|
|
|
|
.functions = (const struct gl_function[]) {
|
2015-03-25 11:40:45 +00:00
|
|
|
DEF_FN_NAME(SwapInterval, "wglSwapIntervalEXT"),
|
2015-03-25 11:29:19 +00:00
|
|
|
{0},
|
2012-10-02 23:54:13 +00:00
|
|
|
},
|
|
|
|
},
|
2014-08-15 21:36:10 +00:00
|
|
|
{
|
|
|
|
.extension = "GLX_SGI_video_sync",
|
2014-10-10 11:44:08 +00:00
|
|
|
.functions = (const struct gl_function[]) {
|
2015-03-25 11:40:45 +00:00
|
|
|
DEF_FN_NAME(GetVideoSync, "glXGetVideoSyncSGI"),
|
|
|
|
DEF_FN_NAME(WaitVideoSync, "glXWaitVideoSyncSGI"),
|
2014-08-15 21:36:10 +00:00
|
|
|
{0},
|
|
|
|
},
|
|
|
|
},
|
2013-11-05 21:06:48 +00:00
|
|
|
// For gl_hwdec_vdpau.c
|
|
|
|
// http://www.opengl.org/registry/specs/NV/vdpau_interop.txt
|
|
|
|
{
|
|
|
|
.extension = "GL_NV_vdpau_interop",
|
|
|
|
.provides = MPGL_CAP_VDPAU,
|
2014-06-10 21:56:05 +00:00
|
|
|
.functions = (const struct gl_function[]) {
|
2013-11-05 21:06:48 +00:00
|
|
|
// (only functions needed by us)
|
|
|
|
DEF_FN(VDPAUInitNV),
|
|
|
|
DEF_FN(VDPAUFiniNV),
|
|
|
|
DEF_FN(VDPAURegisterOutputSurfaceNV),
|
2016-06-19 17:58:40 +00:00
|
|
|
DEF_FN(VDPAURegisterVideoSurfaceNV),
|
2013-11-05 21:06:48 +00:00
|
|
|
DEF_FN(VDPAUUnregisterSurfaceNV),
|
|
|
|
DEF_FN(VDPAUSurfaceAccessNV),
|
|
|
|
DEF_FN(VDPAUMapSurfacesNV),
|
|
|
|
DEF_FN(VDPAUUnmapSurfacesNV),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2015-11-30 13:56:02 +00:00
|
|
|
#if HAVE_GL_DXINTEROP
|
|
|
|
{
|
|
|
|
.extension = "WGL_NV_DX_interop",
|
|
|
|
.provides = MPGL_CAP_DXINTEROP,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN_NAME(DXSetResourceShareHandleNV, "wglDXSetResourceShareHandleNV"),
|
|
|
|
DEF_FN_NAME(DXOpenDeviceNV, "wglDXOpenDeviceNV"),
|
|
|
|
DEF_FN_NAME(DXCloseDeviceNV, "wglDXCloseDeviceNV"),
|
|
|
|
DEF_FN_NAME(DXRegisterObjectNV, "wglDXRegisterObjectNV"),
|
|
|
|
DEF_FN_NAME(DXUnregisterObjectNV, "wglDXUnregisterObjectNV"),
|
|
|
|
DEF_FN_NAME(DXLockObjectsNV, "wglDXLockObjectsNV"),
|
|
|
|
DEF_FN_NAME(DXUnlockObjectsNV, "wglDXUnlockObjectsNV"),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
#endif
|
2013-11-13 20:52:34 +00:00
|
|
|
// Apple Packed YUV Formats
|
|
|
|
// For gl_hwdec_vda.c
|
|
|
|
// http://www.opengl.org/registry/specs/APPLE/rgb_422.txt
|
|
|
|
{
|
|
|
|
.extension = "GL_APPLE_rgb_422",
|
|
|
|
.provides = MPGL_CAP_APPLE_RGB_422,
|
|
|
|
},
|
2014-12-23 01:46:44 +00:00
|
|
|
{
|
|
|
|
.ver_core = 430,
|
|
|
|
.extension = "GL_ARB_debug_output",
|
|
|
|
.provides = MPGL_CAP_DEBUG,
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
// (only functions needed by us)
|
|
|
|
DEF_FN(DebugMessageCallback),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2015-07-03 14:37:01 +00:00
|
|
|
// These don't exist - they are for the sake of mpv internals, and libmpv
|
|
|
|
// interaction (see libmpv/opengl_cb.h).
|
client API: add a new way to pass X11 Display etc. to render API
Hardware decoding things often need access to additional handles from
the windowing system, such as the X11 or Wayland display when using
vaapi. The opengl-cb had nothing dedicated for this, and used the weird
GL_MP_MPGetNativeDisplay GL extension (which was mpv specific and not
officially registered with OpenGL).
This was awkward, and a pain due to having to emulate GL context
behavior (like needing a TLS variable to store context for the pseudo GL
extension function). In addition (and not inherently due to this), we
could pass only one resource from mpv builtin context backends to
hwdecs. It was also all GL specific.
Replace this with a newer mechanism. It works for all RA backends, not
just GL. the API user can explicitly pass the objects at init time via
mpv_render_context_create(). Multiple resources are naturally possible.
The API uses MPV_RENDER_PARAM_* defines, but internally we use strings.
This is done for 2 reasons: 1. trying to leave libmpv and internal
mechanisms decoupled, 2. not having to add public API for some of the
internal resource types (especially D3D/GL interop stuff).
To remain sane, drop support for obscure half-working opengl-cb things,
like the DRM interop (was missing necessary things), the RPI window
thing (nobody used it), and obscure D3D interop things (not needed with
ANGLE, others were undocumented). In order not to break ABI and the C
API, we don't remove the associated structs from opengl_cb.h.
The parts which are still needed (in particular DRM interop) needs to be
ported to the render API.
2018-03-22 16:05:01 +00:00
|
|
|
// This is not used by the render API, only the deprecated opengl-cb API.
|
2015-09-24 19:07:16 +00:00
|
|
|
{
|
|
|
|
.extension = "GL_MP_MPGetNativeDisplay",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(MPGetNativeDisplay),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
// Same, but using the old name.
|
2015-07-03 14:37:01 +00:00
|
|
|
{
|
|
|
|
.extension = "GL_MP_D3D_interfaces",
|
|
|
|
.functions = (const struct gl_function[]) {
|
2015-09-24 19:07:16 +00:00
|
|
|
DEF_FN_NAME(MPGetNativeDisplay, "glMPGetD3DInterface"),
|
2015-07-03 14:37:01 +00:00
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2016-05-12 09:17:49 +00:00
|
|
|
{
|
|
|
|
.extension = "GL_ANGLE_translated_shader_source",
|
|
|
|
.functions = (const struct gl_function[]) {
|
|
|
|
DEF_FN(GetTranslatedShaderSourceANGLE),
|
|
|
|
{0}
|
|
|
|
},
|
|
|
|
},
|
2012-10-02 23:54:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#undef FN_OFFS
|
|
|
|
#undef DEF_FN_HARD
|
|
|
|
#undef DEF_FN
|
2015-03-25 11:40:45 +00:00
|
|
|
#undef DEF_FN_NAME
|
2012-10-02 23:54:13 +00:00
|
|
|
|
egl_helpers: fix create_context fallback behavior
The EGL stuff is really complicated because of historical reasons
(tl;dr: blame EGL). There was one edge case with EGL context creation
that lead to incorrect behavior. EGL_KHR_create_context was created with
EGL 1.4 (which mpv does support) but it is still possible for an EGL 1.4
device to not implement this extension. That means that none of the EGL
attrs that pass a specific opengl version work. So for this obscure
case, there is a fallback context creation at the very end which simply
creates an EGLContext without passing any special attrs.
This has another problem however. mpv has a hard requirement on at least
desktop opengl 2.1 or opengl ES 2.0 to function (we're not asking for
much here). Since the fallback EGL context creation has no version
checking, it is entirely possible to create an EGL context with a
desktop opengl version under 2.1. As you get further along in the code,
the user will encounter the hard opengl version check and then error
out. However, we're supposed to also actually check if GLES works
(that's what the opengl-es=auto option is for) so this is a bug.
The fix is to do a bit of code duplication and make a mpgl_check_version
specifically for if we hit the edge case of needing to create an EGL
context without the EGL_KHR_create_context extension. Grab the version
with the function pointer, check if it's under 210, if so destroy the
EGL context and set it to NULL. After that, if the user has set
opengl-es to auto, mpv will try GLES next. If it is set to no, then mpv
will simply fail as desired. Fixes #5915.
Sidenote: the ra_gl_ctx_test_version originally testing 140 doesn't make
any sense. Passing the version number in that function only does
something if the user has set opengl-restrict. What we need to do is to
pass the version of the created context to that function. If the version
is higher than the opengl-restrict option, then make this a failure.
2021-07-24 01:08:05 +00:00
|
|
|
void mpgl_check_version(GL *gl, void *(*get_fn)(void *ctx, const char *n),
|
|
|
|
void *fn_ctx)
|
|
|
|
{
|
|
|
|
gl->GetString = get_fn(fn_ctx, "glGetString");
|
|
|
|
if (!gl->GetString) {
|
|
|
|
gl->version = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const char *version_string = gl->GetString(GL_VERSION);
|
|
|
|
if (!version_string) {
|
|
|
|
gl->version = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int major = 0, minor = 0;
|
|
|
|
if (sscanf(version_string, "%d.%d", &major, &minor) < 2) {
|
|
|
|
gl->version = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gl->version = MPGL_VER(major, minor);
|
|
|
|
}
|
2012-10-02 23:54:13 +00:00
|
|
|
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
// Fill the GL struct with function pointers and extensions from the current
|
2013-03-01 14:55:08 +00:00
|
|
|
// GL context. Called by the backend.
|
2016-04-16 14:16:50 +00:00
|
|
|
// get_fn: function to resolve function names
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
// ext2: an extra extension string
|
2013-09-11 22:57:32 +00:00
|
|
|
// log: used to output messages
|
2014-12-09 16:47:02 +00:00
|
|
|
void mpgl_load_functions2(GL *gl, void *(*get_fn)(void *ctx, const char *n),
|
|
|
|
void *fn_ctx, const char *ext2, struct mp_log *log)
|
2011-09-25 09:18:07 +00:00
|
|
|
{
|
2019-11-29 19:23:27 +00:00
|
|
|
talloc_free(gl->extensions);
|
2012-10-02 23:54:13 +00:00
|
|
|
*gl = (GL) {
|
|
|
|
.extensions = talloc_strdup(gl, ext2 ? ext2 : ""),
|
|
|
|
};
|
2011-09-25 09:18:07 +00:00
|
|
|
|
2014-12-09 16:47:02 +00:00
|
|
|
gl->GetString = get_fn(fn_ctx, "glGetString");
|
2014-05-26 20:56:13 +00:00
|
|
|
if (!gl->GetString) {
|
|
|
|
mp_err(log, "Can't load OpenGL functions.\n");
|
2015-01-21 19:32:42 +00:00
|
|
|
goto error;
|
2014-05-26 20:56:13 +00:00
|
|
|
}
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
|
2014-05-26 21:05:22 +00:00
|
|
|
int major = 0, minor = 0;
|
2014-12-18 22:51:21 +00:00
|
|
|
const char *version_string = gl->GetString(GL_VERSION);
|
2016-04-08 08:57:21 +00:00
|
|
|
if (!version_string) {
|
|
|
|
mp_fatal(log, "glGetString(GL_VERSION) returned NULL.\n");
|
2015-02-20 17:53:35 +00:00
|
|
|
goto error;
|
2016-04-08 08:57:21 +00:00
|
|
|
}
|
2015-02-20 17:53:35 +00:00
|
|
|
mp_verbose(log, "GL_VERSION='%s'\n", version_string);
|
2014-12-18 22:51:21 +00:00
|
|
|
if (strncmp(version_string, "OpenGL ES ", 10) == 0) {
|
|
|
|
version_string += 10;
|
2014-12-18 20:06:17 +00:00
|
|
|
gl->es = 100;
|
2014-12-17 20:48:23 +00:00
|
|
|
}
|
2015-02-20 17:53:35 +00:00
|
|
|
if (sscanf(version_string, "%d.%d", &major, &minor) < 2)
|
|
|
|
goto error;
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
gl->version = MPGL_VER(major, minor);
|
2014-12-18 20:06:17 +00:00
|
|
|
mp_verbose(log, "Detected %s %d.%d.\n", gl->es ? "GLES" : "desktop OpenGL",
|
|
|
|
major, minor);
|
2014-12-17 20:48:23 +00:00
|
|
|
|
2014-12-18 20:06:17 +00:00
|
|
|
if (gl->es) {
|
|
|
|
gl->es = gl->version;
|
2014-12-19 00:03:08 +00:00
|
|
|
gl->version = 0;
|
|
|
|
if (gl->es < 200) {
|
|
|
|
mp_fatal(log, "At least GLESv2 required.\n");
|
2015-01-21 19:32:42 +00:00
|
|
|
goto error;
|
2014-12-18 20:06:17 +00:00
|
|
|
}
|
2014-12-17 20:48:23 +00:00
|
|
|
}
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
|
2014-05-26 21:08:07 +00:00
|
|
|
mp_verbose(log, "GL_VENDOR='%s'\n", gl->GetString(GL_VENDOR));
|
|
|
|
mp_verbose(log, "GL_RENDERER='%s'\n", gl->GetString(GL_RENDERER));
|
2014-08-27 23:32:22 +00:00
|
|
|
const char *shader = gl->GetString(GL_SHADING_LANGUAGE_VERSION);
|
|
|
|
if (shader)
|
|
|
|
mp_verbose(log, "GL_SHADING_LANGUAGE_VERSION='%s'\n", shader);
|
2013-03-01 20:19:20 +00:00
|
|
|
|
2014-12-19 00:03:08 +00:00
|
|
|
if (gl->version >= 300) {
|
2014-12-09 16:47:02 +00:00
|
|
|
gl->GetStringi = get_fn(fn_ctx, "glGetStringi");
|
|
|
|
gl->GetIntegerv = get_fn(fn_ctx, "glGetIntegerv");
|
2012-03-30 22:33:53 +00:00
|
|
|
|
|
|
|
if (!(gl->GetStringi && gl->GetIntegerv))
|
2015-01-21 19:32:42 +00:00
|
|
|
goto error;
|
2012-03-30 22:33:53 +00:00
|
|
|
|
|
|
|
GLint exts;
|
|
|
|
gl->GetIntegerv(GL_NUM_EXTENSIONS, &exts);
|
|
|
|
for (int n = 0; n < exts; n++) {
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
const char *ext = gl->GetStringi(GL_EXTENSIONS, n);
|
|
|
|
gl->extensions = talloc_asprintf_append(gl->extensions, " %s", ext);
|
2012-03-30 22:33:53 +00:00
|
|
|
}
|
2012-10-02 23:54:13 +00:00
|
|
|
|
gl_common: simplify window/context creation
Allow the backend code to create a GL context on best effort basis,
instead of having to implement separate functions for each variation.
This means there's only a single create_window callback now. Also,
getFunctions() doesn't have the gl3 parameter anymore, which was
confusing and hard to explain.
create_window() tries to create a GL context of any version. The field
MPGLContext.requested_gl_version is taken as a hint whether a GL3 or a
legacy context is preferred. (This should be easy on all platforms.)
The cocoa part always assumes that GL 3 is always available on
OSX 10.7.0 and higher, and miserably fails if it's not. One could try
to put more effort into probing the context, but apparently this
situation never happens, so don't bother. (And even if, mpv should be
able to fall back to vo_corevideo.)
The X11 part doesn't change much, but moving these functions around
makes the diff bigger.
Note about some corner cases:
This doesn't handle CONTEXT_FORWARD_COMPATIBLE_BIT_ARB on OpenGL 3.0
correctly. This was the one thing getFunctions() actually needed the
gl3 parameter, and we just make sure we never use forward compatible
contexts on 3.0. It should work with any version above (e.g. 3.1, 3.2
and 3.3 should be fine). This is because the GL_ARB_compatibility
extension is specified for 3.1 and up only. There doesn't seem to be
any way to detect presence of legacy GL on 3.0 with a forward
compatible context. As a counter measure, remove the FORWARD_COMPATIBLE
flags from the win32 code. Maybe this will go wrong. (Should this
happen, the flag has the be added back, and the win32 will have to
explicitly check for GL 3.0 and add "GL_ARB_compatibility" to the extra
extension string.)
Note about GLX:
Probing GL versions by trying to create a context on an existing window
was (probably) not always possible. Old code used GLX 1.2 to create
legacy contexts, and it required code different from GLX 1.3 even before
creation of the X window (the problem was selections of the X Visual).
That's why there were two functions for window creation (create_window_old
and create_window_gl3). However, the legacy context creation code was
updated to GLX 1.3 in commit b3b20cc, so having different functions for
window creation is not needed anymore.
2013-02-24 22:31:57 +00:00
|
|
|
} else {
|
2012-03-30 22:33:53 +00:00
|
|
|
const char *ext = (char*)gl->GetString(GL_EXTENSIONS);
|
2012-10-02 23:54:13 +00:00
|
|
|
gl->extensions = talloc_asprintf_append(gl->extensions, " %s", ext);
|
2012-03-30 22:33:53 +00:00
|
|
|
}
|
2012-10-02 23:54:13 +00:00
|
|
|
|
2014-05-26 21:08:07 +00:00
|
|
|
mp_dbg(log, "Combined OpenGL extensions string:\n%s\n", gl->extensions);
|
2012-10-02 23:54:13 +00:00
|
|
|
|
2015-08-01 19:11:38 +00:00
|
|
|
for (int n = 0; n < MP_ARRAY_SIZE(gl_functions); n++) {
|
2014-06-10 21:56:05 +00:00
|
|
|
const struct gl_functions *section = &gl_functions[n];
|
2014-12-18 22:51:21 +00:00
|
|
|
int version = gl->es ? gl->es : gl->version;
|
|
|
|
int ver_core = gl->es ? section->ver_es_core : section->ver_core;
|
2012-10-02 23:54:13 +00:00
|
|
|
|
2012-12-27 16:08:17 +00:00
|
|
|
// NOTE: Function entrypoints can exist, even if they do not work.
|
|
|
|
// We must always check extension strings and versions.
|
2012-03-30 22:33:53 +00:00
|
|
|
|
2016-05-12 18:08:49 +00:00
|
|
|
if (gl->version && section->ver_exclude &&
|
|
|
|
gl->version >= section->ver_exclude)
|
|
|
|
continue;
|
|
|
|
if (gl->es && section->ver_es_exclude &&
|
|
|
|
gl->es >= section->ver_es_exclude)
|
|
|
|
continue;
|
|
|
|
|
2014-12-19 17:54:21 +00:00
|
|
|
bool exists = false, must_exist = false;
|
2014-12-18 22:51:21 +00:00
|
|
|
if (ver_core)
|
2014-12-19 17:54:21 +00:00
|
|
|
must_exist = version >= ver_core;
|
2012-12-27 16:08:17 +00:00
|
|
|
|
2017-10-02 15:30:27 +00:00
|
|
|
if (section->extension)
|
|
|
|
exists = gl_check_extension(gl->extensions, section->extension);
|
2012-12-27 16:08:17 +00:00
|
|
|
|
2014-12-19 17:54:21 +00:00
|
|
|
exists |= must_exist;
|
2012-12-27 16:08:17 +00:00
|
|
|
if (!exists)
|
2012-10-02 23:54:13 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
void *loaded[MAX_FN_COUNT] = {0};
|
|
|
|
bool all_loaded = true;
|
2014-12-18 14:29:47 +00:00
|
|
|
const struct gl_function *fnlist = section->functions;
|
2012-10-02 23:54:13 +00:00
|
|
|
|
2015-03-25 11:40:45 +00:00
|
|
|
for (int i = 0; fnlist && fnlist[i].name; i++) {
|
2014-12-18 14:29:47 +00:00
|
|
|
const struct gl_function *fn = &fnlist[i];
|
2015-03-25 11:40:45 +00:00
|
|
|
void *ptr = get_fn(fn_ctx, fn->name);
|
2012-10-02 23:54:13 +00:00
|
|
|
if (!ptr) {
|
|
|
|
all_loaded = false;
|
2016-06-22 19:55:00 +00:00
|
|
|
if (must_exist) {
|
|
|
|
mp_err(log, "GL %d.%d function %s not found.\n",
|
|
|
|
MPGL_VER_GET_MAJOR(ver_core),
|
|
|
|
MPGL_VER_GET_MINOR(ver_core), fn->name);
|
2015-01-21 19:32:42 +00:00
|
|
|
goto error;
|
2016-06-22 19:55:00 +00:00
|
|
|
} else {
|
|
|
|
mp_warn(log, "Function %s from extension %s not found.\n",
|
|
|
|
fn->name, section->extension);
|
|
|
|
}
|
2015-01-21 19:32:42 +00:00
|
|
|
break;
|
2012-10-02 23:54:13 +00:00
|
|
|
}
|
|
|
|
assert(i < MAX_FN_COUNT);
|
|
|
|
loaded[i] = ptr;
|
|
|
|
}
|
|
|
|
|
2015-01-21 19:32:42 +00:00
|
|
|
if (all_loaded) {
|
2012-10-02 23:54:13 +00:00
|
|
|
gl->mpgl_caps |= section->provides;
|
2015-03-25 11:40:45 +00:00
|
|
|
for (int i = 0; fnlist && fnlist[i].name; i++) {
|
2014-12-18 14:29:47 +00:00
|
|
|
const struct gl_function *fn = &fnlist[i];
|
2012-10-02 23:54:13 +00:00
|
|
|
void **funcptr = (void**)(((char*)gl) + fn->offset);
|
|
|
|
if (loaded[i])
|
|
|
|
*funcptr = loaded[i];
|
|
|
|
}
|
2016-05-23 15:06:38 +00:00
|
|
|
if (!must_exist && section->extension)
|
|
|
|
mp_verbose(log, "Loaded extension %s.\n", section->extension);
|
2011-09-25 09:18:07 +00:00
|
|
|
}
|
2012-10-02 23:54:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gl->glsl_version = 0;
|
2014-12-17 20:48:23 +00:00
|
|
|
if (gl->es) {
|
2014-12-19 00:03:08 +00:00
|
|
|
if (gl->es >= 200)
|
|
|
|
gl->glsl_version = 100;
|
2014-12-18 22:51:21 +00:00
|
|
|
if (gl->es >= 300)
|
2014-12-17 20:48:23 +00:00
|
|
|
gl->glsl_version = 300;
|
|
|
|
} else {
|
2017-11-03 10:53:00 +00:00
|
|
|
gl->glsl_version = 120;
|
2015-11-09 13:28:56 +00:00
|
|
|
int glsl_major = 0, glsl_minor = 0;
|
2015-11-19 16:12:51 +00:00
|
|
|
if (shader && sscanf(shader, "%d.%d", &glsl_major, &glsl_minor) == 2)
|
2015-11-09 13:28:56 +00:00
|
|
|
gl->glsl_version = glsl_major * 100 + glsl_minor;
|
2017-07-04 22:25:32 +00:00
|
|
|
// restrict GLSL version to be forwards compatible
|
2017-08-26 03:58:50 +00:00
|
|
|
gl->glsl_version = MPMIN(gl->glsl_version, 440);
|
2014-12-17 20:48:23 +00:00
|
|
|
}
|
2012-10-02 23:54:13 +00:00
|
|
|
|
2015-07-11 16:40:37 +00:00
|
|
|
if (is_software_gl(gl)) {
|
2014-12-19 17:37:16 +00:00
|
|
|
gl->mpgl_caps |= MPGL_CAP_SW;
|
2015-07-11 16:40:37 +00:00
|
|
|
mp_verbose(log, "Detected suspected software renderer.\n");
|
2014-12-19 17:37:16 +00:00
|
|
|
}
|
2014-12-09 21:28:16 +00:00
|
|
|
|
2017-07-24 19:15:15 +00:00
|
|
|
// GL_ARB_compute_shader & GL_ARB_shader_image_load_store
|
2017-08-07 17:14:18 +00:00
|
|
|
if (gl->DispatchCompute && gl->BindImageTexture)
|
2017-07-24 19:15:15 +00:00
|
|
|
gl->mpgl_caps |= MPGL_CAP_COMPUTE_SHADER;
|
|
|
|
|
2014-12-09 21:28:16 +00:00
|
|
|
// Provided for simpler handling if no framebuffer support is available.
|
|
|
|
if (!gl->BindFramebuffer)
|
|
|
|
gl->BindFramebuffer = &dummy_glBindFramebuffer;
|
2015-01-21 19:32:42 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
error:
|
|
|
|
gl->version = 0;
|
|
|
|
gl->es = 0;
|
|
|
|
gl->mpgl_caps = 0;
|
2005-07-26 10:16:18 +00:00
|
|
|
}
|
|
|
|
|
2014-12-09 16:47:02 +00:00
|
|
|
static void *get_procaddr_wrapper(void *ctx, const char *name)
|
|
|
|
{
|
|
|
|
void *(*getProcAddress)(const GLubyte *) = ctx;
|
|
|
|
return getProcAddress ? getProcAddress((const GLubyte*)name) : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mpgl_load_functions(GL *gl, void *(*getProcAddress)(const GLubyte *),
|
|
|
|
const char *ext2, struct mp_log *log)
|
|
|
|
{
|
|
|
|
mpgl_load_functions2(gl, get_procaddr_wrapper, getProcAddress, ext2, log);
|
|
|
|
}
|