/* * Copyright (C) 2007-2009 Gabest * http://www.gabest.org * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GNU Make; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA USA. * http://www.gnu.org/copyleft/gpl.html * */ #pragma once #include "GSRenderer.h" #include "GSFastList.h" #include "GSDirtyRect.h" class GSTextureCache { public: enum {RenderTarget, DepthStencil}; class Surface : public GSAlignedClass<32> { protected: GSRenderer* m_renderer; public: GSTexture* m_texture; GIFRegTEX0 m_TEX0; GIFRegTEXA m_TEXA; int m_age; uint8* m_temp; bool m_32_bits_fmt; // Allow to detect the casting of 32 bits as 16 bits texture bool m_shared_texture; public: Surface(GSRenderer* r, uint8* temp); virtual ~Surface(); void UpdateAge(); }; class Palette { private: uint32* m_clut; // Pointer to a copy of relevant clut GSTexture* m_tex_palette; // Pointer to valid texture with relevant clut as content, if instantiated by the constructor const GSRenderer* m_renderer; // Pointer to the current renderer, needed to recycle the eventually referenced GSTexture on destruction public: Palette(const GSRenderer* renderer, uint16 pal, bool need_gs_texture); // Creates a copy of the current clut and, if needed (need_gs_texture == true), a texture with its content ~Palette(); // Default destructor, frees clut copy and eventually recycles palette texture // Disable copy constructor and copy operator Palette(const Palette&) = delete; Palette& operator=(const Palette&) = delete; // Disable move constructor and move operator Palette(const Palette&&) = delete; Palette& operator=(const Palette&&) = delete; // Getter for clut pointer uint32* GetClut(); // Getter for palette texture pointer, may be nullptr if object has been instantiated with need_gs_texture == false GSTexture* GetPaletteGSTexture(); }; struct PaletteKey { const uint32* clut; uint16 pal; }; struct PaletteKeyHash { // Calculate hash std::size_t operator()(const PaletteKey &key) const; }; struct PaletteKeyEqual { // Compare clut contents bool operator()(const PaletteKey &lhs, const PaletteKey &rhs) const; }; class Source : public Surface { struct {GSVector4i* rect; uint32 count;} m_write; void Write(const GSVector4i& r, int layer); void Flush(uint32 count, int layer); public: std::shared_ptr m_palette_obj; // Shared pointer to the relevant Palette object (if any) GSTexture* m_palette; bool m_should_have_tex_palette; // Enables m_clut (and possibly m_palette) recycling on object destruction uint32 m_valid[MAX_PAGES]; // each uint32 bits map to the 32 blocks of that page uint32* m_clut; bool m_target; bool m_complete; bool m_repeating; bool m_spritehack_t; std::vector* m_p2t; // Keep a trace of the target origin. There is no guarantee that pointer will // still be valid on future. However it ought to be good when the source is created // so it can be used to access un-converted data for the current draw call. GSTexture* m_from_target; GIFRegTEX0 m_layer_TEX0[7]; // Detect already loaded value // Keep a GSTextureCache::SourceMap::m_map iterator to allow fast erase std::array m_erase_it; uint32* m_pages_as_bit; public: Source(GSRenderer* r, const GIFRegTEX0& TEX0, const GIFRegTEXA& TEXA, uint8* temp, bool dummy_container = false); virtual ~Source(); void Update(const GSVector4i& rect, int layer = 0); void UpdateLayer(const GIFRegTEX0& TEX0, const GSVector4i& rect, int layer = 0); }; class Target : public Surface { public: int m_type; bool m_used; GSDirtyRectList m_dirty; GSVector4i m_valid; bool m_depth_supported; bool m_dirty_alpha; uint32 m_end_block; // Hint of the target area public: Target(GSRenderer* r, const GIFRegTEX0& TEX0, uint8* temp, bool depth_supported); void UpdateValidity(const GSVector4i& rect); bool Inside(uint32 bp, uint32 bw, uint32 psm, const GSVector4i& rect); void Update(); }; class PaletteMap { private: static const uint16 MAX_SIZE = 65535; // Max size of each map. const GSRenderer* m_renderer; // Reference to the current renderer // Array of 2 maps, the first for 64B palettes and the second for 1024B palettes. // Each map stores the key PaletteKey (clut copy, pal value) pointing to the relevant shared pointer to Palette object. // There is one PaletteKey per Palette, and the hashing and comparison of PaletteKey is done with custom operators PaletteKeyHash and PaletteKeyEqual. std::array, PaletteKeyHash, PaletteKeyEqual>, 2> m_maps; public: PaletteMap(const GSRenderer* renderer); // Default constructor // Retrieves a shared pointer to a valid Palette from m_maps or creates a new one adding it to the data structure std::shared_ptr LookupPalette(uint16 pal, bool need_gs_texture); void Clear(); // Clears m_maps, thus deletes Palette objects }; class SourceMap { public: std::unordered_set m_surfaces; std::array, MAX_PAGES> m_map; uint32 m_pages[16]; // bitmap of all pages bool m_used; SourceMap() : m_used(false) {memset(m_pages, 0, sizeof(m_pages));} void Add(Source* s, const GIFRegTEX0& TEX0, GSOffset* off); void RemoveAll(); void RemovePartial(); void RemoveAt(Source* s); }; protected: GSRenderer* m_renderer; PaletteMap m_palette_map; SourceMap m_src; FastList m_dst[2]; bool m_paltex; int m_spritehack; bool m_preload_frame; uint8* m_temp; bool m_can_convert_depth; bool m_cpu_fb_conversion; CRCHackLevel m_crc_hack_level; static bool m_disable_partial_invalidation; bool m_texture_inside_rt; static bool m_wrap_gs_mem; virtual Source* CreateSource(const GIFRegTEX0& TEX0, const GIFRegTEXA& TEXA, Target* t = NULL, bool half_right = false, int x_offset = 0, int y_offset = 0); virtual Target* CreateTarget(const GIFRegTEX0& TEX0, int w, int h, int type); virtual int Get8bitFormat() = 0; // TODO: virtual void Write(Source* s, const GSVector4i& r) = 0; // TODO: virtual void Write(Target* t, const GSVector4i& r) = 0; virtual bool CanConvertDepth() { return m_can_convert_depth; } public: GSTextureCache(GSRenderer* r); virtual ~GSTextureCache(); virtual void Read(Target* t, const GSVector4i& r) = 0; virtual void Read(Source* t, const GSVector4i& r) = 0; void RemoveAll(); void RemovePartial(); Source* LookupSource(const GIFRegTEX0& TEX0, const GIFRegTEXA& TEXA, const GSVector4i& r); Source* LookupDepthSource(const GIFRegTEX0& TEX0, const GIFRegTEXA& TEXA, const GSVector4i& r, bool palette = false); Target* LookupTarget(const GIFRegTEX0& TEX0, int w, int h, int type, bool used, uint32 fbmask = 0); Target* LookupTarget(const GIFRegTEX0& TEX0, int w, int h, int real_h); void InvalidateVideoMemType(int type, uint32 bp); void InvalidateVideoMemSubTarget(GSTextureCache::Target* rt); void InvalidateVideoMem(GSOffset* off, const GSVector4i& r, bool target = true); void InvalidateLocalMem(GSOffset* off, const GSVector4i& r); void IncAge(); bool UserHacks_HalfPixelOffset; void ScaleTexture(GSTexture* texture); const char* to_string(int type) { return (type == DepthStencil) ? "Depth" : "Color"; } void PrintMemoryUsage(); void AttachPaletteToSource(Source* s, uint16 pal, bool need_gs_texture); };