| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | // Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2019-11-27 05:51:13 -05:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | #include <type_traits>
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | #include <unordered_map>
 | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "common/hash.h"
 | 
					
						
							|  |  |  | #include "video_core/engines/const_buffer_engine_interface.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | #include "video_core/engines/maxwell_3d.h"
 | 
					
						
							| 
									
										
										
										
											2019-11-18 18:35:21 -03:00
										 |  |  | #include "video_core/engines/shader_type.h"
 | 
					
						
							| 
									
										
										
										
											2020-01-03 16:16:29 -04:00
										 |  |  | #include "video_core/guest_driver.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace VideoCommon::Shader { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  | using KeyMap = std::unordered_map<std::pair<u32, u32>, u32, Common::PairHash>; | 
					
						
							|  |  |  | using BoundSamplerMap = std::unordered_map<u32, Tegra::Engines::SamplerDescriptor>; | 
					
						
							|  |  |  | using BindlessSamplerMap = | 
					
						
							|  |  |  |     std::unordered_map<std::pair<u32, u32>, Tegra::Engines::SamplerDescriptor, Common::PairHash>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | struct GraphicsInfo { | 
					
						
							|  |  |  |     Tegra::Engines::Maxwell3D::Regs::PrimitiveTopology primitive_topology{}; | 
					
						
							| 
									
										
										
										
											2020-02-29 04:03:22 -03:00
										 |  |  |     Tegra::Engines::Maxwell3D::Regs::TessellationPrimitive tessellation_primitive{}; | 
					
						
							|  |  |  |     Tegra::Engines::Maxwell3D::Regs::TessellationSpacing tessellation_spacing{}; | 
					
						
							|  |  |  |     bool tessellation_clockwise = false; | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-02-29 04:03:22 -03:00
										 |  |  | static_assert(std::is_trivially_copyable_v<GraphicsInfo> && | 
					
						
							|  |  |  |               std::is_standard_layout_v<GraphicsInfo>); | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct ComputeInfo { | 
					
						
							|  |  |  |     std::array<u32, 3> workgroup_size{}; | 
					
						
							|  |  |  |     u32 shared_memory_size_in_words = 0; | 
					
						
							|  |  |  |     u32 local_memory_size_in_words = 0; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-02-29 04:03:22 -03:00
										 |  |  | static_assert(std::is_trivially_copyable_v<ComputeInfo> && std::is_standard_layout_v<ComputeInfo>); | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct SerializedRegistryInfo { | 
					
						
							|  |  |  |     VideoCore::GuestDriverProfile guest_driver_profile; | 
					
						
							|  |  |  |     u32 bound_buffer = 0; | 
					
						
							|  |  |  |     GraphicsInfo graphics; | 
					
						
							|  |  |  |     ComputeInfo compute; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-17 10:35:16 -04:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  |  * The Registry is a class use to interface the 3D and compute engines with the shader compiler. | 
					
						
							|  |  |  |  * With it, the shader can obtain required data from GPU state and store it for disk shader | 
					
						
							|  |  |  |  * compilation. | 
					
						
							| 
									
										
										
										
											2019-11-18 18:35:21 -03:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  | class Registry { | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  |     explicit Registry(Tegra::Engines::ShaderType shader_stage, const SerializedRegistryInfo& info); | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  |     explicit Registry(Tegra::Engines::ShaderType shader_stage, | 
					
						
							|  |  |  |                       Tegra::Engines::ConstBufferEngineInterface& engine); | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  |     ~Registry(); | 
					
						
							| 
									
										
										
										
											2019-10-17 10:35:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  |     /// Retrieves a key from the registry, if it's registered, it will give the registered value, if
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// not it will obtain it from maxwell3d and register it.
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  |     std::optional<u32> ObtainKey(u32 buffer, u32 offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  |     std::optional<Tegra::Engines::SamplerDescriptor> ObtainBoundSampler(u32 offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::optional<Tegra::Engines::SamplerDescriptor> ObtainBindlessSampler(u32 buffer, u32 offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// Inserts a key.
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  |     void InsertKey(u32 buffer, u32 offset, u32 value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// Inserts a bound sampler key.
 | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  |     void InsertBoundSampler(u32 offset, Tegra::Engines::SamplerDescriptor sampler); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// Inserts a bindless sampler key.
 | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  |     void InsertBindlessSampler(u32 buffer, u32 offset, Tegra::Engines::SamplerDescriptor sampler); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  |     /// Checks keys and samplers against engine's current const buffers.
 | 
					
						
							|  |  |  |     /// Returns true if they are the same value, false otherwise.
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     bool IsConsistent() const; | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-28 20:53:10 -03:00
										 |  |  |     /// Returns true if the keys are equal to the other ones in the registry.
 | 
					
						
							|  |  |  |     bool HasEqualKeys(const Registry& rhs) const; | 
					
						
							| 
									
										
										
										
											2019-09-26 00:23:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// Gives an getter to the const buffer keys in the database.
 | 
					
						
							|  |  |  |     const KeyMap& GetKeys() const { | 
					
						
							|  |  |  |         return keys; | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// Gets samplers database.
 | 
					
						
							|  |  |  |     const BoundSamplerMap& GetBoundSamplers() const { | 
					
						
							|  |  |  |         return bound_samplers; | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     /// Gets bindless samplers database.
 | 
					
						
							|  |  |  |     const BindlessSamplerMap& GetBindlessSamplers() const { | 
					
						
							|  |  |  |         return bindless_samplers; | 
					
						
							| 
									
										
										
										
											2019-09-25 09:53:18 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-24 10:44:34 -04:00
										 |  |  |     /// Gets bound buffer used on this shader
 | 
					
						
							| 
									
										
										
										
											2020-01-03 18:15:24 -04:00
										 |  |  |     u32 GetBoundBuffer() const { | 
					
						
							|  |  |  |         return bound_buffer; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  |     /// Returns compute information from this shader
 | 
					
						
							|  |  |  |     const GraphicsInfo& GetGraphicsInfo() const { | 
					
						
							|  |  |  |         ASSERT(stage != Tegra::Engines::ShaderType::Compute); | 
					
						
							|  |  |  |         return graphics_info; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Returns compute information from this shader
 | 
					
						
							|  |  |  |     const ComputeInfo& GetComputeInfo() const { | 
					
						
							|  |  |  |         ASSERT(stage == Tegra::Engines::ShaderType::Compute); | 
					
						
							|  |  |  |         return compute_info; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-24 10:44:34 -04:00
										 |  |  |     /// Obtains access to the guest driver's profile.
 | 
					
						
							| 
									
										
										
										
											2020-02-26 16:13:47 -03:00
										 |  |  |     VideoCore::GuestDriverProfile& AccessGuestDriverProfile() { | 
					
						
							|  |  |  |         return engine ? engine->AccessGuestDriverProfile() : stored_guest_driver_profile; | 
					
						
							| 
									
										
										
										
											2020-01-03 16:16:29 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     const Tegra::Engines::ShaderType stage; | 
					
						
							| 
									
										
										
										
											2020-02-26 16:13:47 -03:00
										 |  |  |     VideoCore::GuestDriverProfile stored_guest_driver_profile; | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  |     Tegra::Engines::ConstBufferEngineInterface* engine = nullptr; | 
					
						
							|  |  |  |     KeyMap keys; | 
					
						
							|  |  |  |     BoundSamplerMap bound_samplers; | 
					
						
							|  |  |  |     BindlessSamplerMap bindless_samplers; | 
					
						
							| 
									
										
										
										
											2020-02-29 03:49:51 -03:00
										 |  |  |     u32 bound_buffer; | 
					
						
							|  |  |  |     GraphicsInfo graphics_info; | 
					
						
							|  |  |  |     ComputeInfo compute_info; | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-09-25 19:19:41 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:02:02 -04:00
										 |  |  | } // namespace VideoCommon::Shader
 |