2023-01-27 13:21:09 +13:00
|
|
|
// Copyright 2023 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2009-02-23 07:23:34 +00:00
|
|
|
|
2023-01-27 13:21:09 +13:00
|
|
|
#include "VideoBackends/OGL/OGLConfig.h"
|
2017-01-23 11:20:20 -05:00
|
|
|
|
2023-03-19 22:20:34 -07:00
|
|
|
#include <cstdio>
|
|
|
|
#include <string>
|
|
|
|
#include <string_view>
|
|
|
|
|
|
|
|
#include "Common/Assert.h"
|
2018-10-03 23:02:45 +10:00
|
|
|
#include "Common/GL/GLContext.h"
|
2023-01-27 13:21:09 +13:00
|
|
|
#include "Common/GL/GLExtensions/GLExtensions.h"
|
2014-12-28 21:46:00 +01:00
|
|
|
#include "Common/Logging/LogManager.h"
|
2017-01-31 23:29:29 -05:00
|
|
|
#include "Common/MsgHandler.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2017-05-18 13:59:38 +01:00
|
|
|
#include "Core/Config/GraphicsSettings.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2023-03-19 22:20:34 -07:00
|
|
|
#include "VideoBackends/OGL/OGLTexture.h"
|
|
|
|
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/DriverDetails.h"
|
2023-03-19 22:20:34 -07:00
|
|
|
#include "VideoCommon/FramebufferManager.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/OnScreenDisplay.h"
|
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2009-02-23 07:23:34 +00:00
|
|
|
|
2023-01-27 13:21:09 +13:00
|
|
|
namespace OGL
|
2013-06-25 18:14:41 +02:00
|
|
|
{
|
2023-01-27 13:21:09 +13:00
|
|
|
void InitDriverInfo()
|
2013-06-11 08:33:56 -05:00
|
|
|
{
|
2019-09-24 00:30:20 -04:00
|
|
|
const std::string_view svendor(g_ogl_config.gl_vendor);
|
|
|
|
const std::string_view srenderer(g_ogl_config.gl_renderer);
|
|
|
|
const std::string_view sversion(g_ogl_config.gl_version);
|
2013-06-11 08:33:56 -05:00
|
|
|
DriverDetails::Vendor vendor = DriverDetails::VENDOR_UNKNOWN;
|
2013-08-21 05:34:42 -05:00
|
|
|
DriverDetails::Driver driver = DriverDetails::DRIVER_UNKNOWN;
|
2016-01-15 12:13:14 -06:00
|
|
|
DriverDetails::Family family = DriverDetails::Family::UNKNOWN;
|
2013-06-11 08:33:56 -05:00
|
|
|
double version = 0.0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-08-21 05:34:42 -05:00
|
|
|
// Get the vendor first
|
2019-09-24 00:30:20 -04:00
|
|
|
if (svendor == "NVIDIA Corporation")
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2019-09-24 00:30:20 -04:00
|
|
|
if (srenderer != "NVIDIA Tegra")
|
|
|
|
{
|
|
|
|
vendor = DriverDetails::VENDOR_NVIDIA;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
vendor = DriverDetails::VENDOR_TEGRA;
|
|
|
|
}
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2013-08-21 05:34:42 -05:00
|
|
|
else if (svendor == "ATI Technologies Inc." || svendor == "Advanced Micro Devices, Inc.")
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-06-11 08:33:56 -05:00
|
|
|
vendor = DriverDetails::VENDOR_ATI;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2019-09-24 00:30:20 -04:00
|
|
|
else if (sversion.find("Mesa") != std::string::npos)
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-08-23 10:52:29 +02:00
|
|
|
vendor = DriverDetails::VENDOR_MESA;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2019-09-24 00:30:20 -04:00
|
|
|
else if (svendor.find("Intel") != std::string::npos)
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-06-11 08:33:56 -05:00
|
|
|
vendor = DriverDetails::VENDOR_INTEL;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2013-06-11 08:33:56 -05:00
|
|
|
else if (svendor == "ARM")
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-06-11 08:33:56 -05:00
|
|
|
vendor = DriverDetails::VENDOR_ARM;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2013-08-21 05:41:32 -05:00
|
|
|
else if (svendor == "http://limadriver.org/")
|
|
|
|
{
|
|
|
|
vendor = DriverDetails::VENDOR_ARM;
|
|
|
|
driver = DriverDetails::DRIVER_LIMA;
|
|
|
|
}
|
2013-06-11 08:33:56 -05:00
|
|
|
else if (svendor == "Qualcomm")
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-06-11 08:33:56 -05:00
|
|
|
vendor = DriverDetails::VENDOR_QUALCOMM;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2013-06-11 08:33:56 -05:00
|
|
|
else if (svendor == "Imagination Technologies")
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-06-11 08:33:56 -05:00
|
|
|
vendor = DriverDetails::VENDOR_IMGTEC;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2013-06-11 08:33:56 -05:00
|
|
|
else if (svendor == "Vivante Corporation")
|
2014-08-15 14:09:53 -04:00
|
|
|
{
|
2013-06-11 08:33:56 -05:00
|
|
|
vendor = DriverDetails::VENDOR_VIVANTE;
|
2014-08-15 14:09:53 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-06-11 08:33:56 -05:00
|
|
|
// Get device family and driver version...if we care about it
|
2014-03-11 00:30:55 +13:00
|
|
|
switch (vendor)
|
2013-06-11 08:33:56 -05:00
|
|
|
{
|
|
|
|
case DriverDetails::VENDOR_QUALCOMM:
|
|
|
|
{
|
2014-12-19 21:31:37 +00:00
|
|
|
driver = DriverDetails::DRIVER_QUALCOMM;
|
2013-06-11 08:33:56 -05:00
|
|
|
double glVersion;
|
|
|
|
sscanf(g_ogl_config.gl_version, "OpenGL ES %lg V@%lg", &glVersion, &version);
|
|
|
|
}
|
|
|
|
break;
|
2013-07-25 05:44:01 +00:00
|
|
|
case DriverDetails::VENDOR_ARM:
|
2014-04-11 23:46:44 -05:00
|
|
|
// Currently the Mali-T line has two families in it.
|
|
|
|
// Mali-T6xx and Mali-T7xx
|
|
|
|
// These two families are similar enough that they share bugs in their drivers.
|
2014-12-19 21:41:12 +00:00
|
|
|
//
|
|
|
|
// Mali drivers provide no way to explicitly find out what video driver is running.
|
|
|
|
// This is similar to how we can't find the Nvidia driver version in Windows.
|
|
|
|
// Good thing is that ARM introduces a new video driver about once every two years so we can
|
|
|
|
// find the driver version by the features it exposes.
|
|
|
|
// r2p0 - No OpenGL ES 3.0 support (We don't support this)
|
|
|
|
// r3p0 - OpenGL ES 3.0 support
|
|
|
|
// r4p0 - Supports 'GL_EXT_shader_pixel_local_storage' extension.
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-12-19 21:41:12 +00:00
|
|
|
driver = DriverDetails::DRIVER_ARM;
|
|
|
|
if (GLExtensions::Supports("GL_EXT_shader_pixel_local_storage"))
|
|
|
|
version = 400;
|
|
|
|
else
|
|
|
|
version = 300;
|
2013-08-23 10:52:29 +02:00
|
|
|
break;
|
|
|
|
case DriverDetails::VENDOR_MESA:
|
|
|
|
{
|
2014-03-11 00:30:55 +13:00
|
|
|
if (svendor == "nouveau")
|
2016-01-15 12:13:14 -06:00
|
|
|
{
|
2013-08-23 10:52:29 +02:00
|
|
|
driver = DriverDetails::DRIVER_NOUVEAU;
|
2016-01-15 12:13:14 -06:00
|
|
|
}
|
2014-03-11 00:30:55 +13:00
|
|
|
else if (svendor == "Intel Open Source Technology Center")
|
2016-01-15 12:13:14 -06:00
|
|
|
{
|
2013-08-23 10:52:29 +02:00
|
|
|
driver = DriverDetails::DRIVER_I965;
|
2016-01-15 12:13:14 -06:00
|
|
|
if (srenderer.find("Sandybridge") != std::string::npos)
|
|
|
|
family = DriverDetails::Family::INTEL_SANDY;
|
|
|
|
else if (srenderer.find("Ivybridge") != std::string::npos)
|
|
|
|
family = DriverDetails::Family::INTEL_IVY;
|
|
|
|
}
|
2019-09-24 00:30:20 -04:00
|
|
|
else if (srenderer.find("AMD") != std::string::npos ||
|
|
|
|
srenderer.find("ATI") != std::string::npos)
|
2016-01-15 12:13:14 -06:00
|
|
|
{
|
2013-08-23 10:52:29 +02:00
|
|
|
driver = DriverDetails::DRIVER_R600;
|
2016-01-15 12:13:14 -06:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-08-23 10:52:29 +02:00
|
|
|
int major = 0;
|
|
|
|
int minor = 0;
|
|
|
|
int release = 0;
|
2015-07-25 12:20:00 +02:00
|
|
|
sscanf(g_ogl_config.gl_version, "%*s (Core Profile) Mesa %d.%d.%d", &major, &minor, &release);
|
2013-08-23 10:52:29 +02:00
|
|
|
version = 100 * major + 10 * minor + release;
|
|
|
|
}
|
2013-07-25 05:44:01 +00:00
|
|
|
break;
|
2014-09-04 18:03:53 -05:00
|
|
|
case DriverDetails::VENDOR_INTEL: // Happens in OS X/Windows
|
|
|
|
{
|
2016-01-15 12:13:14 -06:00
|
|
|
u32 market_name;
|
|
|
|
sscanf(g_ogl_config.gl_renderer, "Intel HD Graphics %d", &market_name);
|
|
|
|
switch (market_name)
|
|
|
|
{
|
|
|
|
case 2000:
|
|
|
|
case 3000:
|
|
|
|
family = DriverDetails::Family::INTEL_SANDY;
|
|
|
|
break;
|
|
|
|
case 2500:
|
|
|
|
case 4000:
|
|
|
|
family = DriverDetails::Family::INTEL_IVY;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
family = DriverDetails::Family::UNKNOWN;
|
|
|
|
break;
|
|
|
|
};
|
2014-09-04 18:03:53 -05:00
|
|
|
#ifdef _WIN32
|
2013-12-30 18:26:55 -06:00
|
|
|
int glmajor = 0;
|
|
|
|
int glminor = 0;
|
|
|
|
int major = 0;
|
|
|
|
int minor = 0;
|
|
|
|
int release = 0;
|
2014-09-04 18:03:53 -05:00
|
|
|
int revision = 0;
|
|
|
|
// Example version string: '4.3.0 - Build 10.18.10.3907'
|
|
|
|
sscanf(g_ogl_config.gl_version, "%d.%d.0 - Build %d.%d.%d.%d", &glmajor, &glminor, &major,
|
|
|
|
&minor, &release, &revision);
|
|
|
|
version = 100000000 * major + 1000000 * minor + 10000 * release + revision;
|
|
|
|
version /= 10000;
|
|
|
|
#endif
|
|
|
|
}
|
2013-12-30 18:26:55 -06:00
|
|
|
break;
|
2014-01-07 18:29:04 -06:00
|
|
|
case DriverDetails::VENDOR_NVIDIA:
|
|
|
|
{
|
|
|
|
int glmajor = 0;
|
|
|
|
int glminor = 0;
|
|
|
|
int glrelease = 0;
|
|
|
|
int major = 0;
|
|
|
|
int minor = 0;
|
2015-01-11 00:17:29 -05:00
|
|
|
// TODO: this is known to be broken on Windows
|
|
|
|
// Nvidia seems to have removed their driver version from this string, so we can't get it.
|
|
|
|
// hopefully we'll never have to workaround Nvidia bugs
|
2014-01-07 19:35:06 -06:00
|
|
|
sscanf(g_ogl_config.gl_version, "%d.%d.%d NVIDIA %d.%d", &glmajor, &glminor, &glrelease, &major,
|
|
|
|
&minor);
|
2014-01-07 18:29:04 -06:00
|
|
|
version = 100 * major + minor;
|
|
|
|
}
|
|
|
|
break;
|
2017-09-02 11:25:12 -07:00
|
|
|
case DriverDetails::VENDOR_IMGTEC:
|
|
|
|
{
|
|
|
|
// Example version string:
|
|
|
|
// "OpenGL ES 3.2 build 1.9@4850625"
|
|
|
|
// Ends up as "109.4850625" - "1.9" being the branch, "4850625" being the build's change ID
|
|
|
|
// The change ID only makes sense to compare within a branch
|
|
|
|
driver = DriverDetails::DRIVER_IMGTEC;
|
|
|
|
double gl_version;
|
|
|
|
int major, minor, change;
|
|
|
|
constexpr double change_scale = 10000000;
|
|
|
|
sscanf(g_ogl_config.gl_version, "OpenGL ES %lg build %d.%d@%d", &gl_version, &major, &minor,
|
|
|
|
&change);
|
|
|
|
version = 100 * major + minor;
|
|
|
|
if (change >= change_scale)
|
|
|
|
{
|
2020-11-09 03:08:00 -05:00
|
|
|
ERROR_LOG_FMT(VIDEO, "Version changeID overflow - change:{} scale:{}", change, change_scale);
|
2017-09-02 11:25:12 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
version += static_cast<double>(change) / change_scale;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2013-06-11 08:33:56 -05:00
|
|
|
// We don't care about these
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2016-08-13 15:16:31 +10:00
|
|
|
DriverDetails::Init(DriverDetails::API_OPENGL, vendor, driver, version, family);
|
2013-06-11 08:33:56 -05:00
|
|
|
}
|
|
|
|
|
2023-01-27 13:21:09 +13:00
|
|
|
bool PopulateConfig(GLContext* m_main_gl_context)
|
2011-05-07 22:00:05 +00:00
|
|
|
{
|
|
|
|
bool bSuccess = true;
|
2019-04-15 23:38:10 +10:00
|
|
|
bool supports_glsl_cache = false;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-01-01 20:06:11 -06:00
|
|
|
g_ogl_config.gl_vendor = (const char*)glGetString(GL_VENDOR);
|
|
|
|
g_ogl_config.gl_renderer = (const char*)glGetString(GL_RENDERER);
|
|
|
|
g_ogl_config.gl_version = (const char*)glGetString(GL_VERSION);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-10-03 23:03:26 +10:00
|
|
|
if (!m_main_gl_context->IsGLES())
|
2010-08-10 01:08:22 +00:00
|
|
|
{
|
2015-12-13 11:39:45 -06:00
|
|
|
if (!GLExtensions::Supports("GL_ARB_framebuffer_object"))
|
|
|
|
{
|
|
|
|
// We want the ogl3 framebuffer instead of the ogl2 one for better blitting support.
|
|
|
|
// It's also compatible with the gles3 one.
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: ERROR: Need GL_ARB_framebuffer_object for multiple render targets.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3.0?");
|
2015-12-13 11:39:45 -06:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-12-13 11:39:45 -06:00
|
|
|
if (!GLExtensions::Supports("GL_ARB_vertex_array_object"))
|
|
|
|
{
|
|
|
|
// This extension is used to replace lots of pointer setting function.
|
|
|
|
// Also gles3 requires to use it.
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Need GL_ARB_vertex_array_object.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3.0?");
|
2015-12-13 11:39:45 -06:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-12-13 11:39:45 -06:00
|
|
|
if (!GLExtensions::Supports("GL_ARB_map_buffer_range"))
|
|
|
|
{
|
|
|
|
// ogl3 buffer mapping for better streaming support.
|
|
|
|
// The ogl2 one also isn't in gles3.
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Need GL_ARB_map_buffer_range.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3.0?");
|
2015-12-13 11:39:45 -06:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-12-13 11:39:45 -06:00
|
|
|
if (!GLExtensions::Supports("GL_ARB_uniform_buffer_object"))
|
|
|
|
{
|
|
|
|
// ubo allow us to keep the current constants on shader switches
|
|
|
|
// we also can stream them much nicer and pack into it whatever we want to
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Need GL_ARB_uniform_buffer_object.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3.1?");
|
2015-12-13 11:39:45 -06:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-10-30 00:56:18 +02:00
|
|
|
else if (DriverDetails::HasBug(DriverDetails::BUG_BROKEN_UBO))
|
2015-12-13 11:39:45 -06:00
|
|
|
{
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT(
|
2015-12-13 11:39:45 -06:00
|
|
|
"Buggy GPU driver detected.\n"
|
|
|
|
"Please either install the closed-source GPU driver or update your Mesa 3D version.");
|
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-12-13 11:39:45 -06:00
|
|
|
if (!GLExtensions::Supports("GL_ARB_sampler_objects"))
|
|
|
|
{
|
|
|
|
// Our sampler cache uses this extension. It could easyly be workaround and it's by far the
|
|
|
|
// highest requirement, but it seems that no driver lacks support for it.
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Need GL_ARB_sampler_objects.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3.3?");
|
2015-12-13 11:39:45 -06:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2013-06-25 18:14:41 +02:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-06-20 23:54:44 +12:00
|
|
|
// Copy the GPU name to g_Config, so Analytics can see it.
|
|
|
|
g_Config.backend_info.AdapterName = g_ogl_config.gl_renderer;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-09-04 20:25:59 -05:00
|
|
|
g_Config.backend_info.bSupportsDualSourceBlend =
|
2016-10-27 21:59:12 +02:00
|
|
|
(GLExtensions::Supports("GL_ARB_blend_func_extended") ||
|
2016-10-29 17:24:10 +02:00
|
|
|
GLExtensions::Supports("GL_EXT_blend_func_extended"));
|
2014-01-17 16:10:10 +01:00
|
|
|
g_Config.backend_info.bSupportsPrimitiveRestart =
|
2016-10-30 00:56:18 +02:00
|
|
|
!DriverDetails::HasBug(DriverDetails::BUG_PRIMITIVE_RESTART) &&
|
2014-01-17 16:10:10 +01:00
|
|
|
((GLExtensions::Version() >= 310) || GLExtensions::Supports("GL_NV_primitive_restart"));
|
OGL: implement Bounding Box on systems w/o SSBO
This commit should have zero performance effect if SSBOs are supported.
If they aren't (e.g. on all Macs), this commit alters FramebufferManager
to attach a new stencil buffer and VertexManager to draw to it when
bounding box is active. `BBoxRead` gets the pixel data from the buffer
and dumbly loops through it to find the bounding box.
This patch can run Paper Mario: The Thousand-Year Door at almost full
speed (50–60 FPS) without Dual-Core enabled for all common bounding
box-using actions I tested (going through pipes, Plane Mode, Paper
Mode, Prof. Frankly's gate, combat, walking around the overworld, etc.)
on my computer (macOS 10.12.3, 2.8 GHz Intel Core i7, 16 GB 1600 MHz
DDR3, and Intel Iris 1536 MB).
A few more demanding scenes (e.g. the self-building bridge on the way
to Petalburg) slow to ~15% of their speed without this patch (though
they don't run quite at full speed even on master). The slowdown is
caused almost solely by `glReadPixels` in `OGL::BoundingBox::Get`.
Other implementation ideas:
- Use a stencil buffer that's separate from the depth buffer. This would
require ARB_texture_stencil8 / OpenGL 4.4, which isn't available on
macOS.
- Use `glGetTexImage` instead of `glReadPixels`. This is ~5 FPS slower
on my computer, presumably because it has to transfer the entire
combined depth-stencil buffer instead of only the stencil data.
Getting only stencil data from `glGetTexImage` requires
ARB_texture_stencil8 / OpenGL 4.4, which (again) is not available on
macOS.
- Don't use a PBO, and use `glReadPixels` synchronously. This has no
visible performance effect on my computer, and is theoretically
slower.
2017-03-05 15:34:30 -08:00
|
|
|
g_Config.backend_info.bSupportsFragmentStoresAndAtomics =
|
2014-11-13 23:26:49 +01:00
|
|
|
GLExtensions::Supports("GL_ARB_shader_storage_buffer_object");
|
2022-09-30 20:37:27 -05:00
|
|
|
g_Config.backend_info.bSupportsVSLinePointExpand =
|
|
|
|
GLExtensions::Supports("GL_ARB_shader_storage_buffer_object");
|
2014-10-29 14:51:12 +01:00
|
|
|
g_Config.backend_info.bSupportsGSInstancing = GLExtensions::Supports("GL_ARB_gpu_shader5");
|
2015-09-07 22:21:11 +02:00
|
|
|
g_Config.backend_info.bSupportsSSAA = GLExtensions::Supports("GL_ARB_gpu_shader5") &&
|
|
|
|
GLExtensions::Supports("GL_ARB_sample_shading");
|
2016-01-09 12:07:08 -06:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders =
|
|
|
|
GLExtensions::Version() >= 320 &&
|
2016-10-30 00:56:18 +02:00
|
|
|
!DriverDetails::HasBug(DriverDetails::BUG_BROKEN_GEOMETRY_SHADERS);
|
2015-09-01 05:23:43 -05:00
|
|
|
g_Config.backend_info.bSupportsPaletteConversion =
|
|
|
|
GLExtensions::Supports("GL_ARB_texture_buffer_object") ||
|
|
|
|
GLExtensions::Supports("GL_OES_texture_buffer") ||
|
|
|
|
GLExtensions::Supports("GL_EXT_texture_buffer");
|
2015-05-19 01:18:04 +02:00
|
|
|
g_Config.backend_info.bSupportsClipControl = GLExtensions::Supports("GL_ARB_clip_control");
|
2015-07-25 12:20:00 +02:00
|
|
|
g_ogl_config.bSupportsCopySubImage =
|
|
|
|
(GLExtensions::Supports("GL_ARB_copy_image") || GLExtensions::Supports("GL_NV_copy_image") ||
|
|
|
|
GLExtensions::Supports("GL_EXT_copy_image") ||
|
|
|
|
GLExtensions::Supports("GL_OES_copy_image")) &&
|
2016-10-30 00:56:18 +02:00
|
|
|
!DriverDetails::HasBug(DriverDetails::BUG_BROKEN_COPYIMAGE);
|
2017-10-22 00:49:40 +10:00
|
|
|
g_ogl_config.bSupportsTextureSubImage = GLExtensions::Supports("ARB_get_texture_sub_image");
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-03-30 13:58:05 -05:00
|
|
|
// Desktop OpenGL supports the binding layout if it supports 420pack
|
|
|
|
// OpenGL ES 3.1 supports it implicitly without an extension
|
|
|
|
g_Config.backend_info.bSupportsBindingLayout =
|
|
|
|
GLExtensions::Supports("GL_ARB_shading_language_420pack");
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-08-05 22:31:34 +02:00
|
|
|
// Clip distance support is useless without a method to clamp the depth range
|
2016-08-17 20:12:44 +02:00
|
|
|
g_Config.backend_info.bSupportsDepthClamp = GLExtensions::Supports("GL_ARB_depth_clamp");
|
2016-08-05 22:31:34 +02:00
|
|
|
|
2017-07-20 15:25:24 +10:00
|
|
|
// Desktop OpenGL supports bitfield manulipation and dynamic sampler indexing if it supports
|
|
|
|
// shader5. OpenGL ES 3.1 supports it implicitly without an extension
|
|
|
|
g_Config.backend_info.bSupportsBitfield = GLExtensions::Supports("GL_ARB_gpu_shader5");
|
|
|
|
g_Config.backend_info.bSupportsDynamicSamplerIndexing =
|
|
|
|
GLExtensions::Supports("GL_ARB_gpu_shader5");
|
|
|
|
|
2018-11-07 21:20:12 +10:00
|
|
|
g_ogl_config.bIsES = m_main_gl_context->IsGLES();
|
2019-04-15 23:38:10 +10:00
|
|
|
supports_glsl_cache = GLExtensions::Supports("GL_ARB_get_program_binary");
|
2013-12-30 07:22:50 -06:00
|
|
|
g_ogl_config.bSupportsGLPinnedMemory = GLExtensions::Supports("GL_AMD_pinned_memory");
|
|
|
|
g_ogl_config.bSupportsGLSync = GLExtensions::Supports("GL_ARB_sync");
|
2014-11-21 18:52:39 -06:00
|
|
|
g_ogl_config.bSupportsGLBaseVertex = GLExtensions::Supports("GL_ARB_draw_elements_base_vertex") ||
|
2015-01-18 15:23:51 -06:00
|
|
|
GLExtensions::Supports("GL_EXT_draw_elements_base_vertex") ||
|
|
|
|
GLExtensions::Supports("GL_OES_draw_elements_base_vertex");
|
|
|
|
g_ogl_config.bSupportsGLBufferStorage = GLExtensions::Supports("GL_ARB_buffer_storage") ||
|
2015-05-19 00:54:00 +02:00
|
|
|
GLExtensions::Supports("GL_EXT_buffer_storage");
|
2014-04-30 18:48:40 +02:00
|
|
|
g_ogl_config.bSupportsMSAA = GLExtensions::Supports("GL_ARB_texture_multisample");
|
2013-12-30 07:22:50 -06:00
|
|
|
g_ogl_config.bSupportViewportFloat = GLExtensions::Supports("GL_ARB_viewport_array");
|
2015-08-26 00:09:32 -05:00
|
|
|
g_ogl_config.bSupportsDebug =
|
|
|
|
GLExtensions::Supports("GL_KHR_debug") || GLExtensions::Supports("GL_ARB_debug_output");
|
2016-11-27 18:14:53 +10:00
|
|
|
g_ogl_config.bSupportsTextureStorage = GLExtensions::Supports("GL_ARB_texture_storage");
|
2023-03-05 22:03:26 -08:00
|
|
|
g_ogl_config.SupportedMultisampleTexStorage = MultisampleTexStorageType::TexStorageNone;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore = GLExtensions::Supports("GL_ARB_shader_image_load_store");
|
2015-10-18 00:43:22 +13:00
|
|
|
g_ogl_config.bSupportsConservativeDepth = GLExtensions::Supports("GL_ARB_conservative_depth");
|
2015-11-19 00:46:26 -06:00
|
|
|
g_ogl_config.bSupportsAniso = GLExtensions::Supports("GL_EXT_texture_filter_anisotropic");
|
2016-11-27 18:14:56 +10:00
|
|
|
g_Config.backend_info.bSupportsComputeShaders = GLExtensions::Supports("GL_ARB_compute_shader");
|
2017-04-16 20:34:02 +10:00
|
|
|
g_Config.backend_info.bSupportsST3CTextures =
|
|
|
|
GLExtensions::Supports("GL_EXT_texture_compression_s3tc");
|
2017-07-27 22:00:04 +10:00
|
|
|
g_Config.backend_info.bSupportsBPTCTextures =
|
|
|
|
GLExtensions::Supports("GL_ARB_texture_compression_bptc");
|
2021-11-13 20:10:20 -08:00
|
|
|
g_Config.backend_info.bSupportsCoarseDerivatives =
|
|
|
|
GLExtensions::Supports("GL_ARB_derivative_control") || GLExtensions::Version() >= 450;
|
2021-11-13 20:10:55 -08:00
|
|
|
g_Config.backend_info.bSupportsTextureQueryLevels =
|
|
|
|
GLExtensions::Supports("GL_ARB_texture_query_levels") || GLExtensions::Version() >= 430;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2022-12-31 19:55:53 -08:00
|
|
|
if (GLExtensions::Supports("GL_EXT_shader_framebuffer_fetch"))
|
|
|
|
{
|
|
|
|
g_ogl_config.SupportedFramebufferFetch = EsFbFetchType::FbFetchExt;
|
|
|
|
}
|
|
|
|
else if (GLExtensions::Supports("GL_ARM_shader_framebuffer_fetch"))
|
|
|
|
{
|
|
|
|
g_ogl_config.SupportedFramebufferFetch = EsFbFetchType::FbFetchArm;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_ogl_config.SupportedFramebufferFetch = EsFbFetchType::FbFetchNone;
|
|
|
|
}
|
|
|
|
g_Config.backend_info.bSupportsFramebufferFetch =
|
|
|
|
g_ogl_config.SupportedFramebufferFetch != EsFbFetchType::FbFetchNone;
|
|
|
|
|
2018-10-03 23:03:26 +10:00
|
|
|
if (m_main_gl_context->IsGLES())
|
2014-03-30 13:58:05 -05:00
|
|
|
{
|
2023-03-05 21:50:00 -08:00
|
|
|
g_ogl_config.SupportedESPointSize =
|
|
|
|
GLExtensions::Supports("GL_OES_geometry_point_size") ? EsPointSizeType::PointSizeOes :
|
|
|
|
GLExtensions::Supports("GL_EXT_geometry_point_size") ? EsPointSizeType::PointSizeExt :
|
|
|
|
EsPointSizeType::PointSizeNone;
|
2021-08-30 00:17:24 +02:00
|
|
|
g_ogl_config.SupportedESTextureBuffer =
|
|
|
|
GLExtensions::Supports("VERSION_GLES_3_2") ? EsTexbufType::TexbufCore :
|
|
|
|
GLExtensions::Supports("GL_OES_texture_buffer") ? EsTexbufType::TexbufOes :
|
|
|
|
GLExtensions::Supports("GL_EXT_texture_buffer") ? EsTexbufType::TexbufExt :
|
|
|
|
EsTexbufType::TexbufNone;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-04-15 23:38:10 +10:00
|
|
|
supports_glsl_cache = true;
|
2015-12-13 11:39:45 -06:00
|
|
|
g_ogl_config.bSupportsGLSync = true;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-08-12 13:55:35 +02:00
|
|
|
// TODO: Implement support for GL_EXT_clip_cull_distance when there is an extension for
|
|
|
|
// depth clamping.
|
2016-08-05 22:31:34 +02:00
|
|
|
g_Config.backend_info.bSupportsDepthClamp = false;
|
|
|
|
|
2018-05-26 00:04:18 +10:00
|
|
|
// GLES does not support logic op.
|
|
|
|
g_Config.backend_info.bSupportsLogicOp = false;
|
|
|
|
|
2020-05-24 16:11:10 +10:00
|
|
|
// glReadPixels() can't be used with non-color formats. But, if we support
|
|
|
|
// ARB_get_texture_sub_image (unlikely, except maybe on NVIDIA), we can use that instead.
|
|
|
|
g_Config.backend_info.bSupportsDepthReadback = g_ogl_config.bSupportsTextureSubImage;
|
|
|
|
|
2021-08-06 23:26:51 -04:00
|
|
|
// GL_TEXTURE_LOD_BIAS is not supported on GLES.
|
|
|
|
g_Config.backend_info.bSupportsLodBiasInSampler = false;
|
|
|
|
|
2016-08-17 19:20:24 -04:00
|
|
|
if (GLExtensions::Version() == 300)
|
2014-03-30 13:58:05 -05:00
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = GlslEs300;
|
2014-12-07 05:29:36 +00:00
|
|
|
g_ogl_config.bSupportsAEP = false;
|
2016-11-27 18:14:53 +10:00
|
|
|
g_ogl_config.bSupportsTextureStorage = true;
|
2014-12-17 00:26:03 +01:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders = false;
|
2014-03-30 13:58:05 -05:00
|
|
|
}
|
2016-08-17 19:20:24 -04:00
|
|
|
else if (GLExtensions::Version() == 310)
|
2014-03-30 13:58:05 -05:00
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = GlslEs310;
|
2014-12-07 05:29:36 +00:00
|
|
|
g_ogl_config.bSupportsAEP = GLExtensions::Supports("GL_ANDROID_extension_pack_es31a");
|
2014-03-30 13:58:05 -05:00
|
|
|
g_Config.backend_info.bSupportsBindingLayout = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore = true;
|
2014-12-17 00:26:03 +01:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders = g_ogl_config.bSupportsAEP;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_Config.backend_info.bSupportsComputeShaders = true;
|
2015-08-26 00:09:32 -05:00
|
|
|
g_Config.backend_info.bSupportsGSInstancing =
|
2023-03-05 21:50:00 -08:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders &&
|
|
|
|
g_ogl_config.SupportedESPointSize != EsPointSizeType::PointSizeNone;
|
2015-09-06 13:58:18 +02:00
|
|
|
g_Config.backend_info.bSupportsSSAA = g_ogl_config.bSupportsAEP;
|
2017-03-05 15:17:54 -08:00
|
|
|
g_Config.backend_info.bSupportsFragmentStoresAndAtomics = true;
|
2015-09-01 00:17:24 -05:00
|
|
|
g_ogl_config.bSupportsMSAA = true;
|
2016-11-27 18:14:53 +10:00
|
|
|
g_ogl_config.bSupportsTextureStorage = true;
|
2023-03-05 22:03:26 -08:00
|
|
|
if (GLExtensions::Supports("GL_OES_texture_storage_multisample_2d_array"))
|
|
|
|
g_ogl_config.SupportedMultisampleTexStorage = MultisampleTexStorageType::TexStorageOes;
|
2017-07-20 15:25:24 +10:00
|
|
|
g_Config.backend_info.bSupportsBitfield = true;
|
|
|
|
g_Config.backend_info.bSupportsDynamicSamplerIndexing = g_ogl_config.bSupportsAEP;
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2015-08-26 00:09:32 -05:00
|
|
|
else
|
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = GlslEs320;
|
2015-08-26 00:09:32 -05:00
|
|
|
g_ogl_config.bSupportsAEP = GLExtensions::Supports("GL_ANDROID_extension_pack_es31a");
|
|
|
|
g_Config.backend_info.bSupportsBindingLayout = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore = true;
|
2015-08-26 00:09:32 -05:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_Config.backend_info.bSupportsComputeShaders = true;
|
2023-03-05 21:50:00 -08:00
|
|
|
g_Config.backend_info.bSupportsGSInstancing =
|
|
|
|
g_ogl_config.SupportedESPointSize != EsPointSizeType::PointSizeNone;
|
2015-09-01 05:23:43 -05:00
|
|
|
g_Config.backend_info.bSupportsPaletteConversion = true;
|
2015-09-01 00:17:24 -05:00
|
|
|
g_Config.backend_info.bSupportsSSAA = true;
|
2017-03-05 15:17:54 -08:00
|
|
|
g_Config.backend_info.bSupportsFragmentStoresAndAtomics = true;
|
2015-08-26 00:09:32 -05:00
|
|
|
g_ogl_config.bSupportsCopySubImage = true;
|
|
|
|
g_ogl_config.bSupportsGLBaseVertex = true;
|
|
|
|
g_ogl_config.bSupportsDebug = true;
|
2015-09-01 00:17:24 -05:00
|
|
|
g_ogl_config.bSupportsMSAA = true;
|
2016-11-27 18:14:53 +10:00
|
|
|
g_ogl_config.bSupportsTextureStorage = true;
|
2023-03-05 22:03:26 -08:00
|
|
|
g_ogl_config.SupportedMultisampleTexStorage = MultisampleTexStorageType::TexStorageCore;
|
2017-07-20 15:25:24 +10:00
|
|
|
g_Config.backend_info.bSupportsBitfield = true;
|
|
|
|
g_Config.backend_info.bSupportsDynamicSamplerIndexing = true;
|
2022-01-30 21:46:06 -05:00
|
|
|
g_Config.backend_info.bSupportsSettingObjectNames = true;
|
2014-03-30 13:58:05 -05:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2013-12-30 07:22:50 -06:00
|
|
|
else
|
2013-04-08 14:50:58 +02:00
|
|
|
{
|
2023-03-05 22:03:26 -08:00
|
|
|
if (GLExtensions::Supports("GL_ARB_texture_storage_multisample"))
|
|
|
|
g_ogl_config.SupportedMultisampleTexStorage = MultisampleTexStorageType::TexStorageCore;
|
|
|
|
|
2016-08-17 19:20:24 -04:00
|
|
|
if (GLExtensions::Version() < 300)
|
2013-12-30 07:22:50 -06:00
|
|
|
{
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Need at least GLSL 1.30\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3.0?\n"
|
|
|
|
"GPU: Your driver supports GLSL {0}",
|
|
|
|
reinterpret_cast<const char*>(glGetString(GL_SHADING_LANGUAGE_VERSION)));
|
2013-12-30 07:22:50 -06:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-08-17 19:20:24 -04:00
|
|
|
else if (GLExtensions::Version() == 300)
|
2013-12-30 07:22:50 -06:00
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl130;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore = false; // layout keyword is only supported on glsl150+
|
2015-10-18 00:43:22 +13:00
|
|
|
g_ogl_config.bSupportsConservativeDepth =
|
|
|
|
false; // layout keyword is only supported on glsl150+
|
2014-12-17 00:26:03 +01:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders =
|
|
|
|
false; // geometry shaders are only supported on glsl150+
|
2013-12-30 07:22:50 -06:00
|
|
|
}
|
2016-08-17 19:20:24 -04:00
|
|
|
else if (GLExtensions::Version() == 310)
|
2013-12-30 07:22:50 -06:00
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl140;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore = false; // layout keyword is only supported on glsl150+
|
2015-10-18 00:43:22 +13:00
|
|
|
g_ogl_config.bSupportsConservativeDepth =
|
2014-12-17 00:26:03 +01:00
|
|
|
false; // layout keyword is only supported on glsl150+
|
2015-09-06 13:58:18 +02:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders =
|
2013-12-30 07:22:50 -06:00
|
|
|
false; // geometry shaders are only supported on glsl150+
|
2015-09-06 13:58:18 +02:00
|
|
|
}
|
2016-08-17 19:20:24 -04:00
|
|
|
else if (GLExtensions::Version() == 320)
|
2015-09-06 13:58:18 +02:00
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl150;
|
2015-09-06 13:58:18 +02:00
|
|
|
}
|
2016-08-17 19:20:24 -04:00
|
|
|
else if (GLExtensions::Version() == 330)
|
2015-09-06 13:58:18 +02:00
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl330;
|
2023-05-28 20:59:02 -05:00
|
|
|
g_ogl_config.bSupportsExplicitLayoutInShader =
|
|
|
|
GLExtensions::Supports("GL_ARB_explicit_attrib_location");
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2016-11-27 18:14:56 +10:00
|
|
|
else if (GLExtensions::Version() >= 430)
|
|
|
|
{
|
|
|
|
// TODO: We should really parse the GL_SHADING_LANGUAGE_VERSION token.
|
2023-01-31 18:10:48 +01:00
|
|
|
if (GLExtensions::Version() >= 450)
|
|
|
|
{
|
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl450;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl430;
|
|
|
|
}
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsTextureStorage = true;
|
2023-03-05 22:03:26 -08:00
|
|
|
g_ogl_config.SupportedMultisampleTexStorage = MultisampleTexStorageType::TexStorageCore;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore = true;
|
2023-05-28 20:59:02 -05:00
|
|
|
g_ogl_config.bSupportsExplicitLayoutInShader = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
g_Config.backend_info.bSupportsSSAA = true;
|
2022-01-30 21:46:06 -05:00
|
|
|
g_Config.backend_info.bSupportsSettingObjectNames = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
|
|
|
|
// Compute shaders are core in GL4.3.
|
|
|
|
g_Config.backend_info.bSupportsComputeShaders = true;
|
2017-10-22 00:49:40 +10:00
|
|
|
if (GLExtensions::Version() >= 450)
|
|
|
|
g_ogl_config.bSupportsTextureSubImage = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
else
|
|
|
|
{
|
2018-01-05 10:01:18 -08:00
|
|
|
g_ogl_config.eSupportedGLSLVersion = Glsl400;
|
2014-12-07 05:29:36 +00:00
|
|
|
g_Config.backend_info.bSupportsSSAA = true;
|
2016-11-27 18:14:56 +10:00
|
|
|
|
|
|
|
if (GLExtensions::Version() == 420)
|
|
|
|
{
|
|
|
|
// Texture storage and shader image load/store are core in GL4.2.
|
|
|
|
g_ogl_config.bSupportsTextureStorage = true;
|
|
|
|
g_ogl_config.bSupportsImageLoadStore = true;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
2014-12-07 05:29:36 +00:00
|
|
|
// Desktop OpenGL can't have the Android Extension Pack
|
|
|
|
g_ogl_config.bSupportsAEP = false;
|
2019-04-21 12:36:41 +10:00
|
|
|
|
|
|
|
// Desktop GL requires GL_PROGRAM_POINT_SIZE set to use gl_PointSize in shaders.
|
|
|
|
// It is implicitly enabled in GLES.
|
|
|
|
glEnable(GL_PROGRAM_POINT_SIZE);
|
2013-04-08 14:50:58 +02:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2022-12-11 01:15:15 -06:00
|
|
|
// Supported by all GS-supporting ES and 4.3+
|
|
|
|
g_Config.backend_info.bSupportsGLLayerInFS = g_Config.backend_info.bSupportsGeometryShaders &&
|
|
|
|
g_ogl_config.eSupportedGLSLVersion >= Glsl430;
|
|
|
|
|
2020-11-24 21:22:39 +01:00
|
|
|
g_Config.backend_info.bSupportsBBox = g_Config.backend_info.bSupportsFragmentStoresAndAtomics;
|
|
|
|
|
2015-10-18 00:43:22 +13:00
|
|
|
// Either method can do early-z tests. See PixelShaderGen for details.
|
|
|
|
g_Config.backend_info.bSupportsEarlyZ =
|
2016-11-27 18:14:56 +10:00
|
|
|
g_ogl_config.bSupportsImageLoadStore || g_ogl_config.bSupportsConservativeDepth;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2023-03-19 22:20:34 -07:00
|
|
|
g_Config.backend_info.AAModes.clear();
|
|
|
|
if (g_ogl_config.bSupportsMSAA)
|
|
|
|
{
|
|
|
|
bool supportsGetInternalFormat =
|
|
|
|
GLExtensions::Supports("VERSION_4_2") || GLExtensions::Supports("VERSION_GLES_3");
|
|
|
|
if (supportsGetInternalFormat)
|
|
|
|
{
|
|
|
|
// Note: GL_TEXTURE_2D_MULTISAMPLE_ARRAY_OES should technically be used for
|
|
|
|
// GL_OES_texture_storage_multisample_2d_array, but both are 0x9102 so it does not matter.
|
|
|
|
|
|
|
|
std::vector<int> color_aa_modes;
|
|
|
|
{
|
|
|
|
GLenum colorInternalFormat = OGLTexture::GetGLInternalFormatForTextureFormat(
|
|
|
|
FramebufferManager::GetEFBColorFormat(), true);
|
|
|
|
GLint num_color_sample_counts = 0;
|
|
|
|
glGetInternalformativ(GL_TEXTURE_2D_MULTISAMPLE_ARRAY, colorInternalFormat,
|
|
|
|
GL_NUM_SAMPLE_COUNTS, 1, &num_color_sample_counts);
|
|
|
|
|
|
|
|
ASSERT_MSG(VIDEO, num_color_sample_counts >= 0,
|
|
|
|
"negative GL_NUM_SAMPLE_COUNTS for colors does not make sense: {}",
|
|
|
|
num_color_sample_counts);
|
|
|
|
color_aa_modes.reserve(num_color_sample_counts + 1);
|
|
|
|
if (num_color_sample_counts > 0)
|
|
|
|
{
|
|
|
|
color_aa_modes.resize(num_color_sample_counts);
|
|
|
|
|
|
|
|
static_assert(sizeof(GLint) == sizeof(u32));
|
|
|
|
glGetInternalformativ(GL_TEXTURE_2D_MULTISAMPLE_ARRAY, colorInternalFormat, GL_SAMPLES,
|
|
|
|
num_color_sample_counts,
|
|
|
|
reinterpret_cast<GLint*>(color_aa_modes.data()));
|
|
|
|
ASSERT_MSG(VIDEO, std::is_sorted(color_aa_modes.rbegin(), color_aa_modes.rend()),
|
|
|
|
"GPU driver didn't return sorted color AA modes: [{}]",
|
|
|
|
fmt::join(color_aa_modes, ", "));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (color_aa_modes.empty() || color_aa_modes.back() != 1)
|
|
|
|
color_aa_modes.push_back(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<int> depth_aa_modes;
|
|
|
|
{
|
|
|
|
GLenum depthInternalFormat = OGLTexture::GetGLInternalFormatForTextureFormat(
|
|
|
|
FramebufferManager::GetEFBColorFormat(), true);
|
|
|
|
GLint num_depth_sample_counts = 0;
|
|
|
|
glGetInternalformativ(GL_TEXTURE_2D_MULTISAMPLE_ARRAY, depthInternalFormat,
|
|
|
|
GL_NUM_SAMPLE_COUNTS, 1, &num_depth_sample_counts);
|
|
|
|
|
|
|
|
ASSERT_MSG(VIDEO, num_depth_sample_counts >= 0,
|
|
|
|
"negative GL_NUM_SAMPLE_COUNTS for depth does not make sense: {}",
|
|
|
|
num_depth_sample_counts);
|
|
|
|
depth_aa_modes.reserve(num_depth_sample_counts + 1);
|
|
|
|
if (num_depth_sample_counts > 0)
|
|
|
|
{
|
|
|
|
depth_aa_modes.resize(num_depth_sample_counts);
|
|
|
|
|
|
|
|
static_assert(sizeof(GLint) == sizeof(u32));
|
|
|
|
glGetInternalformativ(GL_TEXTURE_2D_MULTISAMPLE_ARRAY, depthInternalFormat, GL_SAMPLES,
|
|
|
|
num_depth_sample_counts,
|
|
|
|
reinterpret_cast<GLint*>(depth_aa_modes.data()));
|
|
|
|
ASSERT_MSG(VIDEO, std::is_sorted(depth_aa_modes.rbegin(), depth_aa_modes.rend()),
|
|
|
|
"GPU driver didn't return sorted depth AA modes: [{}]",
|
|
|
|
fmt::join(depth_aa_modes, ", "));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (depth_aa_modes.empty() || depth_aa_modes.back() != 1)
|
|
|
|
depth_aa_modes.push_back(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// The spec says supported sample formats are returned in descending numeric order.
|
|
|
|
// It also says "Only positive values are returned", but does not specify whether 1 is
|
|
|
|
// included or not; it seems like NVIDIA and Intel GPUs do not include it.
|
|
|
|
// We've inserted 1 at the back of both if not present to handle this.
|
|
|
|
g_Config.backend_info.AAModes.clear();
|
|
|
|
g_Config.backend_info.AAModes.reserve(std::min(color_aa_modes.size(), depth_aa_modes.size()));
|
|
|
|
// We only want AA modes that are supported for both the color and depth textures. Probably
|
|
|
|
// the support is the same, though. rbegin/rend are used to swap the order ahead of time.
|
|
|
|
std::set_intersection(color_aa_modes.rbegin(), color_aa_modes.rend(), depth_aa_modes.rbegin(),
|
|
|
|
depth_aa_modes.rend(),
|
|
|
|
std::back_inserter(g_Config.backend_info.AAModes));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// The documentation for glGetInternalformativ says its value is at least the minimum of
|
|
|
|
// GL_MAX_SAMPLES, GL_MAX_COLOR_TEXTURE_SAMPLES, and GL_MAX_DEPTH_TEXTURE_SAMPLES (and
|
|
|
|
// GL_MAX_INTEGER_SAMPLES for integer textures, assumed not applicable here).
|
|
|
|
GLint max_color_samples = 0;
|
|
|
|
glGetIntegerv(GL_MAX_COLOR_TEXTURE_SAMPLES, &max_color_samples);
|
|
|
|
GLint max_depth_samples = 0;
|
|
|
|
glGetIntegerv(GL_MAX_COLOR_TEXTURE_SAMPLES, &max_depth_samples);
|
|
|
|
// Note: The desktop OpenGL ref pages don't actually say that GL_MAX_SAMPLES is a valid
|
|
|
|
// parameter for glGetIntegerv (though the ES ones do). However, MAX_SAMPLES is
|
|
|
|
// referenced in the GL 3.1 spec and by GL_ARB_texture_multisample (which is written against
|
|
|
|
// the OpenGL 3.1 spec), so presumably it is valid.
|
|
|
|
GLint max_samples = 0;
|
|
|
|
glGetIntegerv(GL_MAX_SAMPLES, &max_samples);
|
|
|
|
u32 supported_max_samples =
|
|
|
|
static_cast<u32>(std::min({max_samples, max_color_samples, max_depth_samples}));
|
|
|
|
|
|
|
|
while (supported_max_samples > 1)
|
|
|
|
{
|
|
|
|
g_Config.backend_info.AAModes.push_back(supported_max_samples);
|
|
|
|
supported_max_samples /= 2;
|
|
|
|
}
|
|
|
|
g_Config.backend_info.AAModes.push_back(1);
|
|
|
|
// The UI wants ascending order
|
|
|
|
std::reverse(g_Config.backend_info.AAModes.begin(), g_Config.backend_info.AAModes.end());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_Config.backend_info.AAModes = {1};
|
|
|
|
}
|
2017-03-31 21:55:26 +10:00
|
|
|
|
2023-01-31 18:10:48 +01:00
|
|
|
const bool bSupportsIsHelperInvocation = g_ogl_config.bIsES ?
|
|
|
|
g_ogl_config.eSupportedGLSLVersion >= GlslEs320 :
|
|
|
|
g_ogl_config.eSupportedGLSLVersion >= Glsl450;
|
|
|
|
g_ogl_config.bSupportsKHRShaderSubgroup =
|
|
|
|
GLExtensions::Supports("GL_KHR_shader_subgroup") && bSupportsIsHelperInvocation;
|
|
|
|
if (g_ogl_config.bSupportsKHRShaderSubgroup)
|
|
|
|
{
|
|
|
|
// Check for the features: basic + arithmetic + ballot
|
|
|
|
GLint supported_features = 0;
|
|
|
|
glGetIntegerv(GL_SUBGROUP_SUPPORTED_FEATURES_KHR, &supported_features);
|
|
|
|
if (~supported_features &
|
|
|
|
(GL_SUBGROUP_FEATURE_BASIC_BIT_KHR | GL_SUBGROUP_FEATURE_ARITHMETIC_BIT_KHR |
|
|
|
|
GL_SUBGROUP_FEATURE_BALLOT_BIT_KHR))
|
|
|
|
{
|
|
|
|
g_ogl_config.bSupportsKHRShaderSubgroup = false;
|
|
|
|
}
|
|
|
|
}
|
2019-03-22 20:39:11 +10:00
|
|
|
|
2016-11-27 18:15:00 +10:00
|
|
|
// We require texel buffers, image load store, and compute shaders to enable GPU texture decoding.
|
|
|
|
// If the driver doesn't expose the extensions, but supports GL4.3/GLES3.1, it will still be
|
|
|
|
// enabled in the version check below.
|
|
|
|
g_Config.backend_info.bSupportsGPUTextureDecoding =
|
|
|
|
g_Config.backend_info.bSupportsPaletteConversion &&
|
|
|
|
g_Config.backend_info.bSupportsComputeShaders && g_ogl_config.bSupportsImageLoadStore;
|
|
|
|
|
2018-02-25 17:56:09 +10:00
|
|
|
// Background compiling is supported only when shared contexts aren't broken.
|
|
|
|
g_Config.backend_info.bSupportsBackgroundCompiling =
|
|
|
|
!DriverDetails::HasBug(DriverDetails::BUG_SHARED_CONTEXT_SHADER_COMPILATION);
|
|
|
|
|
2019-04-15 23:38:10 +10:00
|
|
|
// Program binaries are supported on GL4.1+, ARB_get_program_binary, or ES3.
|
|
|
|
if (supports_glsl_cache)
|
|
|
|
{
|
|
|
|
// We need to check the number of formats supported. If zero, don't bother getting the binaries.
|
|
|
|
GLint num_formats = 0;
|
|
|
|
glGetIntegerv(GL_NUM_PROGRAM_BINARY_FORMATS, &num_formats);
|
|
|
|
supports_glsl_cache = num_formats > 0;
|
|
|
|
}
|
|
|
|
g_Config.backend_info.bSupportsPipelineCacheData = supports_glsl_cache;
|
|
|
|
|
2013-11-05 17:37:32 +01:00
|
|
|
int samples;
|
|
|
|
glGetIntegerv(GL_SAMPLES, &samples);
|
2014-03-11 00:30:55 +13:00
|
|
|
if (samples > 1)
|
2013-11-05 17:37:32 +01:00
|
|
|
{
|
|
|
|
// MSAA on default framebuffer isn't working because of glBlitFramebuffer.
|
|
|
|
// It also isn't useful as we don't render anything to the default framebuffer.
|
|
|
|
// We also try to get a non-msaa fb, so this only happens when forced by the driver.
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT(
|
|
|
|
"The graphics driver is forcibly enabling anti-aliasing for Dolphin. You need to "
|
|
|
|
"turn this off in the graphics driver's settings in order for Dolphin to work.\n\n"
|
|
|
|
"(MSAA with {0} samples found on default framebuffer)",
|
|
|
|
samples);
|
2013-11-05 17:37:32 +01:00
|
|
|
bSuccess = false;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-09-02 13:14:45 +02:00
|
|
|
if (!bSuccess)
|
2023-01-27 13:21:09 +13:00
|
|
|
return false;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-10-30 14:56:40 +01:00
|
|
|
g_Config.VerifyValidity();
|
2013-01-21 18:44:33 +01:00
|
|
|
UpdateActiveConfig();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2020-11-09 03:08:00 -05:00
|
|
|
OSD::AddMessage(fmt::format("Video Info: {}, {}, {}", g_ogl_config.gl_vendor,
|
|
|
|
g_ogl_config.gl_renderer, g_ogl_config.gl_version),
|
2013-08-24 01:44:16 +02:00
|
|
|
5000);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-03-12 00:19:14 +01:00
|
|
|
if (!g_ogl_config.bSupportsGLBufferStorage && !g_ogl_config.bSupportsGLPinnedMemory)
|
|
|
|
{
|
2020-11-09 03:08:00 -05:00
|
|
|
OSD::AddMessage(fmt::format("Your OpenGL driver does not support {}_buffer_storage.",
|
|
|
|
m_main_gl_context->IsGLES() ? "EXT" : "ARB"),
|
2018-10-03 23:02:45 +10:00
|
|
|
60000);
|
2021-06-05 15:08:47 -07:00
|
|
|
OSD::AddMessage("This device's performance may be poor.", 60000);
|
2018-03-12 00:19:14 +01:00
|
|
|
}
|
|
|
|
|
2022-02-23 18:30:02 -08:00
|
|
|
INFO_LOG_FMT(VIDEO, "Video Info: {}, {}, {}", g_ogl_config.gl_vendor, g_ogl_config.gl_renderer,
|
|
|
|
g_ogl_config.gl_version);
|
2020-11-09 03:08:00 -05:00
|
|
|
WARN_LOG_FMT(VIDEO, "Missing OGL Extensions: {}{}{}{}{}{}{}{}{}{}{}{}{}{}",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsDualSourceBlend ? "" : "DualSourceBlend ",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsPrimitiveRestart ? "" : "PrimitiveRestart ",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsEarlyZ ? "" : "EarlyZ ",
|
|
|
|
g_ogl_config.bSupportsGLPinnedMemory ? "" : "PinnedMemory ",
|
|
|
|
supports_glsl_cache ? "" : "ShaderCache ",
|
|
|
|
g_ogl_config.bSupportsGLBaseVertex ? "" : "BaseVertex ",
|
|
|
|
g_ogl_config.bSupportsGLBufferStorage ? "" : "BufferStorage ",
|
|
|
|
g_ogl_config.bSupportsGLSync ? "" : "Sync ",
|
|
|
|
g_ogl_config.bSupportsMSAA ? "" : "MSAA ",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsSSAA ? "" : "SSAA ",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsGSInstancing ? "" : "GSInstancing ",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsClipControl ? "" : "ClipControl ",
|
|
|
|
g_ogl_config.bSupportsCopySubImage ? "" : "CopyImageSubData ",
|
|
|
|
g_ActiveConfig.backend_info.bSupportsDepthClamp ? "" : "DepthClamp ");
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-02-09 20:52:25 +10:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:02:45 +10:00
|
|
|
} // namespace OGL
|