2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2011 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2011-11-30 21:00:21 -06:00
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2011-11-30 21:00:21 -06:00
|
|
|
|
2017-09-08 19:42:56 +10:00
|
|
|
#include <atomic>
|
2017-07-20 15:25:29 +10:00
|
|
|
#include <memory>
|
2018-02-24 22:14:31 +10:00
|
|
|
#include <mutex>
|
2019-07-26 17:48:33 -04:00
|
|
|
#include <string_view>
|
2017-09-08 19:42:56 +10:00
|
|
|
#include <unordered_map>
|
2016-01-01 22:28:42 -05:00
|
|
|
|
2015-09-19 04:40:00 +12:00
|
|
|
#include "Common/GL/GLUtil.h"
|
2018-02-25 17:56:09 +10:00
|
|
|
#include "VideoCommon/AsyncShaderCompiler.h"
|
2017-07-20 15:25:29 +10:00
|
|
|
|
2011-12-26 02:58:52 -05:00
|
|
|
namespace OGL
|
2011-12-26 00:15:54 -05:00
|
|
|
{
|
2017-09-08 19:42:56 +10:00
|
|
|
class OGLShader;
|
2017-07-20 15:25:29 +10:00
|
|
|
class GLVertexFormat;
|
2017-09-08 19:42:56 +10:00
|
|
|
class StreamBuffer;
|
2017-07-20 15:25:29 +10:00
|
|
|
|
2013-02-13 13:12:19 +01:00
|
|
|
struct SHADER
|
|
|
|
{
|
|
|
|
void Destroy()
|
|
|
|
{
|
2017-07-20 15:25:29 +10:00
|
|
|
DestroyShaders();
|
|
|
|
if (glprogid)
|
|
|
|
{
|
|
|
|
glDeleteProgram(glprogid);
|
|
|
|
glprogid = 0;
|
|
|
|
}
|
2013-02-13 13:12:19 +01:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-07-20 15:25:29 +10:00
|
|
|
GLuint vsid = 0;
|
|
|
|
GLuint gsid = 0;
|
|
|
|
GLuint psid = 0;
|
|
|
|
GLuint glprogid = 0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-02-13 13:12:19 +01:00
|
|
|
void SetProgramVariables();
|
2016-11-27 18:14:56 +10:00
|
|
|
void SetProgramBindings(bool is_compute);
|
2017-04-22 23:44:34 -05:00
|
|
|
void Bind() const;
|
2017-07-20 15:25:29 +10:00
|
|
|
void DestroyShaders();
|
2013-02-13 13:12:19 +01:00
|
|
|
};
|
2011-12-26 02:58:52 -05:00
|
|
|
|
2017-09-08 19:42:56 +10:00
|
|
|
struct PipelineProgramKey
|
|
|
|
{
|
2019-02-15 11:59:50 +10:00
|
|
|
u64 vertex_shader_id;
|
|
|
|
u64 geometry_shader_id;
|
|
|
|
u64 pixel_shader_id;
|
2017-09-08 19:42:56 +10:00
|
|
|
|
|
|
|
bool operator==(const PipelineProgramKey& rhs) const;
|
|
|
|
bool operator<(const PipelineProgramKey& rhs) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct PipelineProgramKeyHash
|
|
|
|
{
|
|
|
|
std::size_t operator()(const PipelineProgramKey& key) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct PipelineProgram
|
|
|
|
{
|
2021-09-03 21:43:19 -07:00
|
|
|
PipelineProgramKey key{};
|
2017-09-08 19:42:56 +10:00
|
|
|
SHADER shader;
|
|
|
|
std::atomic_size_t reference_count{1};
|
2019-04-15 23:38:10 +10:00
|
|
|
bool binary_retrieved = false;
|
2017-09-08 19:42:56 +10:00
|
|
|
};
|
|
|
|
|
2011-12-26 02:58:52 -05:00
|
|
|
class ProgramShaderCache
|
2011-11-30 21:00:21 -06:00
|
|
|
{
|
|
|
|
public:
|
2017-07-20 15:25:29 +10:00
|
|
|
static void BindVertexFormat(const GLVertexFormat* vertex_format);
|
2022-11-15 00:38:24 -06:00
|
|
|
static void ReBindVertexFormat();
|
2018-11-27 17:16:53 +10:00
|
|
|
static bool IsValidVertexFormatBound();
|
2017-07-20 15:25:29 +10:00
|
|
|
static void InvalidateVertexFormat();
|
2019-03-05 23:02:40 +10:00
|
|
|
static void InvalidateVertexFormatIfBound(GLuint vao);
|
2017-09-08 19:42:56 +10:00
|
|
|
static void InvalidateLastProgram();
|
2013-10-29 01:23:17 -04:00
|
|
|
|
2019-07-26 17:48:33 -04:00
|
|
|
static bool CompileComputeShader(SHADER& shader, std::string_view code);
|
|
|
|
static GLuint CompileSingleShader(GLenum type, std::string_view code);
|
|
|
|
static bool CheckShaderCompileResult(GLuint id, GLenum type, std::string_view code);
|
|
|
|
static bool CheckProgramLinkResult(GLuint id, std::string_view vcode, std::string_view pcode,
|
|
|
|
std::string_view gcode);
|
2017-09-08 19:42:56 +10:00
|
|
|
static StreamBuffer* GetUniformBuffer();
|
|
|
|
static u32 GetUniformBufferAlignment();
|
2013-01-14 13:58:11 +01:00
|
|
|
static void UploadConstants();
|
2018-11-27 17:16:53 +10:00
|
|
|
static void UploadConstants(const void* data, u32 data_size);
|
2011-12-26 02:58:52 -05:00
|
|
|
|
2014-08-15 14:09:53 -04:00
|
|
|
static void Init();
|
|
|
|
static void Shutdown();
|
|
|
|
static void CreateHeader();
|
2011-12-26 02:58:52 -05:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
// This counter increments with each shader object allocated, in order to give it a unique ID.
|
|
|
|
// Since the shaders can be destroyed after a pipeline is created, we can't use the shader pointer
|
|
|
|
// as a key for GL programs. For the same reason, we can't use the GL objects either. This ID is
|
|
|
|
// guaranteed to be unique for the emulation session, even if the memory allocator or GL driver
|
|
|
|
// re-uses pointers, therefore we won't have any collisions where the shaders attached to a
|
|
|
|
// pipeline do not match the pipeline configuration.
|
|
|
|
static u64 GenerateShaderID();
|
|
|
|
|
2019-04-15 23:38:10 +10:00
|
|
|
static PipelineProgram* GetPipelineProgram(const GLVertexFormat* vertex_format,
|
|
|
|
const OGLShader* vertex_shader,
|
|
|
|
const OGLShader* geometry_shader,
|
|
|
|
const OGLShader* pixel_shader, const void* cache_data,
|
|
|
|
size_t cache_data_size);
|
|
|
|
static void ReleasePipelineProgram(PipelineProgram* prog);
|
2017-09-08 19:42:56 +10:00
|
|
|
|
2011-12-26 02:58:52 -05:00
|
|
|
private:
|
2019-07-26 18:43:41 -04:00
|
|
|
using PipelineProgramMap =
|
|
|
|
std::unordered_map<PipelineProgramKey, std::unique_ptr<PipelineProgram>,
|
|
|
|
PipelineProgramKeyHash>;
|
2017-07-20 15:25:29 +10:00
|
|
|
|
2018-01-21 00:59:10 +10:00
|
|
|
static void CreateAttributelessVAO();
|
2017-06-24 19:20:34 +10:00
|
|
|
|
2018-02-25 17:56:09 +10:00
|
|
|
static PipelineProgramMap s_pipeline_programs;
|
|
|
|
static std::mutex s_pipeline_program_lock;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-01-14 13:58:11 +01:00
|
|
|
static u32 s_ubo_buffer_size;
|
2013-10-07 17:19:47 +02:00
|
|
|
static s32 s_ubo_align;
|
2018-01-21 00:59:10 +10:00
|
|
|
|
|
|
|
static GLuint s_attributeless_VBO;
|
|
|
|
static GLuint s_attributeless_VAO;
|
|
|
|
static GLuint s_last_VAO;
|
2011-11-30 21:00:21 -06:00
|
|
|
};
|
|
|
|
|
2018-02-25 17:56:09 +10:00
|
|
|
class SharedContextAsyncShaderCompiler : public VideoCommon::AsyncShaderCompiler
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
bool WorkerThreadInitMainThread(void** param) override;
|
|
|
|
bool WorkerThreadInitWorkerThread(void* param) override;
|
|
|
|
void WorkerThreadExit(void* param) override;
|
|
|
|
};
|
|
|
|
|
2011-11-30 21:00:21 -06:00
|
|
|
} // namespace OGL
|