2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2008-12-08 05:25:12 +00:00
|
|
|
|
2011-02-14 02:18:03 +00:00
|
|
|
// OpenGL Backend Documentation
|
2009-06-15 06:39:26 +00:00
|
|
|
/*
|
2009-06-06 13:36:33 +00:00
|
|
|
|
|
|
|
1.1 Display settings
|
2009-06-15 06:39:26 +00:00
|
|
|
|
2009-09-09 19:52:45 +00:00
|
|
|
Internal and fullscreen resolution: Since the only internal resolutions allowed
|
|
|
|
are also fullscreen resolution allowed by the system there is only need for one
|
|
|
|
resolution setting that applies to both the internal resolution and the
|
|
|
|
fullscreen resolution. - Apparently no, someone else doesn't agree
|
2009-06-06 13:36:33 +00:00
|
|
|
|
2009-09-09 19:52:45 +00:00
|
|
|
Todo: Make the internal resolution option apply instantly, currently only the
|
|
|
|
native and 2x option applies instantly. To do this we need to be able to change
|
|
|
|
the reinitialize FramebufferManager:Init() while a game is running.
|
2009-06-06 13:36:33 +00:00
|
|
|
|
2009-06-07 11:51:53 +00:00
|
|
|
1.2 Screenshots
|
2009-06-15 06:39:26 +00:00
|
|
|
|
2009-06-07 11:51:53 +00:00
|
|
|
|
2009-09-09 19:52:45 +00:00
|
|
|
The screenshots should be taken from the internal representation of the picture
|
|
|
|
regardless of what the current window size is. Since AA and wireframe is
|
|
|
|
applied together with the picture resizing this rule is not currently applied
|
|
|
|
to AA or wireframe pictures, they are instead taken from whatever the window
|
|
|
|
size is.
|
2009-06-07 11:51:53 +00:00
|
|
|
|
2009-09-09 19:52:45 +00:00
|
|
|
Todo: Render AA and wireframe to a separate picture used for the screenshot in
|
|
|
|
addition to the one for display.
|
2009-06-07 11:51:53 +00:00
|
|
|
|
|
|
|
1.3 AA
|
2009-06-15 06:39:26 +00:00
|
|
|
|
2009-06-07 11:51:53 +00:00
|
|
|
Make AA apply instantly during gameplay if possible
|
|
|
|
|
2009-06-15 06:39:26 +00:00
|
|
|
*/
|
2009-06-06 13:36:33 +00:00
|
|
|
|
2015-12-20 21:49:49 -05:00
|
|
|
#include <memory>
|
2016-01-02 15:01:12 -05:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2018-06-30 04:53:24 -04:00
|
|
|
#include "Common/Common.h"
|
2018-10-03 23:02:45 +10:00
|
|
|
#include "Common/GL/GLContext.h"
|
2015-09-19 04:40:00 +12:00
|
|
|
#include "Common/GL/GLUtil.h"
|
2017-03-10 00:01:23 +10:00
|
|
|
#include "Common/MsgHandler.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2018-10-03 23:03:36 +10:00
|
|
|
#include "Core/Config/GraphicsSettings.h"
|
|
|
|
|
2020-09-15 05:00:24 -07:00
|
|
|
#include "VideoBackends/OGL/OGLBoundingBox.h"
|
|
|
|
#include "VideoBackends/OGL/OGLPerfQuery.h"
|
|
|
|
#include "VideoBackends/OGL/OGLRender.h"
|
|
|
|
#include "VideoBackends/OGL/OGLVertexManager.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoBackends/OGL/ProgramShaderCache.h"
|
|
|
|
#include "VideoBackends/OGL/SamplerCache.h"
|
|
|
|
#include "VideoBackends/OGL/VideoBackend.h"
|
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
#include "VideoCommon/FramebufferManager.h"
|
|
|
|
#include "VideoCommon/TextureCacheBase.h"
|
2019-12-22 13:40:40 -05:00
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/VideoConfig.h"
|
|
|
|
|
2011-01-31 01:28:32 +00:00
|
|
|
namespace OGL
|
2010-12-19 19:43:18 +00:00
|
|
|
{
|
2014-03-11 06:55:00 +01:00
|
|
|
std::string VideoBackend::GetName() const
|
2013-05-01 23:51:43 +10:00
|
|
|
{
|
2020-09-06 12:56:45 +02:00
|
|
|
return NAME;
|
2013-05-01 23:51:43 +10:00
|
|
|
}
|
|
|
|
|
2014-03-11 06:55:00 +01:00
|
|
|
std::string VideoBackend::GetDisplayName() const
|
2009-07-02 17:11:27 +00:00
|
|
|
{
|
2018-11-07 21:20:12 +10:00
|
|
|
if (g_ogl_config.bIsES)
|
2018-06-30 04:53:24 -04:00
|
|
|
return _trans("OpenGL ES");
|
2014-01-18 04:11:59 +00:00
|
|
|
else
|
2018-06-30 04:53:24 -04:00
|
|
|
return _trans("OpenGL");
|
2009-09-09 19:52:45 +00:00
|
|
|
}
|
2009-02-22 04:24:53 +00:00
|
|
|
|
2016-01-13 21:38:11 +01:00
|
|
|
void VideoBackend::InitBackendInfo()
|
2009-09-09 19:52:45 +00:00
|
|
|
{
|
2016-07-21 19:04:57 -04:00
|
|
|
g_Config.backend_info.api_type = APIType::OpenGL;
|
2017-03-10 00:01:23 +10:00
|
|
|
g_Config.backend_info.MaxTextureSize = 16384;
|
2019-02-15 11:59:50 +10:00
|
|
|
g_Config.backend_info.bUsesLowerLeftOrigin = true;
|
2014-07-19 20:18:03 +02:00
|
|
|
g_Config.backend_info.bSupportsExclusiveFullscreen = false;
|
2013-10-14 15:16:42 +02:00
|
|
|
g_Config.backend_info.bSupportsOversizedViewports = true;
|
2014-12-17 00:26:03 +01:00
|
|
|
g_Config.backend_info.bSupportsGeometryShaders = true;
|
2016-11-27 18:14:55 +10:00
|
|
|
g_Config.backend_info.bSupportsComputeShaders = false;
|
2014-11-14 00:24:53 +01:00
|
|
|
g_Config.backend_info.bSupports3DVision = false;
|
2015-01-03 01:33:30 +01:00
|
|
|
g_Config.backend_info.bSupportsPostProcessing = true;
|
2015-09-01 00:17:24 -05:00
|
|
|
g_Config.backend_info.bSupportsSSAA = true;
|
2016-08-21 17:47:24 +02:00
|
|
|
g_Config.backend_info.bSupportsReversedDepthRange = true;
|
2018-05-26 00:04:18 +10:00
|
|
|
g_Config.backend_info.bSupportsLogicOp = true;
|
2016-08-13 22:08:46 +10:00
|
|
|
g_Config.backend_info.bSupportsMultithreading = false;
|
2017-05-29 17:02:09 -05:00
|
|
|
g_Config.backend_info.bSupportsCopyToVram = true;
|
2019-02-15 11:59:50 +10:00
|
|
|
g_Config.backend_info.bSupportsLargePoints = true;
|
2020-05-24 16:11:10 +10:00
|
|
|
g_Config.backend_info.bSupportsDepthReadback = true;
|
2019-03-02 15:03:49 +10:00
|
|
|
g_Config.backend_info.bSupportsPartialDepthCopies = true;
|
2019-04-15 21:55:26 +10:00
|
|
|
g_Config.backend_info.bSupportsShaderBinaries = false;
|
|
|
|
g_Config.backend_info.bSupportsPipelineCacheData = false;
|
2016-11-27 18:15:00 +10:00
|
|
|
|
2019-03-29 08:38:17 -04:00
|
|
|
// TODO: There is a bug here, if texel buffers or SSBOs/atomics are not supported the graphics
|
|
|
|
// options will show the option when it is not supported. The only way around this would be
|
2016-11-27 18:15:00 +10:00
|
|
|
// creating a context when calling this function to determine what is available.
|
|
|
|
g_Config.backend_info.bSupportsGPUTextureDecoding = true;
|
2019-03-29 08:38:17 -04:00
|
|
|
g_Config.backend_info.bSupportsBBox = true;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-01-13 21:14:20 +01:00
|
|
|
// Overwritten in Render.cpp later
|
|
|
|
g_Config.backend_info.bSupportsDualSourceBlend = true;
|
|
|
|
g_Config.backend_info.bSupportsPrimitiveRestart = true;
|
|
|
|
g_Config.backend_info.bSupportsPaletteConversion = true;
|
|
|
|
g_Config.backend_info.bSupportsClipControl = true;
|
2016-08-05 22:31:34 +02:00
|
|
|
g_Config.backend_info.bSupportsDepthClamp = true;
|
2017-04-16 19:30:11 +10:00
|
|
|
g_Config.backend_info.bSupportsST3CTextures = false;
|
2017-07-27 22:00:04 +10:00
|
|
|
g_Config.backend_info.bSupportsBPTCTextures = false;
|
2016-01-13 21:14:20 +01:00
|
|
|
|
2014-04-12 00:17:36 -07:00
|
|
|
g_Config.backend_info.Adapters.clear();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-09-14 18:49:48 -07:00
|
|
|
// aamodes - 1 is to stay consistent with D3D (means no AA)
|
2015-10-10 16:47:58 +02:00
|
|
|
g_Config.backend_info.AAModes = {1, 2, 4, 8};
|
2011-05-31 20:16:59 +00:00
|
|
|
}
|
2010-11-22 22:17:35 +00:00
|
|
|
|
2018-10-03 23:03:26 +10:00
|
|
|
bool VideoBackend::InitializeGLExtensions(GLContext* context)
|
2017-03-10 00:01:23 +10:00
|
|
|
{
|
|
|
|
// Init extension support.
|
2018-10-03 23:03:26 +10:00
|
|
|
if (!GLExtensions::Init(context))
|
2017-03-10 00:01:23 +10:00
|
|
|
{
|
|
|
|
// OpenGL 2.0 is required for all shader based drawings. There is no way to get this by
|
|
|
|
// extensions
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Does your video card support OpenGL 2.0?");
|
2017-03-10 00:01:23 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GLExtensions::Version() < 300)
|
|
|
|
{
|
|
|
|
// integer vertex attributes require a gl3 only function
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Need OpenGL version 3.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 3?");
|
2017-03-10 00:01:23 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VideoBackend::FillBackendInfo()
|
|
|
|
{
|
2019-03-09 23:31:40 +10:00
|
|
|
InitBackendInfo();
|
|
|
|
|
2017-03-10 00:01:23 +10:00
|
|
|
// check for the max vertex attributes
|
|
|
|
GLint numvertexattribs = 0;
|
|
|
|
glGetIntegerv(GL_MAX_VERTEX_ATTRIBS, &numvertexattribs);
|
|
|
|
if (numvertexattribs < 16)
|
|
|
|
{
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GPU: OGL ERROR: Number of attributes {0} not enough.\n"
|
|
|
|
"GPU: Does your video card support OpenGL 2.x?",
|
|
|
|
numvertexattribs);
|
2017-03-10 00:01:23 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check the max texture width and height
|
|
|
|
GLint max_texture_size = 0;
|
|
|
|
glGetIntegerv(GL_MAX_TEXTURE_SIZE, &max_texture_size);
|
|
|
|
g_Config.backend_info.MaxTextureSize = static_cast<u32>(max_texture_size);
|
|
|
|
if (max_texture_size < 1024)
|
|
|
|
{
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("GL_MAX_TEXTURE_SIZE is {0} - must be at least 1024.", max_texture_size);
|
2017-03-10 00:01:23 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Move the remaining fields from the Renderer constructor here.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:03:22 +10:00
|
|
|
bool VideoBackend::Initialize(const WindowSystemInfo& wsi)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2018-10-03 23:03:26 +10:00
|
|
|
std::unique_ptr<GLContext> main_gl_context =
|
2019-03-09 23:31:40 +10:00
|
|
|
GLContext::Create(wsi, g_Config.stereo_mode == StereoMode::QuadBuffer, true, false,
|
2018-10-03 23:03:36 +10:00
|
|
|
Config::Get(Config::GFX_PREFER_GLES));
|
2018-10-03 23:03:26 +10:00
|
|
|
if (!main_gl_context)
|
2013-06-19 09:17:33 +02:00
|
|
|
return false;
|
|
|
|
|
2018-10-03 23:03:26 +10:00
|
|
|
if (!InitializeGLExtensions(main_gl_context.get()) || !FillBackendInfo())
|
2018-01-26 15:09:07 +10:00
|
|
|
return false;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-09 23:31:40 +10:00
|
|
|
InitializeShared();
|
2019-01-19 00:35:00 +10:00
|
|
|
g_renderer = std::make_unique<Renderer>(std::move(main_gl_context), wsi.render_surface_scale);
|
2019-02-15 11:59:50 +10:00
|
|
|
ProgramShaderCache::Init();
|
2015-12-20 21:49:49 -05:00
|
|
|
g_vertex_manager = std::make_unique<VertexManager>();
|
2019-02-15 11:59:50 +10:00
|
|
|
g_shader_cache = std::make_unique<VideoCommon::ShaderCache>();
|
|
|
|
g_framebuffer_manager = std::make_unique<FramebufferManager>();
|
2015-02-21 16:58:53 -06:00
|
|
|
g_perf_query = GetPerfQuery();
|
2019-02-15 11:59:50 +10:00
|
|
|
g_texture_cache = std::make_unique<TextureCacheBase>();
|
2015-12-20 21:49:49 -05:00
|
|
|
g_sampler_cache = std::make_unique<SamplerCache>();
|
2019-02-15 11:59:50 +10:00
|
|
|
BoundingBox::Init();
|
|
|
|
|
|
|
|
if (!g_vertex_manager->Initialize() || !g_shader_cache->Initialize() ||
|
|
|
|
!g_renderer->Initialize() || !g_framebuffer_manager->Initialize() ||
|
|
|
|
!g_texture_cache->Initialize())
|
|
|
|
{
|
2020-12-02 13:17:27 -05:00
|
|
|
PanicAlertFmtT("Failed to initialize renderer classes");
|
2019-02-15 11:59:50 +10:00
|
|
|
Shutdown();
|
2018-02-09 20:52:25 +10:00
|
|
|
return false;
|
2019-02-15 11:59:50 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
g_shader_cache->InitializeShaderCache();
|
|
|
|
return true;
|
2010-09-30 15:24:34 +00:00
|
|
|
}
|
|
|
|
|
2011-01-31 01:28:32 +00:00
|
|
|
void VideoBackend::Shutdown()
|
2010-09-30 15:24:34 +00:00
|
|
|
{
|
2018-02-25 01:15:35 +10:00
|
|
|
g_shader_cache->Shutdown();
|
2018-01-26 15:09:07 +10:00
|
|
|
g_renderer->Shutdown();
|
2015-12-20 21:49:49 -05:00
|
|
|
BoundingBox::Shutdown();
|
|
|
|
g_sampler_cache.reset();
|
|
|
|
g_texture_cache.reset();
|
|
|
|
g_perf_query.reset();
|
|
|
|
g_vertex_manager.reset();
|
2019-02-15 11:59:50 +10:00
|
|
|
g_framebuffer_manager.reset();
|
|
|
|
g_shader_cache.reset();
|
|
|
|
ProgramShaderCache::Shutdown();
|
2015-12-20 21:49:49 -05:00
|
|
|
g_renderer.reset();
|
2018-01-26 15:09:07 +10:00
|
|
|
ShutdownShared();
|
2010-09-30 15:24:34 +00:00
|
|
|
}
|
2018-10-03 23:02:45 +10:00
|
|
|
} // namespace OGL
|