mirror of
				https://git.suyu.dev/suyu/suyu.git
				synced 2025-10-25 11:56:42 +08:00 
			
		
		
		
	texture_cache: Remove old rasterizer cache
This commit is contained in:
		
							parent
							
								
									2d83553ea7
								
							
						
					
					
						commit
						4db28f72f6
					
				
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -1,575 +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 <memory> | ||||
| #include <string> | ||||
| #include <tuple> | ||||
| #include <vector> | ||||
| 
 | ||||
| #include "common/alignment.h" | ||||
| #include "common/bit_util.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/hash.h" | ||||
| #include "common/math_util.h" | ||||
| #include "video_core/engines/fermi_2d.h" | ||||
| #include "video_core/engines/maxwell_3d.h" | ||||
| #include "video_core/rasterizer_cache.h" | ||||
| #include "video_core/renderer_opengl/gl_resource_manager.h" | ||||
| #include "video_core/renderer_opengl/gl_shader_gen.h" | ||||
| #include "video_core/surface.h" | ||||
| #include "video_core/textures/decoders.h" | ||||
| #include "video_core/textures/texture.h" | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| class CachedSurface; | ||||
| using Surface = std::shared_ptr<CachedSurface>; | ||||
| using SurfaceSurfaceRect_Tuple = std::tuple<Surface, Surface, Common::Rectangle<u32>>; | ||||
| 
 | ||||
| using SurfaceTarget = VideoCore::Surface::SurfaceTarget; | ||||
| using SurfaceType = VideoCore::Surface::SurfaceType; | ||||
| using PixelFormat = VideoCore::Surface::PixelFormat; | ||||
| using ComponentType = VideoCore::Surface::ComponentType; | ||||
| using Maxwell = Tegra::Engines::Maxwell3D::Regs; | ||||
| 
 | ||||
| struct SurfaceParams { | ||||
|     enum class SurfaceClass { | ||||
|         Uploaded, | ||||
|         RenderTarget, | ||||
|         DepthBuffer, | ||||
|         Copy, | ||||
|     }; | ||||
| 
 | ||||
|     static std::string SurfaceTargetName(SurfaceTarget target) { | ||||
|         switch (target) { | ||||
|         case SurfaceTarget::Texture1D: | ||||
|             return "Texture1D"; | ||||
|         case SurfaceTarget::Texture2D: | ||||
|             return "Texture2D"; | ||||
|         case SurfaceTarget::Texture3D: | ||||
|             return "Texture3D"; | ||||
|         case SurfaceTarget::Texture1DArray: | ||||
|             return "Texture1DArray"; | ||||
|         case SurfaceTarget::Texture2DArray: | ||||
|             return "Texture2DArray"; | ||||
|         case SurfaceTarget::TextureCubemap: | ||||
|             return "TextureCubemap"; | ||||
|         case SurfaceTarget::TextureCubeArray: | ||||
|             return "TextureCubeArray"; | ||||
|         default: | ||||
|             LOG_CRITICAL(HW_GPU, "Unimplemented surface_target={}", static_cast<u32>(target)); | ||||
|             UNREACHABLE(); | ||||
|             return fmt::format("TextureUnknown({})", static_cast<u32>(target)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     u32 GetFormatBpp() const { | ||||
|         return VideoCore::Surface::GetFormatBpp(pixel_format); | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the rectangle corresponding to this surface
 | ||||
|     Common::Rectangle<u32> GetRect(u32 mip_level = 0) const; | ||||
| 
 | ||||
|     /// Returns the total size of this surface in bytes, adjusted for compression
 | ||||
|     std::size_t SizeInBytesRaw(bool ignore_tiled = false) const { | ||||
|         const u32 compression_factor{GetCompressionFactor(pixel_format)}; | ||||
|         const u32 bytes_per_pixel{GetBytesPerPixel(pixel_format)}; | ||||
|         const size_t uncompressed_size{ | ||||
|             Tegra::Texture::CalculateSize((ignore_tiled ? false : is_tiled), bytes_per_pixel, width, | ||||
|                                           height, depth, block_height, block_depth)}; | ||||
| 
 | ||||
|         // Divide by compression_factor^2, as height and width are factored by this
 | ||||
|         return uncompressed_size / (compression_factor * compression_factor); | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the size of this surface as an OpenGL texture in bytes
 | ||||
|     std::size_t SizeInBytesGL() const { | ||||
|         return SizeInBytesRaw(true); | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the size of this surface as a cube face in bytes
 | ||||
|     std::size_t SizeInBytesCubeFace() const { | ||||
|         return size_in_bytes / 6; | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the size of this surface as an OpenGL cube face in bytes
 | ||||
|     std::size_t SizeInBytesCubeFaceGL() const { | ||||
|         return size_in_bytes_gl / 6; | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the exact size of memory occupied by the texture in VRAM, including mipmaps.
 | ||||
|     std::size_t MemorySize() const { | ||||
|         std::size_t size = InnerMemorySize(false, is_layered); | ||||
|         if (is_layered) | ||||
|             return size * depth; | ||||
|         return size; | ||||
|     } | ||||
| 
 | ||||
|     /// Returns true if the parameters constitute a valid rasterizer surface.
 | ||||
|     bool IsValid() const { | ||||
|         return gpu_addr && host_ptr && height && width; | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the exact size of the memory occupied by a layer in a texture in VRAM, including
 | ||||
|     /// mipmaps.
 | ||||
|     std::size_t LayerMemorySize() const { | ||||
|         return InnerMemorySize(false, true); | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the size of a layer of this surface in OpenGL.
 | ||||
|     std::size_t LayerSizeGL(u32 mip_level) const { | ||||
|         return InnerMipmapMemorySize(mip_level, true, is_layered, false); | ||||
|     } | ||||
| 
 | ||||
|     std::size_t GetMipmapSizeGL(u32 mip_level, bool ignore_compressed = true) const { | ||||
|         std::size_t size = InnerMipmapMemorySize(mip_level, true, is_layered, ignore_compressed); | ||||
|         if (is_layered) | ||||
|             return size * depth; | ||||
|         return size; | ||||
|     } | ||||
| 
 | ||||
|     std::size_t GetMipmapLevelOffset(u32 mip_level) const { | ||||
|         std::size_t offset = 0; | ||||
|         for (u32 i = 0; i < mip_level; i++) | ||||
|             offset += InnerMipmapMemorySize(i, false, is_layered); | ||||
|         return offset; | ||||
|     } | ||||
| 
 | ||||
|     std::size_t GetMipmapLevelOffsetGL(u32 mip_level) const { | ||||
|         std::size_t offset = 0; | ||||
|         for (u32 i = 0; i < mip_level; i++) | ||||
|             offset += InnerMipmapMemorySize(i, true, is_layered); | ||||
|         return offset; | ||||
|     } | ||||
| 
 | ||||
|     std::size_t GetMipmapSingleSize(u32 mip_level) const { | ||||
|         return InnerMipmapMemorySize(mip_level, false, is_layered); | ||||
|     } | ||||
| 
 | ||||
|     u32 MipWidth(u32 mip_level) const { | ||||
|         return std::max(1U, width >> mip_level); | ||||
|     } | ||||
| 
 | ||||
|     u32 MipWidthGobAligned(u32 mip_level) const { | ||||
|         return Common::AlignUp(std::max(1U, width >> mip_level), 64U * 8U / GetFormatBpp()); | ||||
|     } | ||||
| 
 | ||||
|     u32 MipHeight(u32 mip_level) const { | ||||
|         return std::max(1U, height >> mip_level); | ||||
|     } | ||||
| 
 | ||||
|     u32 MipDepth(u32 mip_level) const { | ||||
|         return is_layered ? depth : std::max(1U, depth >> mip_level); | ||||
|     } | ||||
| 
 | ||||
|     // Auto block resizing algorithm from:
 | ||||
|     // https://cgit.freedesktop.org/mesa/mesa/tree/src/gallium/drivers/nouveau/nv50/nv50_miptree.c
 | ||||
|     u32 MipBlockHeight(u32 mip_level) const { | ||||
|         if (mip_level == 0) | ||||
|             return block_height; | ||||
|         u32 alt_height = MipHeight(mip_level); | ||||
|         u32 h = GetDefaultBlockHeight(pixel_format); | ||||
|         u32 blocks_in_y = (alt_height + h - 1) / h; | ||||
|         u32 bh = 16; | ||||
|         while (bh > 1 && blocks_in_y <= bh * 4) { | ||||
|             bh >>= 1; | ||||
|         } | ||||
|         return bh; | ||||
|     } | ||||
| 
 | ||||
|     u32 MipBlockDepth(u32 mip_level) const { | ||||
|         if (mip_level == 0) { | ||||
|             return block_depth; | ||||
|         } | ||||
| 
 | ||||
|         if (is_layered) { | ||||
|             return 1; | ||||
|         } | ||||
| 
 | ||||
|         const u32 mip_depth = MipDepth(mip_level); | ||||
|         u32 bd = 32; | ||||
|         while (bd > 1 && mip_depth * 2 <= bd) { | ||||
|             bd >>= 1; | ||||
|         } | ||||
| 
 | ||||
|         if (bd == 32) { | ||||
|             const u32 bh = MipBlockHeight(mip_level); | ||||
|             if (bh >= 4) { | ||||
|                 return 16; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         return bd; | ||||
|     } | ||||
| 
 | ||||
|     u32 RowAlign(u32 mip_level) const { | ||||
|         const u32 m_width = MipWidth(mip_level); | ||||
|         const u32 bytes_per_pixel = GetBytesPerPixel(pixel_format); | ||||
|         const u32 l2 = Common::CountTrailingZeroes32(m_width * bytes_per_pixel); | ||||
|         return (1U << l2); | ||||
|     } | ||||
| 
 | ||||
|     /// Creates SurfaceParams from a texture configuration
 | ||||
|     static SurfaceParams CreateForTexture(const Tegra::Texture::FullTextureInfo& config, | ||||
|                                           const GLShader::SamplerEntry& entry); | ||||
| 
 | ||||
|     /// Creates SurfaceParams from a framebuffer configuration
 | ||||
|     static SurfaceParams CreateForFramebuffer(std::size_t index); | ||||
| 
 | ||||
|     /// Creates SurfaceParams for a depth buffer configuration
 | ||||
|     static SurfaceParams CreateForDepthBuffer( | ||||
|         u32 zeta_width, u32 zeta_height, GPUVAddr zeta_address, Tegra::DepthFormat format, | ||||
|         u32 block_width, u32 block_height, u32 block_depth, | ||||
|         Tegra::Engines::Maxwell3D::Regs::InvMemoryLayout type); | ||||
| 
 | ||||
|     /// Creates SurfaceParams for a Fermi2D surface copy
 | ||||
|     static SurfaceParams CreateForFermiCopySurface( | ||||
|         const Tegra::Engines::Fermi2D::Regs::Surface& config); | ||||
| 
 | ||||
|     /// Checks if surfaces are compatible for caching
 | ||||
|     bool IsCompatibleSurface(const SurfaceParams& other) const { | ||||
|         if (std::tie(pixel_format, type, width, height, target, depth, is_tiled) == | ||||
|             std::tie(other.pixel_format, other.type, other.width, other.height, other.target, | ||||
|                      other.depth, other.is_tiled)) { | ||||
|             if (!is_tiled) | ||||
|                 return true; | ||||
|             return std::tie(block_height, block_depth, tile_width_spacing) == | ||||
|                    std::tie(other.block_height, other.block_depth, other.tile_width_spacing); | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     /// Initializes parameters for caching, should be called after everything has been initialized
 | ||||
|     void InitCacheParameters(GPUVAddr gpu_addr); | ||||
| 
 | ||||
|     std::string TargetName() const { | ||||
|         switch (target) { | ||||
|         case SurfaceTarget::Texture1D: | ||||
|             return "1D"; | ||||
|         case SurfaceTarget::TextureBuffer: | ||||
|             return "Buffer"; | ||||
|         case SurfaceTarget::Texture2D: | ||||
|             return "2D"; | ||||
|         case SurfaceTarget::Texture3D: | ||||
|             return "3D"; | ||||
|         case SurfaceTarget::Texture1DArray: | ||||
|             return "1DArray"; | ||||
|         case SurfaceTarget::Texture2DArray: | ||||
|             return "2DArray"; | ||||
|         case SurfaceTarget::TextureCubemap: | ||||
|             return "Cube"; | ||||
|         default: | ||||
|             LOG_CRITICAL(HW_GPU, "Unimplemented surface_target={}", static_cast<u32>(target)); | ||||
|             UNREACHABLE(); | ||||
|             return fmt::format("TUK({})", static_cast<u32>(target)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     std::string ClassName() const { | ||||
|         switch (identity) { | ||||
|         case SurfaceClass::Uploaded: | ||||
|             return "UP"; | ||||
|         case SurfaceClass::RenderTarget: | ||||
|             return "RT"; | ||||
|         case SurfaceClass::DepthBuffer: | ||||
|             return "DB"; | ||||
|         case SurfaceClass::Copy: | ||||
|             return "CP"; | ||||
|         default: | ||||
|             LOG_CRITICAL(HW_GPU, "Unimplemented surface_class={}", static_cast<u32>(identity)); | ||||
|             UNREACHABLE(); | ||||
|             return fmt::format("CUK({})", static_cast<u32>(identity)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     std::string IdentityString() const { | ||||
|         return ClassName() + '_' + TargetName() + '_' + (is_tiled ? 'T' : 'L'); | ||||
|     } | ||||
| 
 | ||||
|     bool is_tiled; | ||||
|     u32 block_width; | ||||
|     u32 block_height; | ||||
|     u32 block_depth; | ||||
|     u32 tile_width_spacing; | ||||
|     PixelFormat pixel_format; | ||||
|     ComponentType component_type; | ||||
|     SurfaceType type; | ||||
|     u32 width; | ||||
|     u32 height; | ||||
|     u32 depth; | ||||
|     u32 unaligned_height; | ||||
|     u32 pitch; | ||||
|     SurfaceTarget target; | ||||
|     SurfaceClass identity; | ||||
|     u32 max_mip_level; | ||||
|     bool is_layered; | ||||
|     bool is_array; | ||||
|     bool srgb_conversion; | ||||
|     // Parameters used for caching
 | ||||
|     u8* host_ptr; | ||||
|     GPUVAddr gpu_addr; | ||||
|     std::size_t size_in_bytes; | ||||
|     std::size_t size_in_bytes_gl; | ||||
| 
 | ||||
|     // Render target specific parameters, not used in caching
 | ||||
|     struct { | ||||
|         u32 index; | ||||
|         u32 array_mode; | ||||
|         u32 volume; | ||||
|         u32 layer_stride; | ||||
|         u32 base_layer; | ||||
|     } rt; | ||||
| 
 | ||||
| private: | ||||
|     std::size_t InnerMipmapMemorySize(u32 mip_level, bool force_gl = false, bool layer_only = false, | ||||
|                                       bool uncompressed = false) const; | ||||
|     std::size_t InnerMemorySize(bool force_gl = false, bool layer_only = false, | ||||
|                                 bool uncompressed = false) const; | ||||
| }; | ||||
| 
 | ||||
| }; // namespace OpenGL
 | ||||
| 
 | ||||
| /// Hashable variation of SurfaceParams, used for a key in the surface cache
 | ||||
| struct SurfaceReserveKey : Common::HashableStruct<OpenGL::SurfaceParams> { | ||||
|     static SurfaceReserveKey Create(const OpenGL::SurfaceParams& params) { | ||||
|         SurfaceReserveKey res; | ||||
|         res.state = params; | ||||
|         res.state.identity = {}; // Ignore the origin of the texture
 | ||||
|         res.state.gpu_addr = {}; // Ignore GPU vaddr in caching
 | ||||
|         res.state.rt = {};       // Ignore rt config in caching
 | ||||
|         return res; | ||||
|     } | ||||
| }; | ||||
| namespace std { | ||||
| template <> | ||||
| struct hash<SurfaceReserveKey> { | ||||
|     std::size_t operator()(const SurfaceReserveKey& k) const { | ||||
|         return k.Hash(); | ||||
|     } | ||||
| }; | ||||
| } // namespace std
 | ||||
| 
 | ||||
| namespace OpenGL { | ||||
| 
 | ||||
| class RasterizerOpenGL; | ||||
| 
 | ||||
| // This is used to store temporary big buffers,
 | ||||
| // instead of creating/destroying all the time
 | ||||
| struct RasterizerTemporaryMemory { | ||||
|     std::vector<std::vector<u8>> gl_buffer; | ||||
| }; | ||||
| 
 | ||||
| class CachedSurface final : public RasterizerCacheObject { | ||||
| public: | ||||
|     explicit CachedSurface(const SurfaceParams& params); | ||||
| 
 | ||||
|     VAddr GetCpuAddr() const override { | ||||
|         return cpu_addr; | ||||
|     } | ||||
| 
 | ||||
|     std::size_t GetSizeInBytes() const override { | ||||
|         return cached_size_in_bytes; | ||||
|     } | ||||
| 
 | ||||
|     std::size_t GetMemorySize() const { | ||||
|         return memory_size; | ||||
|     } | ||||
| 
 | ||||
|     const OGLTexture& Texture() const { | ||||
|         return texture; | ||||
|     } | ||||
| 
 | ||||
|     const OGLTexture& Texture(bool as_array) { | ||||
|         if (params.is_array == as_array) { | ||||
|             return texture; | ||||
|         } else { | ||||
|             EnsureTextureDiscrepantView(); | ||||
|             return discrepant_view; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     GLenum Target() const { | ||||
|         return gl_target; | ||||
|     } | ||||
| 
 | ||||
|     const SurfaceParams& GetSurfaceParams() const { | ||||
|         return params; | ||||
|     } | ||||
| 
 | ||||
|     // Read/Write data in Switch memory to/from gl_buffer
 | ||||
|     void LoadGLBuffer(RasterizerTemporaryMemory& res_cache_tmp_mem); | ||||
|     void FlushGLBuffer(RasterizerTemporaryMemory& res_cache_tmp_mem); | ||||
| 
 | ||||
|     // Upload data in gl_buffer to this surface's texture
 | ||||
|     void UploadGLTexture(RasterizerTemporaryMemory& res_cache_tmp_mem, GLuint read_fb_handle, | ||||
|                          GLuint draw_fb_handle); | ||||
| 
 | ||||
|     void UpdateSwizzle(Tegra::Texture::SwizzleSource swizzle_x, | ||||
|                        Tegra::Texture::SwizzleSource swizzle_y, | ||||
|                        Tegra::Texture::SwizzleSource swizzle_z, | ||||
|                        Tegra::Texture::SwizzleSource swizzle_w); | ||||
| 
 | ||||
|     void MarkReinterpreted() { | ||||
|         reinterpreted = true; | ||||
|     } | ||||
| 
 | ||||
|     bool IsReinterpreted() const { | ||||
|         return reinterpreted; | ||||
|     } | ||||
| 
 | ||||
|     void MarkForReload(bool reload) { | ||||
|         must_reload = reload; | ||||
|     } | ||||
| 
 | ||||
|     bool MustReload() const { | ||||
|         return must_reload; | ||||
|     } | ||||
| 
 | ||||
|     bool IsUploaded() const { | ||||
|         return params.identity == SurfaceParams::SurfaceClass::Uploaded; | ||||
|     } | ||||
| 
 | ||||
| private: | ||||
|     void UploadGLMipmapTexture(RasterizerTemporaryMemory& res_cache_tmp_mem, u32 mip_map, | ||||
|                                GLuint read_fb_handle, GLuint draw_fb_handle); | ||||
| 
 | ||||
|     void EnsureTextureDiscrepantView(); | ||||
| 
 | ||||
|     OGLTexture texture; | ||||
|     OGLTexture discrepant_view; | ||||
|     OGLBuffer texture_buffer; | ||||
|     SurfaceParams params{}; | ||||
|     GLenum gl_target{}; | ||||
|     GLenum gl_internal_format{}; | ||||
|     std::size_t cached_size_in_bytes{}; | ||||
|     std::array<GLenum, 4> swizzle{GL_RED, GL_GREEN, GL_BLUE, GL_ALPHA}; | ||||
|     std::size_t memory_size; | ||||
|     bool reinterpreted = false; | ||||
|     bool must_reload = false; | ||||
|     VAddr cpu_addr{}; | ||||
| }; | ||||
| 
 | ||||
| class RasterizerCacheOpenGL final : public RasterizerCache<Surface> { | ||||
| public: | ||||
|     explicit RasterizerCacheOpenGL(RasterizerOpenGL& rasterizer); | ||||
| 
 | ||||
|     /// Get a surface based on the texture configuration
 | ||||
|     Surface GetTextureSurface(const Tegra::Texture::FullTextureInfo& config, | ||||
|                               const GLShader::SamplerEntry& entry); | ||||
| 
 | ||||
|     /// Get the depth surface based on the framebuffer configuration
 | ||||
|     Surface GetDepthBufferSurface(bool preserve_contents); | ||||
| 
 | ||||
|     /// Get the color surface based on the framebuffer configuration and the specified render target
 | ||||
|     Surface GetColorBufferSurface(std::size_t index, bool preserve_contents); | ||||
| 
 | ||||
|     /// Tries to find a framebuffer using on the provided CPU address
 | ||||
|     Surface TryFindFramebufferSurface(const u8* host_ptr) const; | ||||
| 
 | ||||
|     /// Copies the contents of one surface to another
 | ||||
|     void FermiCopySurface(const Tegra::Engines::Fermi2D::Regs::Surface& src_config, | ||||
|                           const Tegra::Engines::Fermi2D::Regs::Surface& dst_config, | ||||
|                           const Common::Rectangle<u32>& src_rect, | ||||
|                           const Common::Rectangle<u32>& dst_rect); | ||||
| 
 | ||||
|     void SignalPreDrawCall(); | ||||
|     void SignalPostDrawCall(); | ||||
| 
 | ||||
| protected: | ||||
|     void FlushObjectInner(const Surface& object) override { | ||||
|         object->FlushGLBuffer(temporal_memory); | ||||
|     } | ||||
| 
 | ||||
| private: | ||||
|     void LoadSurface(const Surface& surface); | ||||
|     Surface GetSurface(const SurfaceParams& params, bool preserve_contents = true); | ||||
| 
 | ||||
|     /// Gets an uncached surface, creating it if need be
 | ||||
|     Surface GetUncachedSurface(const SurfaceParams& params); | ||||
| 
 | ||||
|     /// Recreates a surface with new parameters
 | ||||
|     Surface RecreateSurface(const Surface& old_surface, const SurfaceParams& new_params); | ||||
| 
 | ||||
|     /// Reserves a unique surface that can be reused later
 | ||||
|     void ReserveSurface(const Surface& surface); | ||||
| 
 | ||||
|     /// Tries to get a reserved surface for the specified parameters
 | ||||
|     Surface TryGetReservedSurface(const SurfaceParams& params); | ||||
| 
 | ||||
|     // Partialy reinterpret a surface based on a triggering_surface that collides with it.
 | ||||
|     // returns true if the reinterpret was successful, false in case it was not.
 | ||||
|     bool PartialReinterpretSurface(Surface triggering_surface, Surface intersect); | ||||
| 
 | ||||
|     /// Performs a slow but accurate surface copy, flushing to RAM and reinterpreting the data
 | ||||
|     void AccurateCopySurface(const Surface& src_surface, const Surface& dst_surface); | ||||
|     void FastLayeredCopySurface(const Surface& src_surface, const Surface& dst_surface); | ||||
|     void FastCopySurface(const Surface& src_surface, const Surface& dst_surface); | ||||
|     void CopySurface(const Surface& src_surface, const Surface& dst_surface, | ||||
|                      const GLuint copy_pbo_handle, const GLenum src_attachment = 0, | ||||
|                      const GLenum dst_attachment = 0, const std::size_t cubemap_face = 0); | ||||
| 
 | ||||
|     /// The surface reserve is a "backup" cache, this is where we put unique surfaces that have
 | ||||
|     /// previously been used. This is to prevent surfaces from being constantly created and
 | ||||
|     /// destroyed when used with different surface parameters.
 | ||||
|     std::unordered_map<SurfaceReserveKey, Surface> surface_reserve; | ||||
| 
 | ||||
|     OGLFramebuffer read_framebuffer; | ||||
|     OGLFramebuffer draw_framebuffer; | ||||
| 
 | ||||
|     bool texception = false; | ||||
| 
 | ||||
|     /// Use a Pixel Buffer Object to download the previous texture and then upload it to the new one
 | ||||
|     /// using the new format.
 | ||||
|     OGLBuffer copy_pbo; | ||||
| 
 | ||||
|     std::array<Surface, Maxwell::NumRenderTargets> last_color_buffers; | ||||
|     std::array<Surface, Maxwell::NumRenderTargets> current_color_buffers; | ||||
|     Surface last_depth_buffer; | ||||
| 
 | ||||
|     RasterizerTemporaryMemory temporal_memory; | ||||
| 
 | ||||
|     using SurfaceIntervalCache = boost::icl::interval_map<CacheAddr, Surface>; | ||||
|     using SurfaceInterval = typename SurfaceIntervalCache::interval_type; | ||||
| 
 | ||||
|     static auto GetReinterpretInterval(const Surface& object) { | ||||
|         return SurfaceInterval::right_open(object->GetCacheAddr() + 1, | ||||
|                                            object->GetCacheAddr() + object->GetMemorySize() - 1); | ||||
|     } | ||||
| 
 | ||||
|     // Reinterpreted surfaces are very fragil as the game may keep rendering into them.
 | ||||
|     SurfaceIntervalCache reinterpreted_surfaces; | ||||
| 
 | ||||
|     void RegisterReinterpretSurface(Surface reinterpret_surface) { | ||||
|         auto interval = GetReinterpretInterval(reinterpret_surface); | ||||
|         reinterpreted_surfaces.insert({interval, reinterpret_surface}); | ||||
|         reinterpret_surface->MarkReinterpreted(); | ||||
|     } | ||||
| 
 | ||||
|     Surface CollideOnReinterpretedSurface(CacheAddr addr) const { | ||||
|         const SurfaceInterval interval{addr}; | ||||
|         for (auto& pair : | ||||
|              boost::make_iterator_range(reinterpreted_surfaces.equal_range(interval))) { | ||||
|             return pair.second; | ||||
|         } | ||||
|         return nullptr; | ||||
|     } | ||||
| 
 | ||||
|     void Register(const Surface& object) override { | ||||
|         RasterizerCache<Surface>::Register(object); | ||||
|     } | ||||
| 
 | ||||
|     /// Unregisters an object from the cache
 | ||||
|     void Unregister(const Surface& object) override { | ||||
|         if (object->IsReinterpreted()) { | ||||
|             auto interval = GetReinterpretInterval(object); | ||||
|             reinterpreted_surfaces.erase(interval); | ||||
|         } | ||||
|         RasterizerCache<Surface>::Unregister(object); | ||||
|     } | ||||
| }; | ||||
| 
 | ||||
| } // namespace OpenGL
 | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user