forked from eden-emu/eden
		
	shader: Remove old shader management
This commit is contained in:
		
							parent
							
								
									58914796c0
								
							
						
					
					
						commit
						c67d64365a
					
				
					 83 changed files with 57 additions and 19625 deletions
				
			
		
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -1,29 +0,0 @@ | |||
| // Copyright 2020 yuzu Emulator Project
 | ||||
| // Licensed under GPLv2 or any later version
 | ||||
| // Refer to the license.txt file included.
 | ||||
| 
 | ||||
| #pragma once | ||||
| 
 | ||||
| #include <string> | ||||
| #include <string_view> | ||||
| 
 | ||||
| #include "common/common_types.h" | ||||
| 
 | ||||
| namespace Tegra::Engines { | ||||
| enum class ShaderType : u32; | ||||
| } | ||||
| 
 | ||||
| namespace VideoCommon::Shader { | ||||
| class ShaderIR; | ||||
| class Registry; | ||||
| } // namespace VideoCommon::Shader
 | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| class Device; | ||||
| 
 | ||||
| std::string DecompileAssemblyShader(const Device& device, const VideoCommon::Shader::ShaderIR& ir, | ||||
|                                     const VideoCommon::Shader::Registry& registry, | ||||
|                                     Tegra::Engines::ShaderType stage, std::string_view identifier); | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
|  | @ -54,40 +54,6 @@ namespace { | |||
| 
 | ||||
| constexpr size_t NUM_SUPPORTED_VERTEX_ATTRIBUTES = 16; | ||||
| 
 | ||||
| struct TextureHandle { | ||||
|     constexpr TextureHandle(u32 data, bool via_header_index) { | ||||
|         const Tegra::Texture::TextureHandle handle{data}; | ||||
|         image = handle.tic_id; | ||||
|         sampler = via_header_index ? image : handle.tsc_id.Value(); | ||||
|     } | ||||
| 
 | ||||
|     u32 image; | ||||
|     u32 sampler; | ||||
| }; | ||||
| 
 | ||||
| template <typename Engine, typename Entry> | ||||
| TextureHandle GetTextureInfo(const Engine& engine, bool via_header_index, const Entry& entry, | ||||
|                              ShaderType shader_type, size_t index = 0) { | ||||
|     if constexpr (std::is_same_v<Entry, SamplerEntry>) { | ||||
|         if (entry.is_separated) { | ||||
|             const u32 buffer_1 = entry.buffer; | ||||
|             const u32 buffer_2 = entry.secondary_buffer; | ||||
|             const u32 offset_1 = entry.offset; | ||||
|             const u32 offset_2 = entry.secondary_offset; | ||||
|             const u32 handle_1 = engine.AccessConstBuffer32(shader_type, buffer_1, offset_1); | ||||
|             const u32 handle_2 = engine.AccessConstBuffer32(shader_type, buffer_2, offset_2); | ||||
|             return TextureHandle(handle_1 | handle_2, via_header_index); | ||||
|         } | ||||
|     } | ||||
|     if (entry.is_bindless) { | ||||
|         const u32 raw = engine.AccessConstBuffer32(shader_type, entry.buffer, entry.offset); | ||||
|         return TextureHandle(raw, via_header_index); | ||||
|     } | ||||
|     const u32 buffer = engine.GetBoundBuffer(); | ||||
|     const u64 offset = (entry.offset + index) * sizeof(u32); | ||||
|     return TextureHandle(engine.AccessConstBuffer32(shader_type, buffer, offset), via_header_index); | ||||
| } | ||||
| 
 | ||||
| /// Translates hardware transform feedback indices
 | ||||
| /// @param location Hardware location
 | ||||
| /// @return Pair of ARB_transform_feedback3 token stream first and third arguments
 | ||||
|  | @ -119,44 +85,6 @@ std::pair<GLint, GLint> TransformFeedbackEnum(u8 location) { | |||
| void oglEnable(GLenum cap, bool state) { | ||||
|     (state ? glEnable : glDisable)(cap); | ||||
| } | ||||
| 
 | ||||
| ImageViewType ImageViewTypeFromEntry(const SamplerEntry& entry) { | ||||
|     if (entry.is_buffer) { | ||||
|         return ImageViewType::Buffer; | ||||
|     } | ||||
|     switch (entry.type) { | ||||
|     case Tegra::Shader::TextureType::Texture1D: | ||||
|         return entry.is_array ? ImageViewType::e1DArray : ImageViewType::e1D; | ||||
|     case Tegra::Shader::TextureType::Texture2D: | ||||
|         return entry.is_array ? ImageViewType::e2DArray : ImageViewType::e2D; | ||||
|     case Tegra::Shader::TextureType::Texture3D: | ||||
|         return ImageViewType::e3D; | ||||
|     case Tegra::Shader::TextureType::TextureCube: | ||||
|         return entry.is_array ? ImageViewType::CubeArray : ImageViewType::Cube; | ||||
|     } | ||||
|     UNREACHABLE(); | ||||
|     return ImageViewType::e2D; | ||||
| } | ||||
| 
 | ||||
| ImageViewType ImageViewTypeFromEntry(const ImageEntry& entry) { | ||||
|     switch (entry.type) { | ||||
|     case Tegra::Shader::ImageType::Texture1D: | ||||
|         return ImageViewType::e1D; | ||||
|     case Tegra::Shader::ImageType::Texture1DArray: | ||||
|         return ImageViewType::e1DArray; | ||||
|     case Tegra::Shader::ImageType::Texture2D: | ||||
|         return ImageViewType::e2D; | ||||
|     case Tegra::Shader::ImageType::Texture2DArray: | ||||
|         return ImageViewType::e2DArray; | ||||
|     case Tegra::Shader::ImageType::Texture3D: | ||||
|         return ImageViewType::e3D; | ||||
|     case Tegra::Shader::ImageType::TextureBuffer: | ||||
|         return ImageViewType::Buffer; | ||||
|     } | ||||
|     UNREACHABLE(); | ||||
|     return ImageViewType::e2D; | ||||
| } | ||||
| 
 | ||||
| } // Anonymous namespace
 | ||||
| 
 | ||||
| RasterizerOpenGL::RasterizerOpenGL(Core::Frontend::EmuWindow& emu_window_, Tegra::GPU& gpu_, | ||||
|  | @ -172,12 +100,7 @@ RasterizerOpenGL::RasterizerOpenGL(Core::Frontend::EmuWindow& emu_window_, Tegra | |||
|       buffer_cache(*this, maxwell3d, kepler_compute, gpu_memory, cpu_memory_, buffer_cache_runtime), | ||||
|       shader_cache(*this, emu_window_, gpu, maxwell3d, kepler_compute, gpu_memory, device), | ||||
|       query_cache(*this, maxwell3d, gpu_memory), accelerate_dma(buffer_cache), | ||||
|       fence_manager(*this, gpu, texture_cache, buffer_cache, query_cache), | ||||
|       async_shaders(emu_window_) { | ||||
|     if (device.UseAsynchronousShaders()) { | ||||
|         async_shaders.AllocateWorkers(); | ||||
|     } | ||||
| } | ||||
|       fence_manager(*this, gpu, texture_cache, buffer_cache, query_cache) {} | ||||
| 
 | ||||
| RasterizerOpenGL::~RasterizerOpenGL() = default; | ||||
| 
 | ||||
|  | @ -244,117 +167,8 @@ void RasterizerOpenGL::SyncVertexInstances() { | |||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::SetupShaders(bool is_indexed) { | ||||
|     u32 clip_distances = 0; | ||||
| 
 | ||||
|     std::array<Shader*, Maxwell::MaxShaderStage> shaders{}; | ||||
|     image_view_indices.clear(); | ||||
|     sampler_handles.clear(); | ||||
| 
 | ||||
|     texture_cache.SynchronizeGraphicsDescriptors(); | ||||
| 
 | ||||
|     for (std::size_t index = 0; index < Maxwell::MaxShaderProgram; ++index) { | ||||
|         const auto& shader_config = maxwell3d.regs.shader_config[index]; | ||||
|         const auto program{static_cast<Maxwell::ShaderProgram>(index)}; | ||||
| 
 | ||||
|         // Skip stages that are not enabled
 | ||||
|         if (!maxwell3d.regs.IsShaderConfigEnabled(index)) { | ||||
|             switch (program) { | ||||
|             case Maxwell::ShaderProgram::Geometry: | ||||
|                 program_manager.UseGeometryShader(0); | ||||
|                 break; | ||||
|             case Maxwell::ShaderProgram::Fragment: | ||||
|                 program_manager.UseFragmentShader(0); | ||||
|                 break; | ||||
|             default: | ||||
|                 break; | ||||
|             } | ||||
|             continue; | ||||
|         } | ||||
|         // Currently this stages are not supported in the OpenGL backend.
 | ||||
|         // TODO(Blinkhawk): Port tesselation shaders from Vulkan to OpenGL
 | ||||
|         if (program == Maxwell::ShaderProgram::TesselationControl || | ||||
|             program == Maxwell::ShaderProgram::TesselationEval) { | ||||
|             continue; | ||||
|         } | ||||
| 
 | ||||
|         Shader* const shader = shader_cache.GetStageProgram(program, async_shaders); | ||||
|         const GLuint program_handle = shader->IsBuilt() ? shader->GetHandle() : 0; | ||||
|         switch (program) { | ||||
|         case Maxwell::ShaderProgram::VertexA: | ||||
|         case Maxwell::ShaderProgram::VertexB: | ||||
|             program_manager.UseVertexShader(program_handle); | ||||
|             break; | ||||
|         case Maxwell::ShaderProgram::Geometry: | ||||
|             program_manager.UseGeometryShader(program_handle); | ||||
|             break; | ||||
|         case Maxwell::ShaderProgram::Fragment: | ||||
|             program_manager.UseFragmentShader(program_handle); | ||||
|             break; | ||||
|         default: | ||||
|             UNIMPLEMENTED_MSG("Unimplemented shader index={}, enable={}, offset=0x{:08X}", index, | ||||
|                               shader_config.enable.Value(), shader_config.offset); | ||||
|             break; | ||||
|         } | ||||
| 
 | ||||
|         // Stage indices are 0 - 5
 | ||||
|         const size_t stage = index == 0 ? 0 : index - 1; | ||||
|         shaders[stage] = shader; | ||||
| 
 | ||||
|         SetupDrawTextures(shader, stage); | ||||
|         SetupDrawImages(shader, stage); | ||||
| 
 | ||||
|         buffer_cache.SetEnabledUniformBuffers(stage, shader->GetEntries().enabled_uniform_buffers); | ||||
| 
 | ||||
|         buffer_cache.UnbindGraphicsStorageBuffers(stage); | ||||
|         u32 ssbo_index = 0; | ||||
|         for (const auto& buffer : shader->GetEntries().global_memory_entries) { | ||||
|             buffer_cache.BindGraphicsStorageBuffer(stage, ssbo_index, buffer.cbuf_index, | ||||
|                                                    buffer.cbuf_offset, buffer.is_written); | ||||
|             ++ssbo_index; | ||||
|         } | ||||
| 
 | ||||
|         // Workaround for Intel drivers.
 | ||||
|         // When a clip distance is enabled but not set in the shader it crops parts of the screen
 | ||||
|         // (sometimes it's half the screen, sometimes three quarters). To avoid this, enable the
 | ||||
|         // clip distances only when it's written by a shader stage.
 | ||||
|         clip_distances |= shader->GetEntries().clip_distances; | ||||
| 
 | ||||
|         // When VertexA is enabled, we have dual vertex shaders
 | ||||
|         if (program == Maxwell::ShaderProgram::VertexA) { | ||||
|             // VertexB was combined with VertexA, so we skip the VertexB iteration
 | ||||
|             ++index; | ||||
|         } | ||||
|     } | ||||
|     SyncClipEnabled(clip_distances); | ||||
|     maxwell3d.dirty.flags[Dirty::Shaders] = false; | ||||
| 
 | ||||
|     buffer_cache.UpdateGraphicsBuffers(is_indexed); | ||||
| 
 | ||||
|     const std::span indices_span(image_view_indices.data(), image_view_indices.size()); | ||||
|     texture_cache.FillGraphicsImageViews(indices_span, image_view_ids); | ||||
| 
 | ||||
|     buffer_cache.BindHostGeometryBuffers(is_indexed); | ||||
| 
 | ||||
|     size_t image_view_index = 0; | ||||
|     size_t texture_index = 0; | ||||
|     size_t image_index = 0; | ||||
|     for (size_t stage = 0; stage < Maxwell::MaxShaderStage; ++stage) { | ||||
|         const Shader* const shader = shaders[stage]; | ||||
|         if (!shader) { | ||||
|             continue; | ||||
|         } | ||||
|         buffer_cache.BindHostStageBuffers(stage); | ||||
|         const auto& base = device.GetBaseBindings(stage); | ||||
|         BindTextures(shader->GetEntries(), base.sampler, base.image, image_view_index, | ||||
|                      texture_index, image_index); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::LoadDiskResources(u64 title_id, std::stop_token stop_loading, | ||||
|                                          const VideoCore::DiskResourceLoadCallback& callback) { | ||||
|     shader_cache.LoadDiskCache(title_id, stop_loading, callback); | ||||
| } | ||||
|                                          const VideoCore::DiskResourceLoadCallback& callback) {} | ||||
| 
 | ||||
| void RasterizerOpenGL::Clear() { | ||||
|     MICROPROFILE_SCOPE(OpenGL_Clears); | ||||
|  | @ -434,7 +248,6 @@ void RasterizerOpenGL::Draw(bool is_indexed, bool is_instanced) { | |||
| 
 | ||||
|     // Setup shaders and their used resources.
 | ||||
|     std::scoped_lock lock{buffer_cache.mutex, texture_cache.mutex}; | ||||
|     SetupShaders(is_indexed); | ||||
| 
 | ||||
|     texture_cache.UpdateRenderTargets(false); | ||||
|     state_tracker.BindFramebuffer(texture_cache.GetFramebuffer()->Handle()); | ||||
|  | @ -488,27 +301,8 @@ void RasterizerOpenGL::Draw(bool is_indexed, bool is_instanced) { | |||
|     gpu.TickWork(); | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::DispatchCompute(GPUVAddr code_addr) { | ||||
|     Shader* const kernel = shader_cache.GetComputeKernel(code_addr); | ||||
| 
 | ||||
|     std::scoped_lock lock{buffer_cache.mutex, texture_cache.mutex}; | ||||
|     BindComputeTextures(kernel); | ||||
| 
 | ||||
|     const auto& entries = kernel->GetEntries(); | ||||
|     buffer_cache.SetEnabledComputeUniformBuffers(entries.enabled_uniform_buffers); | ||||
|     buffer_cache.UnbindComputeStorageBuffers(); | ||||
|     u32 ssbo_index = 0; | ||||
|     for (const auto& buffer : entries.global_memory_entries) { | ||||
|         buffer_cache.BindComputeStorageBuffer(ssbo_index, buffer.cbuf_index, buffer.cbuf_offset, | ||||
|                                               buffer.is_written); | ||||
|         ++ssbo_index; | ||||
|     } | ||||
|     buffer_cache.UpdateComputeBuffers(); | ||||
|     buffer_cache.BindHostComputeBuffers(); | ||||
| 
 | ||||
|     const auto& launch_desc = kepler_compute.launch_description; | ||||
|     glDispatchCompute(launch_desc.grid_dim_x, launch_desc.grid_dim_y, launch_desc.grid_dim_z); | ||||
|     ++num_queued_commands; | ||||
| void RasterizerOpenGL::DispatchCompute() { | ||||
|     UNREACHABLE_MSG("Not implemented"); | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::ResetCounter(VideoCore::QueryType type) { | ||||
|  | @ -726,106 +520,6 @@ bool RasterizerOpenGL::AccelerateDisplay(const Tegra::FramebufferConfig& config, | |||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::BindComputeTextures(Shader* kernel) { | ||||
|     image_view_indices.clear(); | ||||
|     sampler_handles.clear(); | ||||
| 
 | ||||
|     texture_cache.SynchronizeComputeDescriptors(); | ||||
| 
 | ||||
|     SetupComputeTextures(kernel); | ||||
|     SetupComputeImages(kernel); | ||||
| 
 | ||||
|     const std::span indices_span(image_view_indices.data(), image_view_indices.size()); | ||||
|     texture_cache.FillComputeImageViews(indices_span, image_view_ids); | ||||
| 
 | ||||
|     program_manager.BindCompute(kernel->GetHandle()); | ||||
|     size_t image_view_index = 0; | ||||
|     size_t texture_index = 0; | ||||
|     size_t image_index = 0; | ||||
|     BindTextures(kernel->GetEntries(), 0, 0, image_view_index, texture_index, image_index); | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::BindTextures(const ShaderEntries& entries, GLuint base_texture, | ||||
|                                     GLuint base_image, size_t& image_view_index, | ||||
|                                     size_t& texture_index, size_t& image_index) { | ||||
|     const GLuint* const samplers = sampler_handles.data() + texture_index; | ||||
|     const GLuint* const textures = texture_handles.data() + texture_index; | ||||
|     const GLuint* const images = image_handles.data() + image_index; | ||||
| 
 | ||||
|     const size_t num_samplers = entries.samplers.size(); | ||||
|     for (const auto& sampler : entries.samplers) { | ||||
|         for (size_t i = 0; i < sampler.size; ++i) { | ||||
|             const ImageViewId image_view_id = image_view_ids[image_view_index++]; | ||||
|             const ImageView& image_view = texture_cache.GetImageView(image_view_id); | ||||
|             const GLuint handle = image_view.Handle(ImageViewTypeFromEntry(sampler)); | ||||
|             texture_handles[texture_index++] = handle; | ||||
|         } | ||||
|     } | ||||
|     const size_t num_images = entries.images.size(); | ||||
|     for (size_t unit = 0; unit < num_images; ++unit) { | ||||
|         // TODO: Mark as modified
 | ||||
|         const ImageViewId image_view_id = image_view_ids[image_view_index++]; | ||||
|         const ImageView& image_view = texture_cache.GetImageView(image_view_id); | ||||
|         const GLuint handle = image_view.Handle(ImageViewTypeFromEntry(entries.images[unit])); | ||||
|         image_handles[image_index] = handle; | ||||
|         ++image_index; | ||||
|     } | ||||
|     if (num_samplers > 0) { | ||||
|         glBindSamplers(base_texture, static_cast<GLsizei>(num_samplers), samplers); | ||||
|         glBindTextures(base_texture, static_cast<GLsizei>(num_samplers), textures); | ||||
|     } | ||||
|     if (num_images > 0) { | ||||
|         glBindImageTextures(base_image, static_cast<GLsizei>(num_images), images); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::SetupDrawTextures(const Shader* shader, size_t stage_index) { | ||||
|     const bool via_header_index = | ||||
|         maxwell3d.regs.sampler_index == Maxwell::SamplerIndex::ViaHeaderIndex; | ||||
|     for (const auto& entry : shader->GetEntries().samplers) { | ||||
|         const auto shader_type = static_cast<ShaderType>(stage_index); | ||||
|         for (size_t index = 0; index < entry.size; ++index) { | ||||
|             const auto handle = | ||||
|                 GetTextureInfo(maxwell3d, via_header_index, entry, shader_type, index); | ||||
|             const Sampler* const sampler = texture_cache.GetGraphicsSampler(handle.sampler); | ||||
|             sampler_handles.push_back(sampler->Handle()); | ||||
|             image_view_indices.push_back(handle.image); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::SetupComputeTextures(const Shader* kernel) { | ||||
|     const bool via_header_index = kepler_compute.launch_description.linked_tsc; | ||||
|     for (const auto& entry : kernel->GetEntries().samplers) { | ||||
|         for (size_t i = 0; i < entry.size; ++i) { | ||||
|             const auto handle = | ||||
|                 GetTextureInfo(kepler_compute, via_header_index, entry, ShaderType::Compute, i); | ||||
|             const Sampler* const sampler = texture_cache.GetComputeSampler(handle.sampler); | ||||
|             sampler_handles.push_back(sampler->Handle()); | ||||
|             image_view_indices.push_back(handle.image); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::SetupDrawImages(const Shader* shader, size_t stage_index) { | ||||
|     const bool via_header_index = | ||||
|         maxwell3d.regs.sampler_index == Maxwell::SamplerIndex::ViaHeaderIndex; | ||||
|     for (const auto& entry : shader->GetEntries().images) { | ||||
|         const auto shader_type = static_cast<ShaderType>(stage_index); | ||||
|         const auto handle = GetTextureInfo(maxwell3d, via_header_index, entry, shader_type); | ||||
|         image_view_indices.push_back(handle.image); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::SetupComputeImages(const Shader* shader) { | ||||
|     const bool via_header_index = kepler_compute.launch_description.linked_tsc; | ||||
|     for (const auto& entry : shader->GetEntries().images) { | ||||
|         const auto handle = | ||||
|             GetTextureInfo(kepler_compute, via_header_index, entry, ShaderType::Compute); | ||||
|         image_view_indices.push_back(handle.image); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RasterizerOpenGL::SyncState() { | ||||
|     SyncViewport(); | ||||
|     SyncRasterizeEnable(); | ||||
|  |  | |||
|  | @ -28,11 +28,9 @@ | |||
| #include "video_core/renderer_opengl/gl_query_cache.h" | ||||
| #include "video_core/renderer_opengl/gl_resource_manager.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_cache.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_decompiler.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_manager.h" | ||||
| #include "video_core/renderer_opengl/gl_state_tracker.h" | ||||
| #include "video_core/renderer_opengl/gl_texture_cache.h" | ||||
| #include "video_core/shader/async_shaders.h" | ||||
| #include "video_core/textures/texture.h" | ||||
| 
 | ||||
| namespace Core::Memory { | ||||
|  | @ -81,7 +79,7 @@ public: | |||
| 
 | ||||
|     void Draw(bool is_indexed, bool is_instanced) override; | ||||
|     void Clear() override; | ||||
|     void DispatchCompute(GPUVAddr code_addr) override; | ||||
|     void DispatchCompute() override; | ||||
|     void ResetCounter(VideoCore::QueryType type) override; | ||||
|     void Query(GPUVAddr gpu_addr, VideoCore::QueryType type, std::optional<u64> timestamp) override; | ||||
|     void BindGraphicsUniformBuffer(size_t stage, u32 index, GPUVAddr gpu_addr, u32 size) override; | ||||
|  | @ -118,36 +116,11 @@ public: | |||
|         return num_queued_commands > 0; | ||||
|     } | ||||
| 
 | ||||
|     VideoCommon::Shader::AsyncShaders& GetAsyncShaders() { | ||||
|         return async_shaders; | ||||
|     } | ||||
| 
 | ||||
|     const VideoCommon::Shader::AsyncShaders& GetAsyncShaders() const { | ||||
|         return async_shaders; | ||||
|     } | ||||
| 
 | ||||
| private: | ||||
|     static constexpr size_t MAX_TEXTURES = 192; | ||||
|     static constexpr size_t MAX_IMAGES = 48; | ||||
|     static constexpr size_t MAX_IMAGE_VIEWS = MAX_TEXTURES + MAX_IMAGES; | ||||
| 
 | ||||
|     void BindComputeTextures(Shader* kernel); | ||||
| 
 | ||||
|     void BindTextures(const ShaderEntries& entries, GLuint base_texture, GLuint base_image, | ||||
|                       size_t& image_view_index, size_t& texture_index, size_t& image_index); | ||||
| 
 | ||||
|     /// Configures the current textures to use for the draw command.
 | ||||
|     void SetupDrawTextures(const Shader* shader, size_t stage_index); | ||||
| 
 | ||||
|     /// Configures the textures used in a compute shader.
 | ||||
|     void SetupComputeTextures(const Shader* kernel); | ||||
| 
 | ||||
|     /// Configures images in a graphics shader.
 | ||||
|     void SetupDrawImages(const Shader* shader, size_t stage_index); | ||||
| 
 | ||||
|     /// Configures images in a compute shader.
 | ||||
|     void SetupComputeImages(const Shader* shader); | ||||
| 
 | ||||
|     /// Syncs state to match guest's
 | ||||
|     void SyncState(); | ||||
| 
 | ||||
|  | @ -230,8 +203,6 @@ private: | |||
|     /// End a transform feedback
 | ||||
|     void EndTransformFeedback(); | ||||
| 
 | ||||
|     void SetupShaders(bool is_indexed); | ||||
| 
 | ||||
|     Tegra::GPU& gpu; | ||||
|     Tegra::Engines::Maxwell3D& maxwell3d; | ||||
|     Tegra::Engines::KeplerCompute& kepler_compute; | ||||
|  | @ -251,8 +222,6 @@ private: | |||
|     AccelerateDMA accelerate_dma; | ||||
|     FenceManagerOpenGL fence_manager; | ||||
| 
 | ||||
|     VideoCommon::Shader::AsyncShaders async_shaders; | ||||
| 
 | ||||
|     boost::container::static_vector<u32, MAX_IMAGE_VIEWS> image_view_indices; | ||||
|     std::array<ImageViewId, MAX_IMAGE_VIEWS> image_view_ids; | ||||
|     boost::container::static_vector<GLuint, MAX_TEXTURES> sampler_handles; | ||||
|  |  | |||
|  | @ -20,307 +20,19 @@ | |||
| #include "video_core/engines/maxwell_3d.h" | ||||
| #include "video_core/engines/shader_type.h" | ||||
| #include "video_core/memory_manager.h" | ||||
| #include "video_core/renderer_opengl/gl_arb_decompiler.h" | ||||
| #include "video_core/renderer_opengl/gl_rasterizer.h" | ||||
| #include "video_core/renderer_opengl/gl_resource_manager.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_cache.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_decompiler.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_disk_cache.h" | ||||
| #include "video_core/renderer_opengl/gl_state_tracker.h" | ||||
| #include "video_core/shader/memory_util.h" | ||||
| #include "video_core/shader/registry.h" | ||||
| #include "video_core/shader/shader_ir.h" | ||||
| #include "video_core/shader_cache.h" | ||||
| #include "video_core/shader_notify.h" | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| using Tegra::Engines::ShaderType; | ||||
| using VideoCommon::Shader::GetShaderAddress; | ||||
| using VideoCommon::Shader::GetShaderCode; | ||||
| using VideoCommon::Shader::GetUniqueIdentifier; | ||||
| using VideoCommon::Shader::KERNEL_MAIN_OFFSET; | ||||
| using VideoCommon::Shader::ProgramCode; | ||||
| using VideoCommon::Shader::Registry; | ||||
| using VideoCommon::Shader::ShaderIR; | ||||
| using VideoCommon::Shader::STAGE_MAIN_OFFSET; | ||||
| 
 | ||||
| namespace { | ||||
| 
 | ||||
| constexpr VideoCommon::Shader::CompilerSettings COMPILER_SETTINGS{}; | ||||
| 
 | ||||
| /// Gets the shader type from a Maxwell program type
 | ||||
| constexpr GLenum GetGLShaderType(ShaderType shader_type) { | ||||
|     switch (shader_type) { | ||||
|     case ShaderType::Vertex: | ||||
|         return GL_VERTEX_SHADER; | ||||
|     case ShaderType::Geometry: | ||||
|         return GL_GEOMETRY_SHADER; | ||||
|     case ShaderType::Fragment: | ||||
|         return GL_FRAGMENT_SHADER; | ||||
|     case ShaderType::Compute: | ||||
|         return GL_COMPUTE_SHADER; | ||||
|     default: | ||||
|         return GL_NONE; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| constexpr const char* GetShaderTypeName(ShaderType shader_type) { | ||||
|     switch (shader_type) { | ||||
|     case ShaderType::Vertex: | ||||
|         return "VS"; | ||||
|     case ShaderType::TesselationControl: | ||||
|         return "HS"; | ||||
|     case ShaderType::TesselationEval: | ||||
|         return "DS"; | ||||
|     case ShaderType::Geometry: | ||||
|         return "GS"; | ||||
|     case ShaderType::Fragment: | ||||
|         return "FS"; | ||||
|     case ShaderType::Compute: | ||||
|         return "CS"; | ||||
|     } | ||||
|     return "UNK"; | ||||
| } | ||||
| 
 | ||||
| constexpr ShaderType GetShaderType(Maxwell::ShaderProgram program_type) { | ||||
|     switch (program_type) { | ||||
|     case Maxwell::ShaderProgram::VertexA: | ||||
|     case Maxwell::ShaderProgram::VertexB: | ||||
|         return ShaderType::Vertex; | ||||
|     case Maxwell::ShaderProgram::TesselationControl: | ||||
|         return ShaderType::TesselationControl; | ||||
|     case Maxwell::ShaderProgram::TesselationEval: | ||||
|         return ShaderType::TesselationEval; | ||||
|     case Maxwell::ShaderProgram::Geometry: | ||||
|         return ShaderType::Geometry; | ||||
|     case Maxwell::ShaderProgram::Fragment: | ||||
|         return ShaderType::Fragment; | ||||
|     } | ||||
|     return {}; | ||||
| } | ||||
| 
 | ||||
| constexpr GLenum AssemblyEnum(ShaderType shader_type) { | ||||
|     switch (shader_type) { | ||||
|     case ShaderType::Vertex: | ||||
|         return GL_VERTEX_PROGRAM_NV; | ||||
|     case ShaderType::TesselationControl: | ||||
|         return GL_TESS_CONTROL_PROGRAM_NV; | ||||
|     case ShaderType::TesselationEval: | ||||
|         return GL_TESS_EVALUATION_PROGRAM_NV; | ||||
|     case ShaderType::Geometry: | ||||
|         return GL_GEOMETRY_PROGRAM_NV; | ||||
|     case ShaderType::Fragment: | ||||
|         return GL_FRAGMENT_PROGRAM_NV; | ||||
|     case ShaderType::Compute: | ||||
|         return GL_COMPUTE_PROGRAM_NV; | ||||
|     } | ||||
|     return {}; | ||||
| } | ||||
| 
 | ||||
| std::string MakeShaderID(u64 unique_identifier, ShaderType shader_type) { | ||||
|     return fmt::format("{}{:016X}", GetShaderTypeName(shader_type), unique_identifier); | ||||
| } | ||||
| 
 | ||||
| std::shared_ptr<Registry> MakeRegistry(const ShaderDiskCacheEntry& entry) { | ||||
|     const VideoCore::GuestDriverProfile guest_profile{entry.texture_handler_size}; | ||||
|     const VideoCommon::Shader::SerializedRegistryInfo info{guest_profile, entry.bound_buffer, | ||||
|                                                            entry.graphics_info, entry.compute_info}; | ||||
|     auto registry = std::make_shared<Registry>(entry.type, info); | ||||
|     for (const auto& [address, value] : entry.keys) { | ||||
|         const auto [buffer, offset] = address; | ||||
|         registry->InsertKey(buffer, offset, value); | ||||
|     } | ||||
|     for (const auto& [offset, sampler] : entry.bound_samplers) { | ||||
|         registry->InsertBoundSampler(offset, sampler); | ||||
|     } | ||||
|     for (const auto& [key, sampler] : entry.bindless_samplers) { | ||||
|         const auto [buffer, offset] = key; | ||||
|         registry->InsertBindlessSampler(buffer, offset, sampler); | ||||
|     } | ||||
|     return registry; | ||||
| } | ||||
| 
 | ||||
| std::unordered_set<GLenum> GetSupportedFormats() { | ||||
|     GLint num_formats; | ||||
|     glGetIntegerv(GL_NUM_PROGRAM_BINARY_FORMATS, &num_formats); | ||||
| 
 | ||||
|     std::vector<GLint> formats(num_formats); | ||||
|     glGetIntegerv(GL_PROGRAM_BINARY_FORMATS, formats.data()); | ||||
| 
 | ||||
|     std::unordered_set<GLenum> supported_formats; | ||||
|     for (const GLint format : formats) { | ||||
|         supported_formats.insert(static_cast<GLenum>(format)); | ||||
|     } | ||||
|     return supported_formats; | ||||
| } | ||||
| 
 | ||||
| } // Anonymous namespace
 | ||||
| 
 | ||||
| ProgramSharedPtr BuildShader(const Device& device, ShaderType shader_type, u64 unique_identifier, | ||||
|                              const ShaderIR& ir, const Registry& registry, bool hint_retrievable) { | ||||
|     if (device.UseDriverCache()) { | ||||
|         // Ignore hint retrievable if we are using the driver cache
 | ||||
|         hint_retrievable = false; | ||||
|     } | ||||
|     const std::string shader_id = MakeShaderID(unique_identifier, shader_type); | ||||
|     LOG_INFO(Render_OpenGL, "{}", shader_id); | ||||
| 
 | ||||
|     auto program = std::make_shared<ProgramHandle>(); | ||||
| 
 | ||||
|     if (device.UseAssemblyShaders()) { | ||||
|         const std::string arb = | ||||
|             DecompileAssemblyShader(device, ir, registry, shader_type, shader_id); | ||||
| 
 | ||||
|         GLuint& arb_prog = program->assembly_program.handle; | ||||
| 
 | ||||
| // Commented out functions signal OpenGL errors but are compatible with apitrace.
 | ||||
| // Use them only to capture and replay on apitrace.
 | ||||
| #if 0 | ||||
|         glGenProgramsNV(1, &arb_prog); | ||||
|         glLoadProgramNV(AssemblyEnum(shader_type), arb_prog, static_cast<GLsizei>(arb.size()), | ||||
|                         reinterpret_cast<const GLubyte*>(arb.data())); | ||||
| #else | ||||
|         glGenProgramsARB(1, &arb_prog); | ||||
|         glNamedProgramStringEXT(arb_prog, AssemblyEnum(shader_type), GL_PROGRAM_FORMAT_ASCII_ARB, | ||||
|                                 static_cast<GLsizei>(arb.size()), arb.data()); | ||||
| #endif | ||||
|         const auto err = reinterpret_cast<const char*>(glGetString(GL_PROGRAM_ERROR_STRING_NV)); | ||||
|         if (err && *err) { | ||||
|             LOG_CRITICAL(Render_OpenGL, "{}", err); | ||||
|             LOG_INFO(Render_OpenGL, "\n{}", arb); | ||||
|         } | ||||
|     } else { | ||||
|         const std::string glsl = DecompileShader(device, ir, registry, shader_type, shader_id); | ||||
|         OGLShader shader; | ||||
|         shader.Create(glsl.c_str(), GetGLShaderType(shader_type)); | ||||
| 
 | ||||
|         program->source_program.Create(true, hint_retrievable, shader.handle); | ||||
|     } | ||||
| 
 | ||||
|     return program; | ||||
| } | ||||
| 
 | ||||
| Shader::Shader(std::shared_ptr<Registry> registry_, ShaderEntries entries_, | ||||
|                ProgramSharedPtr program_, bool is_built_) | ||||
|     : registry{std::move(registry_)}, entries{std::move(entries_)}, program{std::move(program_)}, | ||||
|       is_built{is_built_} { | ||||
|     handle = program->assembly_program.handle; | ||||
|     if (handle == 0) { | ||||
|         handle = program->source_program.handle; | ||||
|     } | ||||
|     if (is_built) { | ||||
|         ASSERT(handle != 0); | ||||
|     } | ||||
| } | ||||
| Shader::Shader() = default; | ||||
| 
 | ||||
| Shader::~Shader() = default; | ||||
| 
 | ||||
| GLuint Shader::GetHandle() const { | ||||
|     DEBUG_ASSERT(registry->IsConsistent()); | ||||
|     return handle; | ||||
| } | ||||
| 
 | ||||
| bool Shader::IsBuilt() const { | ||||
|     return is_built; | ||||
| } | ||||
| 
 | ||||
| void Shader::AsyncOpenGLBuilt(OGLProgram new_program) { | ||||
|     program->source_program = std::move(new_program); | ||||
|     handle = program->source_program.handle; | ||||
|     is_built = true; | ||||
| } | ||||
| 
 | ||||
| void Shader::AsyncGLASMBuilt(OGLAssemblyProgram new_program) { | ||||
|     program->assembly_program = std::move(new_program); | ||||
|     handle = program->assembly_program.handle; | ||||
|     is_built = true; | ||||
| } | ||||
| 
 | ||||
| std::unique_ptr<Shader> Shader::CreateStageFromMemory( | ||||
|     const ShaderParameters& params, Maxwell::ShaderProgram program_type, ProgramCode code, | ||||
|     ProgramCode code_b, VideoCommon::Shader::AsyncShaders& async_shaders, VAddr cpu_addr) { | ||||
|     const auto shader_type = GetShaderType(program_type); | ||||
| 
 | ||||
|     auto& gpu = params.gpu; | ||||
|     gpu.ShaderNotify().MarkSharderBuilding(); | ||||
| 
 | ||||
|     auto registry = std::make_shared<Registry>(shader_type, gpu.Maxwell3D()); | ||||
|     if (!async_shaders.IsShaderAsync(gpu) || !params.device.UseAsynchronousShaders()) { | ||||
|         const ShaderIR ir(code, STAGE_MAIN_OFFSET, COMPILER_SETTINGS, *registry); | ||||
|         // TODO(Rodrigo): Handle VertexA shaders
 | ||||
|         // std::optional<ShaderIR> ir_b;
 | ||||
|         // if (!code_b.empty()) {
 | ||||
|         //     ir_b.emplace(code_b, STAGE_MAIN_OFFSET);
 | ||||
|         // }
 | ||||
|         auto program = | ||||
|             BuildShader(params.device, shader_type, params.unique_identifier, ir, *registry); | ||||
|         ShaderDiskCacheEntry entry; | ||||
|         entry.type = shader_type; | ||||
|         entry.code = std::move(code); | ||||
|         entry.code_b = std::move(code_b); | ||||
|         entry.unique_identifier = params.unique_identifier; | ||||
|         entry.bound_buffer = registry->GetBoundBuffer(); | ||||
|         entry.graphics_info = registry->GetGraphicsInfo(); | ||||
|         entry.keys = registry->GetKeys(); | ||||
|         entry.bound_samplers = registry->GetBoundSamplers(); | ||||
|         entry.bindless_samplers = registry->GetBindlessSamplers(); | ||||
|         params.disk_cache.SaveEntry(std::move(entry)); | ||||
| 
 | ||||
|         gpu.ShaderNotify().MarkShaderComplete(); | ||||
| 
 | ||||
|         return std::unique_ptr<Shader>(new Shader(std::move(registry), | ||||
|                                                   MakeEntries(params.device, ir, shader_type), | ||||
|                                                   std::move(program), true)); | ||||
|     } else { | ||||
|         // Required for entries
 | ||||
|         const ShaderIR ir(code, STAGE_MAIN_OFFSET, COMPILER_SETTINGS, *registry); | ||||
|         auto entries = MakeEntries(params.device, ir, shader_type); | ||||
| 
 | ||||
|         async_shaders.QueueOpenGLShader(params.device, shader_type, params.unique_identifier, | ||||
|                                         std::move(code), std::move(code_b), STAGE_MAIN_OFFSET, | ||||
|                                         COMPILER_SETTINGS, *registry, cpu_addr); | ||||
| 
 | ||||
|         auto program = std::make_shared<ProgramHandle>(); | ||||
|         return std::unique_ptr<Shader>( | ||||
|             new Shader(std::move(registry), std::move(entries), std::move(program), false)); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| std::unique_ptr<Shader> Shader::CreateKernelFromMemory(const ShaderParameters& params, | ||||
|                                                        ProgramCode code) { | ||||
|     auto& gpu = params.gpu; | ||||
|     gpu.ShaderNotify().MarkSharderBuilding(); | ||||
| 
 | ||||
|     auto registry = std::make_shared<Registry>(ShaderType::Compute, params.engine); | ||||
|     const ShaderIR ir(code, KERNEL_MAIN_OFFSET, COMPILER_SETTINGS, *registry); | ||||
|     const u64 uid = params.unique_identifier; | ||||
|     auto program = BuildShader(params.device, ShaderType::Compute, uid, ir, *registry); | ||||
| 
 | ||||
|     ShaderDiskCacheEntry entry; | ||||
|     entry.type = ShaderType::Compute; | ||||
|     entry.code = std::move(code); | ||||
|     entry.unique_identifier = uid; | ||||
|     entry.bound_buffer = registry->GetBoundBuffer(); | ||||
|     entry.compute_info = registry->GetComputeInfo(); | ||||
|     entry.keys = registry->GetKeys(); | ||||
|     entry.bound_samplers = registry->GetBoundSamplers(); | ||||
|     entry.bindless_samplers = registry->GetBindlessSamplers(); | ||||
|     params.disk_cache.SaveEntry(std::move(entry)); | ||||
| 
 | ||||
|     gpu.ShaderNotify().MarkShaderComplete(); | ||||
| 
 | ||||
|     return std::unique_ptr<Shader>(new Shader(std::move(registry), | ||||
|                                               MakeEntries(params.device, ir, ShaderType::Compute), | ||||
|                                               std::move(program))); | ||||
| } | ||||
| 
 | ||||
| std::unique_ptr<Shader> Shader::CreateFromCache(const ShaderParameters& params, | ||||
|                                                 const PrecompiledShader& precompiled_shader) { | ||||
|     return std::unique_ptr<Shader>(new Shader( | ||||
|         precompiled_shader.registry, precompiled_shader.entries, precompiled_shader.program)); | ||||
| } | ||||
| 
 | ||||
| ShaderCacheOpenGL::ShaderCacheOpenGL(RasterizerOpenGL& rasterizer_, | ||||
|                                      Core::Frontend::EmuWindow& emu_window_, Tegra::GPU& gpu_, | ||||
|                                      Tegra::Engines::Maxwell3D& maxwell3d_, | ||||
|  | @ -331,278 +43,4 @@ ShaderCacheOpenGL::ShaderCacheOpenGL(RasterizerOpenGL& rasterizer_, | |||
| 
 | ||||
| ShaderCacheOpenGL::~ShaderCacheOpenGL() = default; | ||||
| 
 | ||||
| void ShaderCacheOpenGL::LoadDiskCache(u64 title_id, std::stop_token stop_loading, | ||||
|                                       const VideoCore::DiskResourceLoadCallback& callback) { | ||||
|     disk_cache.BindTitleID(title_id); | ||||
|     const std::optional transferable = disk_cache.LoadTransferable(); | ||||
| 
 | ||||
|     LOG_INFO(Render_OpenGL, "Total Shader Count: {}", | ||||
|              transferable.has_value() ? transferable->size() : 0); | ||||
| 
 | ||||
|     if (!transferable) { | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     std::vector<ShaderDiskCachePrecompiled> gl_cache; | ||||
|     if (!device.UseAssemblyShaders() && !device.UseDriverCache()) { | ||||
|         // Only load precompiled cache when we are not using assembly shaders
 | ||||
|         gl_cache = disk_cache.LoadPrecompiled(); | ||||
|     } | ||||
|     const auto supported_formats = GetSupportedFormats(); | ||||
| 
 | ||||
|     // Track if precompiled cache was altered during loading to know if we have to
 | ||||
|     // serialize the virtual precompiled cache file back to the hard drive
 | ||||
|     bool precompiled_cache_altered = false; | ||||
| 
 | ||||
|     // Inform the frontend about shader build initialization
 | ||||
|     if (callback) { | ||||
|         callback(VideoCore::LoadCallbackStage::Build, 0, transferable->size()); | ||||
|     } | ||||
| 
 | ||||
|     std::mutex mutex; | ||||
|     std::size_t built_shaders = 0; // It doesn't have be atomic since it's used behind a mutex
 | ||||
|     std::atomic_bool gl_cache_failed = false; | ||||
| 
 | ||||
|     const auto find_precompiled = [&gl_cache](u64 id) { | ||||
|         return std::ranges::find(gl_cache, id, &ShaderDiskCachePrecompiled::unique_identifier); | ||||
|     }; | ||||
| 
 | ||||
|     const auto worker = [&](Core::Frontend::GraphicsContext* context, std::size_t begin, | ||||
|                             std::size_t end) { | ||||
|         const auto scope = context->Acquire(); | ||||
| 
 | ||||
|         for (std::size_t i = begin; i < end; ++i) { | ||||
|             if (stop_loading.stop_requested()) { | ||||
|                 return; | ||||
|             } | ||||
|             const auto& entry = (*transferable)[i]; | ||||
|             const u64 uid = entry.unique_identifier; | ||||
|             const auto it = find_precompiled(uid); | ||||
|             const auto precompiled_entry = it != gl_cache.end() ? &*it : nullptr; | ||||
| 
 | ||||
|             const bool is_compute = entry.type == ShaderType::Compute; | ||||
|             const u32 main_offset = is_compute ? KERNEL_MAIN_OFFSET : STAGE_MAIN_OFFSET; | ||||
|             auto registry = MakeRegistry(entry); | ||||
|             const ShaderIR ir(entry.code, main_offset, COMPILER_SETTINGS, *registry); | ||||
| 
 | ||||
|             ProgramSharedPtr program; | ||||
|             if (precompiled_entry) { | ||||
|                 // If the shader is precompiled, attempt to load it with
 | ||||
|                 program = GeneratePrecompiledProgram(entry, *precompiled_entry, supported_formats); | ||||
|                 if (!program) { | ||||
|                     gl_cache_failed = true; | ||||
|                 } | ||||
|             } | ||||
|             if (!program) { | ||||
|                 // Otherwise compile it from GLSL
 | ||||
|                 program = BuildShader(device, entry.type, uid, ir, *registry, true); | ||||
|             } | ||||
| 
 | ||||
|             PrecompiledShader shader; | ||||
|             shader.program = std::move(program); | ||||
|             shader.registry = std::move(registry); | ||||
|             shader.entries = MakeEntries(device, ir, entry.type); | ||||
| 
 | ||||
|             std::scoped_lock lock{mutex}; | ||||
|             if (callback) { | ||||
|                 callback(VideoCore::LoadCallbackStage::Build, ++built_shaders, | ||||
|                          transferable->size()); | ||||
|             } | ||||
|             runtime_cache.emplace(entry.unique_identifier, std::move(shader)); | ||||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     const std::size_t num_workers{std::max(1U, std::thread::hardware_concurrency())}; | ||||
|     const std::size_t bucket_size{transferable->size() / num_workers}; | ||||
|     std::vector<std::unique_ptr<Core::Frontend::GraphicsContext>> contexts(num_workers); | ||||
|     std::vector<std::thread> threads(num_workers); | ||||
|     for (std::size_t i = 0; i < num_workers; ++i) { | ||||
|         const bool is_last_worker = i + 1 == num_workers; | ||||
|         const std::size_t start{bucket_size * i}; | ||||
|         const std::size_t end{is_last_worker ? transferable->size() : start + bucket_size}; | ||||
| 
 | ||||
|         // On some platforms the shared context has to be created from the GUI thread
 | ||||
|         contexts[i] = emu_window.CreateSharedContext(); | ||||
|         threads[i] = std::thread(worker, contexts[i].get(), start, end); | ||||
|     } | ||||
|     for (auto& thread : threads) { | ||||
|         thread.join(); | ||||
|     } | ||||
| 
 | ||||
|     if (gl_cache_failed) { | ||||
|         // Invalidate the precompiled cache if a shader dumped shader was rejected
 | ||||
|         disk_cache.InvalidatePrecompiled(); | ||||
|         precompiled_cache_altered = true; | ||||
|         return; | ||||
|     } | ||||
|     if (stop_loading.stop_requested()) { | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     if (device.UseAssemblyShaders() || device.UseDriverCache()) { | ||||
|         // Don't store precompiled binaries for assembly shaders or when using the driver cache
 | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     // TODO(Rodrigo): Do state tracking for transferable shaders and do a dummy draw
 | ||||
|     // before precompiling them
 | ||||
| 
 | ||||
|     for (std::size_t i = 0; i < transferable->size(); ++i) { | ||||
|         const u64 id = (*transferable)[i].unique_identifier; | ||||
|         const auto it = find_precompiled(id); | ||||
|         if (it == gl_cache.end()) { | ||||
|             const GLuint program = runtime_cache.at(id).program->source_program.handle; | ||||
|             disk_cache.SavePrecompiled(id, program); | ||||
|             precompiled_cache_altered = true; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (precompiled_cache_altered) { | ||||
|         disk_cache.SaveVirtualPrecompiledFile(); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| ProgramSharedPtr ShaderCacheOpenGL::GeneratePrecompiledProgram( | ||||
|     const ShaderDiskCacheEntry& entry, const ShaderDiskCachePrecompiled& precompiled_entry, | ||||
|     const std::unordered_set<GLenum>& supported_formats) { | ||||
|     if (!supported_formats.contains(precompiled_entry.binary_format)) { | ||||
|         LOG_INFO(Render_OpenGL, "Precompiled cache entry with unsupported format, removing"); | ||||
|         return {}; | ||||
|     } | ||||
| 
 | ||||
|     auto program = std::make_shared<ProgramHandle>(); | ||||
|     GLuint& handle = program->source_program.handle; | ||||
|     handle = glCreateProgram(); | ||||
|     glProgramParameteri(handle, GL_PROGRAM_SEPARABLE, GL_TRUE); | ||||
|     glProgramBinary(handle, precompiled_entry.binary_format, precompiled_entry.binary.data(), | ||||
|                     static_cast<GLsizei>(precompiled_entry.binary.size())); | ||||
| 
 | ||||
|     GLint link_status; | ||||
|     glGetProgramiv(handle, GL_LINK_STATUS, &link_status); | ||||
|     if (link_status == GL_FALSE) { | ||||
|         LOG_INFO(Render_OpenGL, "Precompiled cache rejected by the driver, removing"); | ||||
|         return {}; | ||||
|     } | ||||
| 
 | ||||
|     return program; | ||||
| } | ||||
| 
 | ||||
| Shader* ShaderCacheOpenGL::GetStageProgram(Maxwell::ShaderProgram program, | ||||
|                                            VideoCommon::Shader::AsyncShaders& async_shaders) { | ||||
|     if (!maxwell3d.dirty.flags[Dirty::Shaders]) { | ||||
|         auto* last_shader = last_shaders[static_cast<std::size_t>(program)]; | ||||
|         if (last_shader->IsBuilt()) { | ||||
|             return last_shader; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     const GPUVAddr address{GetShaderAddress(maxwell3d, program)}; | ||||
| 
 | ||||
|     if (device.UseAsynchronousShaders() && async_shaders.HasCompletedWork()) { | ||||
|         auto completed_work = async_shaders.GetCompletedWork(); | ||||
|         for (auto& work : completed_work) { | ||||
|             Shader* shader = TryGet(work.cpu_address); | ||||
|             gpu.ShaderNotify().MarkShaderComplete(); | ||||
|             if (shader == nullptr) { | ||||
|                 continue; | ||||
|             } | ||||
|             using namespace VideoCommon::Shader; | ||||
|             if (work.backend == AsyncShaders::Backend::OpenGL) { | ||||
|                 shader->AsyncOpenGLBuilt(std::move(work.program.opengl)); | ||||
|             } else if (work.backend == AsyncShaders::Backend::GLASM) { | ||||
|                 shader->AsyncGLASMBuilt(std::move(work.program.glasm)); | ||||
|             } | ||||
| 
 | ||||
|             auto& registry = shader->GetRegistry(); | ||||
| 
 | ||||
|             ShaderDiskCacheEntry entry; | ||||
|             entry.type = work.shader_type; | ||||
|             entry.code = std::move(work.code); | ||||
|             entry.code_b = std::move(work.code_b); | ||||
|             entry.unique_identifier = work.uid; | ||||
|             entry.bound_buffer = registry.GetBoundBuffer(); | ||||
|             entry.graphics_info = registry.GetGraphicsInfo(); | ||||
|             entry.keys = registry.GetKeys(); | ||||
|             entry.bound_samplers = registry.GetBoundSamplers(); | ||||
|             entry.bindless_samplers = registry.GetBindlessSamplers(); | ||||
|             disk_cache.SaveEntry(std::move(entry)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // Look up shader in the cache based on address
 | ||||
|     const std::optional<VAddr> cpu_addr{gpu_memory.GpuToCpuAddress(address)}; | ||||
|     if (Shader* const shader{cpu_addr ? TryGet(*cpu_addr) : null_shader.get()}) { | ||||
|         return last_shaders[static_cast<std::size_t>(program)] = shader; | ||||
|     } | ||||
| 
 | ||||
|     const u8* const host_ptr{gpu_memory.GetPointer(address)}; | ||||
| 
 | ||||
|     // No shader found - create a new one
 | ||||
|     ProgramCode code{GetShaderCode(gpu_memory, address, host_ptr, false)}; | ||||
|     ProgramCode code_b; | ||||
|     if (program == Maxwell::ShaderProgram::VertexA) { | ||||
|         const GPUVAddr address_b{GetShaderAddress(maxwell3d, Maxwell::ShaderProgram::VertexB)}; | ||||
|         const u8* host_ptr_b = gpu_memory.GetPointer(address_b); | ||||
|         code_b = GetShaderCode(gpu_memory, address_b, host_ptr_b, false); | ||||
|     } | ||||
|     const std::size_t code_size = code.size() * sizeof(u64); | ||||
| 
 | ||||
|     const u64 unique_identifier = GetUniqueIdentifier( | ||||
|         GetShaderType(program), program == Maxwell::ShaderProgram::VertexA, code, code_b); | ||||
| 
 | ||||
|     const ShaderParameters params{gpu,       maxwell3d, disk_cache,       device, | ||||
|                                   *cpu_addr, host_ptr,  unique_identifier}; | ||||
| 
 | ||||
|     std::unique_ptr<Shader> shader; | ||||
|     const auto found = runtime_cache.find(unique_identifier); | ||||
|     if (found == runtime_cache.end()) { | ||||
|         shader = Shader::CreateStageFromMemory(params, program, std::move(code), std::move(code_b), | ||||
|                                                async_shaders, cpu_addr.value_or(0)); | ||||
|     } else { | ||||
|         shader = Shader::CreateFromCache(params, found->second); | ||||
|     } | ||||
| 
 | ||||
|     Shader* const result = shader.get(); | ||||
|     if (cpu_addr) { | ||||
|         Register(std::move(shader), *cpu_addr, code_size); | ||||
|     } else { | ||||
|         null_shader = std::move(shader); | ||||
|     } | ||||
| 
 | ||||
|     return last_shaders[static_cast<std::size_t>(program)] = result; | ||||
| } | ||||
| 
 | ||||
| Shader* ShaderCacheOpenGL::GetComputeKernel(GPUVAddr code_addr) { | ||||
|     const std::optional<VAddr> cpu_addr{gpu_memory.GpuToCpuAddress(code_addr)}; | ||||
| 
 | ||||
|     if (Shader* const kernel = cpu_addr ? TryGet(*cpu_addr) : null_kernel.get()) { | ||||
|         return kernel; | ||||
|     } | ||||
| 
 | ||||
|     // No kernel found, create a new one
 | ||||
|     const u8* host_ptr{gpu_memory.GetPointer(code_addr)}; | ||||
|     ProgramCode code{GetShaderCode(gpu_memory, code_addr, host_ptr, true)}; | ||||
|     const std::size_t code_size{code.size() * sizeof(u64)}; | ||||
|     const u64 unique_identifier{GetUniqueIdentifier(ShaderType::Compute, false, code)}; | ||||
| 
 | ||||
|     const ShaderParameters params{gpu,       kepler_compute, disk_cache,       device, | ||||
|                                   *cpu_addr, host_ptr,       unique_identifier}; | ||||
| 
 | ||||
|     std::unique_ptr<Shader> kernel; | ||||
|     const auto found = runtime_cache.find(unique_identifier); | ||||
|     if (found == runtime_cache.end()) { | ||||
|         kernel = Shader::CreateKernelFromMemory(params, std::move(code)); | ||||
|     } else { | ||||
|         kernel = Shader::CreateFromCache(params, found->second); | ||||
|     } | ||||
| 
 | ||||
|     Shader* const result = kernel.get(); | ||||
|     if (cpu_addr) { | ||||
|         Register(std::move(kernel), *cpu_addr, code_size); | ||||
|     } else { | ||||
|         null_kernel = std::move(kernel); | ||||
|     } | ||||
|     return result; | ||||
| } | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
|  |  | |||
|  | @ -19,10 +19,6 @@ | |||
| #include "common/common_types.h" | ||||
| #include "video_core/engines/shader_type.h" | ||||
| #include "video_core/renderer_opengl/gl_resource_manager.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_decompiler.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_disk_cache.h" | ||||
| #include "video_core/shader/registry.h" | ||||
| #include "video_core/shader/shader_ir.h" | ||||
| #include "video_core/shader_cache.h" | ||||
| 
 | ||||
| namespace Tegra { | ||||
|  | @ -33,10 +29,6 @@ namespace Core::Frontend { | |||
| class EmuWindow; | ||||
| } | ||||
| 
 | ||||
| namespace VideoCommon::Shader { | ||||
| class AsyncShaders; | ||||
| } | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| class Device; | ||||
|  | @ -44,77 +36,10 @@ class RasterizerOpenGL; | |||
| 
 | ||||
| using Maxwell = Tegra::Engines::Maxwell3D::Regs; | ||||
| 
 | ||||
| struct ProgramHandle { | ||||
|     OGLProgram source_program; | ||||
|     OGLAssemblyProgram assembly_program; | ||||
| }; | ||||
| using ProgramSharedPtr = std::shared_ptr<ProgramHandle>; | ||||
| 
 | ||||
| struct PrecompiledShader { | ||||
|     ProgramSharedPtr program; | ||||
|     std::shared_ptr<VideoCommon::Shader::Registry> registry; | ||||
|     ShaderEntries entries; | ||||
| }; | ||||
| 
 | ||||
| struct ShaderParameters { | ||||
|     Tegra::GPU& gpu; | ||||
|     Tegra::Engines::ConstBufferEngineInterface& engine; | ||||
|     ShaderDiskCacheOpenGL& disk_cache; | ||||
|     const Device& device; | ||||
|     VAddr cpu_addr; | ||||
|     const u8* host_ptr; | ||||
|     u64 unique_identifier; | ||||
| }; | ||||
| 
 | ||||
| ProgramSharedPtr BuildShader(const Device& device, Tegra::Engines::ShaderType shader_type, | ||||
|                              u64 unique_identifier, const VideoCommon::Shader::ShaderIR& ir, | ||||
|                              const VideoCommon::Shader::Registry& registry, | ||||
|                              bool hint_retrievable = false); | ||||
| 
 | ||||
| class Shader final { | ||||
| class Shader { | ||||
| public: | ||||
|     explicit Shader(); | ||||
|     ~Shader(); | ||||
| 
 | ||||
|     /// Gets the GL program handle for the shader
 | ||||
|     GLuint GetHandle() const; | ||||
| 
 | ||||
|     bool IsBuilt() const; | ||||
| 
 | ||||
|     /// Gets the shader entries for the shader
 | ||||
|     const ShaderEntries& GetEntries() const { | ||||
|         return entries; | ||||
|     } | ||||
| 
 | ||||
|     const VideoCommon::Shader::Registry& GetRegistry() const { | ||||
|         return *registry; | ||||
|     } | ||||
| 
 | ||||
|     /// Mark a OpenGL shader as built
 | ||||
|     void AsyncOpenGLBuilt(OGLProgram new_program); | ||||
| 
 | ||||
|     /// Mark a GLASM shader as built
 | ||||
|     void AsyncGLASMBuilt(OGLAssemblyProgram new_program); | ||||
| 
 | ||||
|     static std::unique_ptr<Shader> CreateStageFromMemory( | ||||
|         const ShaderParameters& params, Maxwell::ShaderProgram program_type, | ||||
|         ProgramCode program_code, ProgramCode program_code_b, | ||||
|         VideoCommon::Shader::AsyncShaders& async_shaders, VAddr cpu_addr); | ||||
| 
 | ||||
|     static std::unique_ptr<Shader> CreateKernelFromMemory(const ShaderParameters& params, | ||||
|                                                           ProgramCode code); | ||||
| 
 | ||||
|     static std::unique_ptr<Shader> CreateFromCache(const ShaderParameters& params, | ||||
|                                                    const PrecompiledShader& precompiled_shader); | ||||
| 
 | ||||
| private: | ||||
|     explicit Shader(std::shared_ptr<VideoCommon::Shader::Registry> registry, ShaderEntries entries, | ||||
|                     ProgramSharedPtr program, bool is_built_ = true); | ||||
| 
 | ||||
|     std::shared_ptr<VideoCommon::Shader::Registry> registry; | ||||
|     ShaderEntries entries; | ||||
|     ProgramSharedPtr program; | ||||
|     GLuint handle = 0; | ||||
|     bool is_built{}; | ||||
| }; | ||||
| 
 | ||||
| class ShaderCacheOpenGL final : public VideoCommon::ShaderCache<Shader> { | ||||
|  | @ -126,36 +51,13 @@ public: | |||
|                                Tegra::MemoryManager& gpu_memory_, const Device& device_); | ||||
|     ~ShaderCacheOpenGL() override; | ||||
| 
 | ||||
|     /// Loads disk cache for the current game
 | ||||
|     void LoadDiskCache(u64 title_id, std::stop_token stop_loading, | ||||
|                        const VideoCore::DiskResourceLoadCallback& callback); | ||||
| 
 | ||||
|     /// Gets the current specified shader stage program
 | ||||
|     Shader* GetStageProgram(Maxwell::ShaderProgram program, | ||||
|                             VideoCommon::Shader::AsyncShaders& async_shaders); | ||||
| 
 | ||||
|     /// Gets a compute kernel in the passed address
 | ||||
|     Shader* GetComputeKernel(GPUVAddr code_addr); | ||||
| 
 | ||||
| private: | ||||
|     ProgramSharedPtr GeneratePrecompiledProgram( | ||||
|         const ShaderDiskCacheEntry& entry, const ShaderDiskCachePrecompiled& precompiled_entry, | ||||
|         const std::unordered_set<GLenum>& supported_formats); | ||||
| 
 | ||||
|     Core::Frontend::EmuWindow& emu_window; | ||||
|     Tegra::GPU& gpu; | ||||
|     Tegra::MemoryManager& gpu_memory; | ||||
|     Tegra::Engines::Maxwell3D& maxwell3d; | ||||
|     Tegra::Engines::KeplerCompute& kepler_compute; | ||||
|     const Device& device; | ||||
| 
 | ||||
|     ShaderDiskCacheOpenGL disk_cache; | ||||
|     std::unordered_map<u64, PrecompiledShader> runtime_cache; | ||||
| 
 | ||||
|     std::unique_ptr<Shader> null_shader; | ||||
|     std::unique_ptr<Shader> null_kernel; | ||||
| 
 | ||||
|     std::array<Shader*, Maxwell::MaxShaderProgram> last_shaders{}; | ||||
| }; | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
|  |  | |||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -1,69 +0,0 @@ | |||
| // Copyright 2018 yuzu Emulator Project
 | ||||
| // Licensed under GPLv2 or any later version
 | ||||
| // Refer to the license.txt file included.
 | ||||
| 
 | ||||
| #pragma once | ||||
| 
 | ||||
| #include <array> | ||||
| #include <string> | ||||
| #include <string_view> | ||||
| #include <utility> | ||||
| #include <vector> | ||||
| #include "common/common_types.h" | ||||
| #include "video_core/engines/maxwell_3d.h" | ||||
| #include "video_core/engines/shader_type.h" | ||||
| #include "video_core/shader/registry.h" | ||||
| #include "video_core/shader/shader_ir.h" | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| class Device; | ||||
| 
 | ||||
| using Maxwell = Tegra::Engines::Maxwell3D::Regs; | ||||
| using SamplerEntry = VideoCommon::Shader::SamplerEntry; | ||||
| using ImageEntry = VideoCommon::Shader::ImageEntry; | ||||
| 
 | ||||
| class ConstBufferEntry : public VideoCommon::Shader::ConstBuffer { | ||||
| public: | ||||
|     explicit ConstBufferEntry(u32 max_offset_, bool is_indirect_, u32 index_) | ||||
|         : ConstBuffer{max_offset_, is_indirect_}, index{index_} {} | ||||
| 
 | ||||
|     u32 GetIndex() const { | ||||
|         return index; | ||||
|     } | ||||
| 
 | ||||
| private: | ||||
|     u32 index = 0; | ||||
| }; | ||||
| 
 | ||||
| struct GlobalMemoryEntry { | ||||
|     constexpr explicit GlobalMemoryEntry(u32 cbuf_index_, u32 cbuf_offset_, bool is_read_, | ||||
|                                          bool is_written_) | ||||
|         : cbuf_index{cbuf_index_}, cbuf_offset{cbuf_offset_}, is_read{is_read_}, is_written{ | ||||
|                                                                                      is_written_} {} | ||||
| 
 | ||||
|     u32 cbuf_index = 0; | ||||
|     u32 cbuf_offset = 0; | ||||
|     bool is_read = false; | ||||
|     bool is_written = false; | ||||
| }; | ||||
| 
 | ||||
| struct ShaderEntries { | ||||
|     std::vector<ConstBufferEntry> const_buffers; | ||||
|     std::vector<GlobalMemoryEntry> global_memory_entries; | ||||
|     std::vector<SamplerEntry> samplers; | ||||
|     std::vector<ImageEntry> images; | ||||
|     std::size_t shader_length{}; | ||||
|     u32 clip_distances{}; | ||||
|     u32 enabled_uniform_buffers{}; | ||||
| }; | ||||
| 
 | ||||
| ShaderEntries MakeEntries(const Device& device, const VideoCommon::Shader::ShaderIR& ir, | ||||
|                           Tegra::Engines::ShaderType stage); | ||||
| 
 | ||||
| std::string DecompileShader(const Device& device, const VideoCommon::Shader::ShaderIR& ir, | ||||
|                             const VideoCommon::Shader::Registry& registry, | ||||
|                             Tegra::Engines::ShaderType stage, std::string_view identifier, | ||||
|                             std::string_view suffix = {}); | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
|  | @ -1,482 +0,0 @@ | |||
| // Copyright 2019 yuzu Emulator Project
 | ||||
| // Licensed under GPLv2 or any later version
 | ||||
| // Refer to the license.txt file included.
 | ||||
| 
 | ||||
| #include <cstring> | ||||
| 
 | ||||
| #include <fmt/format.h> | ||||
| 
 | ||||
| #include "common/assert.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/fs/file.h" | ||||
| #include "common/fs/fs.h" | ||||
| #include "common/fs/path_util.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "common/scm_rev.h" | ||||
| #include "common/settings.h" | ||||
| #include "common/zstd_compression.h" | ||||
| #include "core/core.h" | ||||
| #include "core/hle/kernel/k_process.h" | ||||
| #include "video_core/engines/shader_type.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_cache.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_disk_cache.h" | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| using Tegra::Engines::ShaderType; | ||||
| using VideoCommon::Shader::BindlessSamplerMap; | ||||
| using VideoCommon::Shader::BoundSamplerMap; | ||||
| using VideoCommon::Shader::KeyMap; | ||||
| using VideoCommon::Shader::SeparateSamplerKey; | ||||
| using ShaderCacheVersionHash = std::array<u8, 64>; | ||||
| 
 | ||||
| struct ConstBufferKey { | ||||
|     u32 cbuf = 0; | ||||
|     u32 offset = 0; | ||||
|     u32 value = 0; | ||||
| }; | ||||
| 
 | ||||
| struct BoundSamplerEntry { | ||||
|     u32 offset = 0; | ||||
|     Tegra::Engines::SamplerDescriptor sampler; | ||||
| }; | ||||
| 
 | ||||
| struct SeparateSamplerEntry { | ||||
|     u32 cbuf1 = 0; | ||||
|     u32 cbuf2 = 0; | ||||
|     u32 offset1 = 0; | ||||
|     u32 offset2 = 0; | ||||
|     Tegra::Engines::SamplerDescriptor sampler; | ||||
| }; | ||||
| 
 | ||||
| struct BindlessSamplerEntry { | ||||
|     u32 cbuf = 0; | ||||
|     u32 offset = 0; | ||||
|     Tegra::Engines::SamplerDescriptor sampler; | ||||
| }; | ||||
| 
 | ||||
| namespace { | ||||
| 
 | ||||
| constexpr u32 NativeVersion = 21; | ||||
| 
 | ||||
| ShaderCacheVersionHash GetShaderCacheVersionHash() { | ||||
|     ShaderCacheVersionHash hash{}; | ||||
|     const std::size_t length = std::min(std::strlen(Common::g_shader_cache_version), hash.size()); | ||||
|     std::memcpy(hash.data(), Common::g_shader_cache_version, length); | ||||
|     return hash; | ||||
| } | ||||
| 
 | ||||
| } // Anonymous namespace
 | ||||
| 
 | ||||
| ShaderDiskCacheEntry::ShaderDiskCacheEntry() = default; | ||||
| 
 | ||||
| ShaderDiskCacheEntry::~ShaderDiskCacheEntry() = default; | ||||
| 
 | ||||
| bool ShaderDiskCacheEntry::Load(Common::FS::IOFile& file) { | ||||
|     if (!file.ReadObject(type)) { | ||||
|         return false; | ||||
|     } | ||||
|     u32 code_size; | ||||
|     u32 code_size_b; | ||||
|     if (!file.ReadObject(code_size) || !file.ReadObject(code_size_b)) { | ||||
|         return false; | ||||
|     } | ||||
|     code.resize(code_size); | ||||
|     code_b.resize(code_size_b); | ||||
|     if (file.Read(code) != code_size) { | ||||
|         return false; | ||||
|     } | ||||
|     if (HasProgramA() && file.Read(code_b) != code_size_b) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     u8 is_texture_handler_size_known; | ||||
|     u32 texture_handler_size_value; | ||||
|     u32 num_keys; | ||||
|     u32 num_bound_samplers; | ||||
|     u32 num_separate_samplers; | ||||
|     u32 num_bindless_samplers; | ||||
|     if (!file.ReadObject(unique_identifier) || !file.ReadObject(bound_buffer) || | ||||
|         !file.ReadObject(is_texture_handler_size_known) || | ||||
|         !file.ReadObject(texture_handler_size_value) || !file.ReadObject(graphics_info) || | ||||
|         !file.ReadObject(compute_info) || !file.ReadObject(num_keys) || | ||||
|         !file.ReadObject(num_bound_samplers) || !file.ReadObject(num_separate_samplers) || | ||||
|         !file.ReadObject(num_bindless_samplers)) { | ||||
|         return false; | ||||
|     } | ||||
|     if (is_texture_handler_size_known) { | ||||
|         texture_handler_size = texture_handler_size_value; | ||||
|     } | ||||
| 
 | ||||
|     std::vector<ConstBufferKey> flat_keys(num_keys); | ||||
|     std::vector<BoundSamplerEntry> flat_bound_samplers(num_bound_samplers); | ||||
|     std::vector<SeparateSamplerEntry> flat_separate_samplers(num_separate_samplers); | ||||
|     std::vector<BindlessSamplerEntry> flat_bindless_samplers(num_bindless_samplers); | ||||
|     if (file.Read(flat_keys) != flat_keys.size() || | ||||
|         file.Read(flat_bound_samplers) != flat_bound_samplers.size() || | ||||
|         file.Read(flat_separate_samplers) != flat_separate_samplers.size() || | ||||
|         file.Read(flat_bindless_samplers) != flat_bindless_samplers.size()) { | ||||
|         return false; | ||||
|     } | ||||
|     for (const auto& entry : flat_keys) { | ||||
|         keys.insert({{entry.cbuf, entry.offset}, entry.value}); | ||||
|     } | ||||
|     for (const auto& entry : flat_bound_samplers) { | ||||
|         bound_samplers.emplace(entry.offset, entry.sampler); | ||||
|     } | ||||
|     for (const auto& entry : flat_separate_samplers) { | ||||
|         SeparateSamplerKey key; | ||||
|         key.buffers = {entry.cbuf1, entry.cbuf2}; | ||||
|         key.offsets = {entry.offset1, entry.offset2}; | ||||
|         separate_samplers.emplace(key, entry.sampler); | ||||
|     } | ||||
|     for (const auto& entry : flat_bindless_samplers) { | ||||
|         bindless_samplers.insert({{entry.cbuf, entry.offset}, entry.sampler}); | ||||
|     } | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| bool ShaderDiskCacheEntry::Save(Common::FS::IOFile& file) const { | ||||
|     if (!file.WriteObject(static_cast<u32>(type)) || | ||||
|         !file.WriteObject(static_cast<u32>(code.size())) || | ||||
|         !file.WriteObject(static_cast<u32>(code_b.size()))) { | ||||
|         return false; | ||||
|     } | ||||
|     if (file.Write(code) != code.size()) { | ||||
|         return false; | ||||
|     } | ||||
|     if (HasProgramA() && file.Write(code_b) != code_b.size()) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     if (!file.WriteObject(unique_identifier) || !file.WriteObject(bound_buffer) || | ||||
|         !file.WriteObject(static_cast<u8>(texture_handler_size.has_value())) || | ||||
|         !file.WriteObject(texture_handler_size.value_or(0)) || !file.WriteObject(graphics_info) || | ||||
|         !file.WriteObject(compute_info) || !file.WriteObject(static_cast<u32>(keys.size())) || | ||||
|         !file.WriteObject(static_cast<u32>(bound_samplers.size())) || | ||||
|         !file.WriteObject(static_cast<u32>(separate_samplers.size())) || | ||||
|         !file.WriteObject(static_cast<u32>(bindless_samplers.size()))) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     std::vector<ConstBufferKey> flat_keys; | ||||
|     flat_keys.reserve(keys.size()); | ||||
|     for (const auto& [address, value] : keys) { | ||||
|         flat_keys.push_back(ConstBufferKey{address.first, address.second, value}); | ||||
|     } | ||||
| 
 | ||||
|     std::vector<BoundSamplerEntry> flat_bound_samplers; | ||||
|     flat_bound_samplers.reserve(bound_samplers.size()); | ||||
|     for (const auto& [address, sampler] : bound_samplers) { | ||||
|         flat_bound_samplers.push_back(BoundSamplerEntry{address, sampler}); | ||||
|     } | ||||
| 
 | ||||
|     std::vector<SeparateSamplerEntry> flat_separate_samplers; | ||||
|     flat_separate_samplers.reserve(separate_samplers.size()); | ||||
|     for (const auto& [key, sampler] : separate_samplers) { | ||||
|         SeparateSamplerEntry entry; | ||||
|         std::tie(entry.cbuf1, entry.cbuf2) = key.buffers; | ||||
|         std::tie(entry.offset1, entry.offset2) = key.offsets; | ||||
|         entry.sampler = sampler; | ||||
|         flat_separate_samplers.push_back(entry); | ||||
|     } | ||||
| 
 | ||||
|     std::vector<BindlessSamplerEntry> flat_bindless_samplers; | ||||
|     flat_bindless_samplers.reserve(bindless_samplers.size()); | ||||
|     for (const auto& [address, sampler] : bindless_samplers) { | ||||
|         flat_bindless_samplers.push_back( | ||||
|             BindlessSamplerEntry{address.first, address.second, sampler}); | ||||
|     } | ||||
| 
 | ||||
|     return file.Write(flat_keys) == flat_keys.size() && | ||||
|            file.Write(flat_bound_samplers) == flat_bound_samplers.size() && | ||||
|            file.Write(flat_separate_samplers) == flat_separate_samplers.size() && | ||||
|            file.Write(flat_bindless_samplers) == flat_bindless_samplers.size(); | ||||
| } | ||||
| 
 | ||||
| ShaderDiskCacheOpenGL::ShaderDiskCacheOpenGL() = default; | ||||
| 
 | ||||
| ShaderDiskCacheOpenGL::~ShaderDiskCacheOpenGL() = default; | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::BindTitleID(u64 title_id_) { | ||||
|     title_id = title_id_; | ||||
| } | ||||
| 
 | ||||
| std::optional<std::vector<ShaderDiskCacheEntry>> ShaderDiskCacheOpenGL::LoadTransferable() { | ||||
|     // Skip games without title id
 | ||||
|     const bool has_title_id = title_id != 0; | ||||
|     if (!Settings::values.use_disk_shader_cache.GetValue() || !has_title_id) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
| 
 | ||||
|     Common::FS::IOFile file{GetTransferablePath(), Common::FS::FileAccessMode::Read, | ||||
|                             Common::FS::FileType::BinaryFile}; | ||||
|     if (!file.IsOpen()) { | ||||
|         LOG_INFO(Render_OpenGL, "No transferable shader cache found"); | ||||
|         is_usable = true; | ||||
|         return std::nullopt; | ||||
|     } | ||||
| 
 | ||||
|     u32 version{}; | ||||
|     if (!file.ReadObject(version)) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to get transferable cache version, skipping it"); | ||||
|         return std::nullopt; | ||||
|     } | ||||
| 
 | ||||
|     if (version < NativeVersion) { | ||||
|         LOG_INFO(Render_OpenGL, "Transferable shader cache is old, removing"); | ||||
|         file.Close(); | ||||
|         InvalidateTransferable(); | ||||
|         is_usable = true; | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     if (version > NativeVersion) { | ||||
|         LOG_WARNING(Render_OpenGL, "Transferable shader cache was generated with a newer version " | ||||
|                                    "of the emulator, skipping"); | ||||
|         return std::nullopt; | ||||
|     } | ||||
| 
 | ||||
|     // Version is valid, load the shaders
 | ||||
|     std::vector<ShaderDiskCacheEntry> entries; | ||||
|     while (static_cast<u64>(file.Tell()) < file.GetSize()) { | ||||
|         ShaderDiskCacheEntry& entry = entries.emplace_back(); | ||||
|         if (!entry.Load(file)) { | ||||
|             LOG_ERROR(Render_OpenGL, "Failed to load transferable raw entry, skipping"); | ||||
|             return std::nullopt; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     is_usable = true; | ||||
|     return {std::move(entries)}; | ||||
| } | ||||
| 
 | ||||
| std::vector<ShaderDiskCachePrecompiled> ShaderDiskCacheOpenGL::LoadPrecompiled() { | ||||
|     if (!is_usable) { | ||||
|         return {}; | ||||
|     } | ||||
| 
 | ||||
|     Common::FS::IOFile file{GetPrecompiledPath(), Common::FS::FileAccessMode::Read, | ||||
|                             Common::FS::FileType::BinaryFile}; | ||||
|     if (!file.IsOpen()) { | ||||
|         LOG_INFO(Render_OpenGL, "No precompiled shader cache found"); | ||||
|         return {}; | ||||
|     } | ||||
| 
 | ||||
|     if (const auto result = LoadPrecompiledFile(file)) { | ||||
|         return *result; | ||||
|     } | ||||
| 
 | ||||
|     LOG_INFO(Render_OpenGL, "Failed to load precompiled cache"); | ||||
|     file.Close(); | ||||
|     InvalidatePrecompiled(); | ||||
|     return {}; | ||||
| } | ||||
| 
 | ||||
| std::optional<std::vector<ShaderDiskCachePrecompiled>> ShaderDiskCacheOpenGL::LoadPrecompiledFile( | ||||
|     Common::FS::IOFile& file) { | ||||
|     // Read compressed file from disk and decompress to virtual precompiled cache file
 | ||||
|     std::vector<u8> compressed(file.GetSize()); | ||||
|     if (file.Read(compressed) != file.GetSize()) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     const std::vector<u8> decompressed = Common::Compression::DecompressDataZSTD(compressed); | ||||
|     SaveArrayToPrecompiled(decompressed.data(), decompressed.size()); | ||||
|     precompiled_cache_virtual_file_offset = 0; | ||||
| 
 | ||||
|     ShaderCacheVersionHash file_hash{}; | ||||
|     if (!LoadArrayFromPrecompiled(file_hash.data(), file_hash.size())) { | ||||
|         precompiled_cache_virtual_file_offset = 0; | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     if (GetShaderCacheVersionHash() != file_hash) { | ||||
|         LOG_INFO(Render_OpenGL, "Precompiled cache is from another version of the emulator"); | ||||
|         precompiled_cache_virtual_file_offset = 0; | ||||
|         return std::nullopt; | ||||
|     } | ||||
| 
 | ||||
|     std::vector<ShaderDiskCachePrecompiled> entries; | ||||
|     while (precompiled_cache_virtual_file_offset < precompiled_cache_virtual_file.GetSize()) { | ||||
|         u32 binary_size; | ||||
|         auto& entry = entries.emplace_back(); | ||||
|         if (!LoadObjectFromPrecompiled(entry.unique_identifier) || | ||||
|             !LoadObjectFromPrecompiled(entry.binary_format) || | ||||
|             !LoadObjectFromPrecompiled(binary_size)) { | ||||
|             return std::nullopt; | ||||
|         } | ||||
| 
 | ||||
|         entry.binary.resize(binary_size); | ||||
|         if (!LoadArrayFromPrecompiled(entry.binary.data(), entry.binary.size())) { | ||||
|             return std::nullopt; | ||||
|         } | ||||
|     } | ||||
|     return entries; | ||||
| } | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::InvalidateTransferable() { | ||||
|     if (!Common::FS::RemoveFile(GetTransferablePath())) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to invalidate transferable file={}", | ||||
|                   Common::FS::PathToUTF8String(GetTransferablePath())); | ||||
|     } | ||||
|     InvalidatePrecompiled(); | ||||
| } | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::InvalidatePrecompiled() { | ||||
|     // Clear virtaul precompiled cache file
 | ||||
|     precompiled_cache_virtual_file.Resize(0); | ||||
| 
 | ||||
|     if (!Common::FS::RemoveFile(GetPrecompiledPath())) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to invalidate precompiled file={}", | ||||
|                   Common::FS::PathToUTF8String(GetPrecompiledPath())); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::SaveEntry(const ShaderDiskCacheEntry& entry) { | ||||
|     if (!is_usable) { | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     const u64 id = entry.unique_identifier; | ||||
|     if (stored_transferable.contains(id)) { | ||||
|         // The shader already exists
 | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     Common::FS::IOFile file = AppendTransferableFile(); | ||||
|     if (!file.IsOpen()) { | ||||
|         return; | ||||
|     } | ||||
|     if (!entry.Save(file)) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to save raw transferable cache entry, removing"); | ||||
|         file.Close(); | ||||
|         InvalidateTransferable(); | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     stored_transferable.insert(id); | ||||
| } | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::SavePrecompiled(u64 unique_identifier, GLuint program) { | ||||
|     if (!is_usable) { | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     // TODO(Rodrigo): This is a design smell. I shouldn't be having to manually write the header
 | ||||
|     // when writing the dump. This should be done the moment I get access to write to the virtual
 | ||||
|     // file.
 | ||||
|     if (precompiled_cache_virtual_file.GetSize() == 0) { | ||||
|         SavePrecompiledHeaderToVirtualPrecompiledCache(); | ||||
|     } | ||||
| 
 | ||||
|     GLint binary_length; | ||||
|     glGetProgramiv(program, GL_PROGRAM_BINARY_LENGTH, &binary_length); | ||||
| 
 | ||||
|     GLenum binary_format; | ||||
|     std::vector<u8> binary(binary_length); | ||||
|     glGetProgramBinary(program, binary_length, nullptr, &binary_format, binary.data()); | ||||
| 
 | ||||
|     if (!SaveObjectToPrecompiled(unique_identifier) || !SaveObjectToPrecompiled(binary_format) || | ||||
|         !SaveObjectToPrecompiled(static_cast<u32>(binary.size())) || | ||||
|         !SaveArrayToPrecompiled(binary.data(), binary.size())) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to save binary program file in shader={:016X}, removing", | ||||
|                   unique_identifier); | ||||
|         InvalidatePrecompiled(); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| Common::FS::IOFile ShaderDiskCacheOpenGL::AppendTransferableFile() const { | ||||
|     if (!EnsureDirectories()) { | ||||
|         return {}; | ||||
|     } | ||||
| 
 | ||||
|     const auto transferable_path{GetTransferablePath()}; | ||||
|     const bool existed = Common::FS::Exists(transferable_path); | ||||
| 
 | ||||
|     Common::FS::IOFile file{transferable_path, Common::FS::FileAccessMode::Append, | ||||
|                             Common::FS::FileType::BinaryFile}; | ||||
|     if (!file.IsOpen()) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to open transferable cache in path={}", | ||||
|                   Common::FS::PathToUTF8String(transferable_path)); | ||||
|         return {}; | ||||
|     } | ||||
|     if (!existed || file.GetSize() == 0) { | ||||
|         // If the file didn't exist, write its version
 | ||||
|         if (!file.WriteObject(NativeVersion)) { | ||||
|             LOG_ERROR(Render_OpenGL, "Failed to write transferable cache version in path={}", | ||||
|                       Common::FS::PathToUTF8String(transferable_path)); | ||||
|             return {}; | ||||
|         } | ||||
|     } | ||||
|     return file; | ||||
| } | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::SavePrecompiledHeaderToVirtualPrecompiledCache() { | ||||
|     const auto hash{GetShaderCacheVersionHash()}; | ||||
|     if (!SaveArrayToPrecompiled(hash.data(), hash.size())) { | ||||
|         LOG_ERROR( | ||||
|             Render_OpenGL, | ||||
|             "Failed to write precompiled cache version hash to virtual precompiled cache file"); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void ShaderDiskCacheOpenGL::SaveVirtualPrecompiledFile() { | ||||
|     precompiled_cache_virtual_file_offset = 0; | ||||
|     const std::vector<u8> uncompressed = precompiled_cache_virtual_file.ReadAllBytes(); | ||||
|     const std::vector<u8> compressed = | ||||
|         Common::Compression::CompressDataZSTDDefault(uncompressed.data(), uncompressed.size()); | ||||
| 
 | ||||
|     const auto precompiled_path = GetPrecompiledPath(); | ||||
|     Common::FS::IOFile file{precompiled_path, Common::FS::FileAccessMode::Write, | ||||
|                             Common::FS::FileType::BinaryFile}; | ||||
| 
 | ||||
|     if (!file.IsOpen()) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to open precompiled cache in path={}", | ||||
|                   Common::FS::PathToUTF8String(precompiled_path)); | ||||
|         return; | ||||
|     } | ||||
|     if (file.Write(compressed) != compressed.size()) { | ||||
|         LOG_ERROR(Render_OpenGL, "Failed to write precompiled cache version in path={}", | ||||
|                   Common::FS::PathToUTF8String(precompiled_path)); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| bool ShaderDiskCacheOpenGL::EnsureDirectories() const { | ||||
|     const auto CreateDir = [](const std::filesystem::path& dir) { | ||||
|         if (!Common::FS::CreateDir(dir)) { | ||||
|             LOG_ERROR(Render_OpenGL, "Failed to create directory={}", | ||||
|                       Common::FS::PathToUTF8String(dir)); | ||||
|             return false; | ||||
|         } | ||||
|         return true; | ||||
|     }; | ||||
| 
 | ||||
|     return CreateDir(Common::FS::GetYuzuPath(Common::FS::YuzuPath::ShaderDir)) && | ||||
|            CreateDir(GetBaseDir()) && CreateDir(GetTransferableDir()) && | ||||
|            CreateDir(GetPrecompiledDir()); | ||||
| } | ||||
| 
 | ||||
| std::filesystem::path ShaderDiskCacheOpenGL::GetTransferablePath() const { | ||||
|     return GetTransferableDir() / fmt::format("{}.bin", GetTitleID()); | ||||
| } | ||||
| 
 | ||||
| std::filesystem::path ShaderDiskCacheOpenGL::GetPrecompiledPath() const { | ||||
|     return GetPrecompiledDir() / fmt::format("{}.bin", GetTitleID()); | ||||
| } | ||||
| 
 | ||||
| std::filesystem::path ShaderDiskCacheOpenGL::GetTransferableDir() const { | ||||
|     return GetBaseDir() / "transferable"; | ||||
| } | ||||
| 
 | ||||
| std::filesystem::path ShaderDiskCacheOpenGL::GetPrecompiledDir() const { | ||||
|     return GetBaseDir() / "precompiled"; | ||||
| } | ||||
| 
 | ||||
| std::filesystem::path ShaderDiskCacheOpenGL::GetBaseDir() const { | ||||
|     return Common::FS::GetYuzuPath(Common::FS::YuzuPath::ShaderDir) / "opengl"; | ||||
| } | ||||
| 
 | ||||
| std::string ShaderDiskCacheOpenGL::GetTitleID() const { | ||||
|     return fmt::format("{:016X}", title_id); | ||||
| } | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
|  | @ -1,176 +0,0 @@ | |||
| // Copyright 2019 yuzu Emulator Project
 | ||||
| // Licensed under GPLv2 or any later version
 | ||||
| // Refer to the license.txt file included.
 | ||||
| 
 | ||||
| #pragma once | ||||
| 
 | ||||
| #include <filesystem> | ||||
| #include <optional> | ||||
| #include <string> | ||||
| #include <tuple> | ||||
| #include <type_traits> | ||||
| #include <unordered_map> | ||||
| #include <unordered_set> | ||||
| #include <utility> | ||||
| #include <vector> | ||||
| 
 | ||||
| #include <glad/glad.h> | ||||
| 
 | ||||
| #include "common/assert.h" | ||||
| #include "common/common_types.h" | ||||
| #include "core/file_sys/vfs_vector.h" | ||||
| #include "video_core/engines/shader_type.h" | ||||
| #include "video_core/shader/registry.h" | ||||
| 
 | ||||
| namespace Common::FS { | ||||
| class IOFile; | ||||
| } | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| using ProgramCode = std::vector<u64>; | ||||
| 
 | ||||
| /// Describes a shader and how it's used by the guest GPU
 | ||||
| struct ShaderDiskCacheEntry { | ||||
|     ShaderDiskCacheEntry(); | ||||
|     ~ShaderDiskCacheEntry(); | ||||
| 
 | ||||
|     bool Load(Common::FS::IOFile& file); | ||||
| 
 | ||||
|     bool Save(Common::FS::IOFile& file) const; | ||||
| 
 | ||||
|     bool HasProgramA() const { | ||||
|         return !code.empty() && !code_b.empty(); | ||||
|     } | ||||
| 
 | ||||
|     Tegra::Engines::ShaderType type{}; | ||||
|     ProgramCode code; | ||||
|     ProgramCode code_b; | ||||
| 
 | ||||
|     u64 unique_identifier = 0; | ||||
|     std::optional<u32> texture_handler_size; | ||||
|     u32 bound_buffer = 0; | ||||
|     VideoCommon::Shader::GraphicsInfo graphics_info; | ||||
|     VideoCommon::Shader::ComputeInfo compute_info; | ||||
|     VideoCommon::Shader::KeyMap keys; | ||||
|     VideoCommon::Shader::BoundSamplerMap bound_samplers; | ||||
|     VideoCommon::Shader::SeparateSamplerMap separate_samplers; | ||||
|     VideoCommon::Shader::BindlessSamplerMap bindless_samplers; | ||||
| }; | ||||
| 
 | ||||
| /// Contains an OpenGL dumped binary program
 | ||||
| struct ShaderDiskCachePrecompiled { | ||||
|     u64 unique_identifier = 0; | ||||
|     GLenum binary_format = 0; | ||||
|     std::vector<u8> binary; | ||||
| }; | ||||
| 
 | ||||
| class ShaderDiskCacheOpenGL { | ||||
| public: | ||||
|     explicit ShaderDiskCacheOpenGL(); | ||||
|     ~ShaderDiskCacheOpenGL(); | ||||
| 
 | ||||
|     /// Binds a title ID for all future operations.
 | ||||
|     void BindTitleID(u64 title_id); | ||||
| 
 | ||||
|     /// Loads transferable cache. If file has a old version or on failure, it deletes the file.
 | ||||
|     std::optional<std::vector<ShaderDiskCacheEntry>> LoadTransferable(); | ||||
| 
 | ||||
|     /// Loads current game's precompiled cache. Invalidates on failure.
 | ||||
|     std::vector<ShaderDiskCachePrecompiled> LoadPrecompiled(); | ||||
| 
 | ||||
|     /// Removes the transferable (and precompiled) cache file.
 | ||||
|     void InvalidateTransferable(); | ||||
| 
 | ||||
|     /// Removes the precompiled cache file and clears virtual precompiled cache file.
 | ||||
|     void InvalidatePrecompiled(); | ||||
| 
 | ||||
|     /// Saves a raw dump to the transferable file. Checks for collisions.
 | ||||
|     void SaveEntry(const ShaderDiskCacheEntry& entry); | ||||
| 
 | ||||
|     /// Saves a dump entry to the precompiled file. Does not check for collisions.
 | ||||
|     void SavePrecompiled(u64 unique_identifier, GLuint program); | ||||
| 
 | ||||
|     /// Serializes virtual precompiled shader cache file to real file
 | ||||
|     void SaveVirtualPrecompiledFile(); | ||||
| 
 | ||||
| private: | ||||
|     /// Loads the transferable cache. Returns empty on failure.
 | ||||
|     std::optional<std::vector<ShaderDiskCachePrecompiled>> LoadPrecompiledFile( | ||||
|         Common::FS::IOFile& file); | ||||
| 
 | ||||
|     /// Opens current game's transferable file and write it's header if it doesn't exist
 | ||||
|     Common::FS::IOFile AppendTransferableFile() const; | ||||
| 
 | ||||
|     /// Save precompiled header to precompiled_cache_in_memory
 | ||||
|     void SavePrecompiledHeaderToVirtualPrecompiledCache(); | ||||
| 
 | ||||
|     /// Create shader disk cache directories. Returns true on success.
 | ||||
|     bool EnsureDirectories() const; | ||||
| 
 | ||||
|     /// Gets current game's transferable file path
 | ||||
|     std::filesystem::path GetTransferablePath() const; | ||||
| 
 | ||||
|     /// Gets current game's precompiled file path
 | ||||
|     std::filesystem::path GetPrecompiledPath() const; | ||||
| 
 | ||||
|     /// Get user's transferable directory path
 | ||||
|     std::filesystem::path GetTransferableDir() const; | ||||
| 
 | ||||
|     /// Get user's precompiled directory path
 | ||||
|     std::filesystem::path GetPrecompiledDir() const; | ||||
| 
 | ||||
|     /// Get user's shader directory path
 | ||||
|     std::filesystem::path GetBaseDir() const; | ||||
| 
 | ||||
|     /// Get current game's title id
 | ||||
|     std::string GetTitleID() const; | ||||
| 
 | ||||
|     template <typename T> | ||||
|     bool SaveArrayToPrecompiled(const T* data, std::size_t length) { | ||||
|         const std::size_t write_length = precompiled_cache_virtual_file.WriteArray( | ||||
|             data, length, precompiled_cache_virtual_file_offset); | ||||
|         precompiled_cache_virtual_file_offset += write_length; | ||||
|         return write_length == sizeof(T) * length; | ||||
|     } | ||||
| 
 | ||||
|     template <typename T> | ||||
|     bool LoadArrayFromPrecompiled(T* data, std::size_t length) { | ||||
|         const std::size_t read_length = precompiled_cache_virtual_file.ReadArray( | ||||
|             data, length, precompiled_cache_virtual_file_offset); | ||||
|         precompiled_cache_virtual_file_offset += read_length; | ||||
|         return read_length == sizeof(T) * length; | ||||
|     } | ||||
| 
 | ||||
|     template <typename T> | ||||
|     bool SaveObjectToPrecompiled(const T& object) { | ||||
|         return SaveArrayToPrecompiled(&object, 1); | ||||
|     } | ||||
| 
 | ||||
|     bool SaveObjectToPrecompiled(bool object) { | ||||
|         const auto value = static_cast<u8>(object); | ||||
|         return SaveArrayToPrecompiled(&value, 1); | ||||
|     } | ||||
| 
 | ||||
|     template <typename T> | ||||
|     bool LoadObjectFromPrecompiled(T& object) { | ||||
|         return LoadArrayFromPrecompiled(&object, 1); | ||||
|     } | ||||
| 
 | ||||
|     // Stores whole precompiled cache which will be read from or saved to the precompiled chache
 | ||||
|     // file
 | ||||
|     FileSys::VectorVfsFile precompiled_cache_virtual_file; | ||||
|     // Stores the current offset of the precompiled cache file for IO purposes
 | ||||
|     std::size_t precompiled_cache_virtual_file_offset = 0; | ||||
| 
 | ||||
|     // Stored transferable shaders
 | ||||
|     std::unordered_set<u64> stored_transferable; | ||||
| 
 | ||||
|     /// Title ID to operate on
 | ||||
|     u64 title_id = 0; | ||||
| 
 | ||||
|     // The cache has been loaded at boot
 | ||||
|     bool is_usable = false; | ||||
| }; | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 ReinUsesLisp
						ReinUsesLisp