2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2010 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2017-03-25 14:45:55 -04:00
|
|
|
#include <array>
|
2017-06-29 23:09:32 +02:00
|
|
|
#include <bitset>
|
2010-10-19 22:24:27 +00:00
|
|
|
#include <map>
|
2015-12-20 21:49:49 -05:00
|
|
|
#include <memory>
|
2017-09-29 00:32:04 -05:00
|
|
|
#include <optional>
|
|
|
|
#include <string>
|
2016-01-23 22:43:01 -05:00
|
|
|
#include <tuple>
|
2015-01-17 10:57:19 +01:00
|
|
|
#include <unordered_map>
|
2016-06-16 21:51:39 +12:00
|
|
|
#include <unordered_set>
|
2018-11-03 00:17:00 +10:00
|
|
|
#include <vector>
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2019-07-16 20:18:48 -04:00
|
|
|
#include "Common/MathUtil.h"
|
2017-04-22 23:44:34 -05:00
|
|
|
#include "VideoCommon/AbstractTexture.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/BPMemory.h"
|
2017-04-22 23:44:34 -05:00
|
|
|
#include "VideoCommon/TextureConfig.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/TextureDecoder.h"
|
2021-05-01 00:59:24 -05:00
|
|
|
#include "VideoCommon/TextureInfo.h"
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
class AbstractFramebuffer;
|
2018-11-03 00:17:00 +10:00
|
|
|
class AbstractStagingTexture;
|
2019-06-29 19:27:53 +10:00
|
|
|
class PointerWrap;
|
2019-07-16 20:18:48 -04:00
|
|
|
struct VideoConfig;
|
2012-05-28 11:31:37 +02:00
|
|
|
|
2017-07-30 12:45:55 -07:00
|
|
|
struct TextureAndTLUTFormat
|
|
|
|
{
|
|
|
|
TextureAndTLUTFormat(TextureFormat texfmt_ = TextureFormat::I4,
|
|
|
|
TLUTFormat tlutfmt_ = TLUTFormat::IA8)
|
|
|
|
: texfmt(texfmt_), tlutfmt(tlutfmt_)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const TextureAndTLUTFormat& other) const
|
|
|
|
{
|
|
|
|
if (IsColorIndexed(texfmt))
|
|
|
|
return texfmt == other.texfmt && tlutfmt == other.tlutfmt;
|
|
|
|
|
|
|
|
return texfmt == other.texfmt;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const TextureAndTLUTFormat& other) const { return !operator==(other); }
|
|
|
|
TextureFormat texfmt;
|
|
|
|
TLUTFormat tlutfmt;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct EFBCopyParams
|
|
|
|
{
|
2021-02-10 18:11:31 -08:00
|
|
|
EFBCopyParams(PixelFormat efb_format_, EFBCopyFormat copy_format_, bool depth_, bool yuv_,
|
|
|
|
bool copy_filter_)
|
2018-05-03 14:09:32 +10:00
|
|
|
: efb_format(efb_format_), copy_format(copy_format_), depth(depth_), yuv(yuv_),
|
|
|
|
copy_filter(copy_filter_)
|
2017-07-30 12:45:55 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator<(const EFBCopyParams& rhs) const
|
|
|
|
{
|
2018-05-03 14:09:32 +10:00
|
|
|
return std::tie(efb_format, copy_format, depth, yuv, copy_filter) <
|
|
|
|
std::tie(rhs.efb_format, rhs.copy_format, rhs.depth, rhs.yuv, rhs.copy_filter);
|
2017-07-30 12:45:55 -07:00
|
|
|
}
|
|
|
|
|
2021-02-10 18:11:31 -08:00
|
|
|
PixelFormat efb_format;
|
2017-07-30 12:45:55 -07:00
|
|
|
EFBCopyFormat copy_format;
|
|
|
|
bool depth;
|
|
|
|
bool yuv;
|
2018-05-03 14:09:32 +10:00
|
|
|
bool copy_filter;
|
2017-07-30 12:45:55 -07:00
|
|
|
};
|
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
// Reduced version of the full coefficient array, with a single value for each row.
|
|
|
|
struct EFBCopyFilterCoefficients
|
|
|
|
{
|
|
|
|
float upper;
|
|
|
|
float middle;
|
|
|
|
float lower;
|
|
|
|
};
|
|
|
|
|
2015-11-06 01:28:05 +01:00
|
|
|
class TextureCacheBase
|
2010-10-19 22:24:27 +00:00
|
|
|
{
|
2017-04-22 23:44:34 -05:00
|
|
|
private:
|
|
|
|
static const int FRAMECOUNT_INVALID = 0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-04-22 23:44:34 -05:00
|
|
|
public:
|
|
|
|
struct TCacheEntry
|
2010-10-19 22:24:27 +00:00
|
|
|
{
|
2011-12-26 17:35:27 +01:00
|
|
|
// common members
|
2017-04-22 23:44:34 -05:00
|
|
|
std::unique_ptr<AbstractTexture> texture;
|
2019-02-15 11:59:50 +10:00
|
|
|
std::unique_ptr<AbstractFramebuffer> framebuffer;
|
2015-02-19 15:19:31 -08:00
|
|
|
u32 addr;
|
2010-10-19 22:24:27 +00:00
|
|
|
u32 size_in_bytes;
|
2015-09-10 22:28:59 +02:00
|
|
|
u64 base_hash;
|
2017-07-30 12:45:55 -07:00
|
|
|
u64 hash; // for paletted textures, hash = base_hash ^ palette_hash
|
|
|
|
TextureAndTLUTFormat format;
|
2016-12-26 18:41:34 +01:00
|
|
|
u32 memory_stride;
|
2015-02-19 15:19:31 -08:00
|
|
|
bool is_efb_copy;
|
2015-03-01 13:04:48 +01:00
|
|
|
bool is_custom_tex;
|
2017-04-22 23:44:34 -05:00
|
|
|
bool may_have_overlapping_textures = true;
|
2017-10-18 02:02:56 -07:00
|
|
|
bool tmem_only = false; // indicates that this texture only exists in the tmem cache
|
|
|
|
bool has_arbitrary_mips = false; // indicates that the mips in this texture are arbitrary
|
|
|
|
// content, aren't just downscaled
|
2017-05-29 17:02:09 -05:00
|
|
|
bool should_force_safe_hashing = false; // for XFB
|
2017-06-26 12:06:50 -05:00
|
|
|
bool is_xfb_copy = false;
|
2019-03-31 14:11:53 +10:00
|
|
|
bool is_xfb_container = false;
|
2017-08-20 14:24:37 -05:00
|
|
|
u64 id;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-10-01 11:19:29 -05:00
|
|
|
bool reference_changed = false; // used by xfb to determine when a reference xfb changed
|
2017-09-29 00:32:04 -05:00
|
|
|
|
2011-12-26 17:35:27 +01:00
|
|
|
unsigned int native_width,
|
|
|
|
native_height; // Texture dimensions from the GameCube's point of view
|
2015-01-11 22:41:04 +01:00
|
|
|
unsigned int native_levels;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-12-26 18:33:05 +01:00
|
|
|
// used to delete textures which haven't been used for TEXTURE_KILL_THRESHOLD frames
|
2017-04-22 23:44:34 -05:00
|
|
|
int frameCount = FRAMECOUNT_INVALID;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-02-22 18:42:19 +01:00
|
|
|
// Keep an iterator to the entry in textures_by_hash, so it does not need to be searched when
|
|
|
|
// removing the cache entry
|
2017-04-22 23:44:34 -05:00
|
|
|
std::multimap<u64, TCacheEntry*>::iterator textures_by_hash_iter;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-06-16 21:51:39 +12:00
|
|
|
// This is used to keep track of both:
|
|
|
|
// * efb copies used by this partially updated texture
|
|
|
|
// * partially updated textures which refer to this efb copy
|
2017-04-22 23:44:34 -05:00
|
|
|
std::unordered_set<TCacheEntry*> references;
|
|
|
|
|
2018-11-03 00:17:00 +10:00
|
|
|
// Pending EFB copy
|
|
|
|
std::unique_ptr<AbstractStagingTexture> pending_efb_copy;
|
|
|
|
u32 pending_efb_copy_width = 0;
|
|
|
|
u32 pending_efb_copy_height = 0;
|
|
|
|
bool pending_efb_copy_invalidated = false;
|
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
explicit TCacheEntry(std::unique_ptr<AbstractTexture> tex,
|
|
|
|
std::unique_ptr<AbstractFramebuffer> fb);
|
2017-04-22 23:44:34 -05:00
|
|
|
|
|
|
|
~TCacheEntry();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-09-02 21:30:34 -05:00
|
|
|
void SetGeneralParameters(u32 _addr, u32 _size, TextureAndTLUTFormat _format,
|
|
|
|
bool force_safe_hashing)
|
2011-12-26 18:05:01 +01:00
|
|
|
{
|
2013-01-29 16:40:15 -06:00
|
|
|
addr = _addr;
|
|
|
|
size_in_bytes = _size;
|
|
|
|
format = _format;
|
2017-05-29 17:02:09 -05:00
|
|
|
should_force_safe_hashing = force_safe_hashing;
|
2011-12-26 18:05:01 +01:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-11 22:41:04 +01:00
|
|
|
void SetDimensions(unsigned int _native_width, unsigned int _native_height,
|
|
|
|
unsigned int _native_levels)
|
2011-12-26 18:05:01 +01:00
|
|
|
{
|
2013-01-29 16:40:15 -06:00
|
|
|
native_width = _native_width;
|
|
|
|
native_height = _native_height;
|
2015-01-11 22:41:04 +01:00
|
|
|
native_levels = _native_levels;
|
2015-09-05 02:45:29 +12:00
|
|
|
memory_stride = _native_width;
|
2011-12-26 18:05:01 +01:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-09-10 22:28:59 +02:00
|
|
|
void SetHashes(u64 _base_hash, u64 _hash)
|
2011-12-26 18:05:01 +01:00
|
|
|
{
|
2015-09-10 22:28:59 +02:00
|
|
|
base_hash = _base_hash;
|
2013-01-29 16:40:15 -06:00
|
|
|
hash = _hash;
|
2011-12-26 18:05:01 +01:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-06-16 21:51:39 +12:00
|
|
|
// This texture entry is used by the other entry as a sub-texture
|
2017-04-22 23:44:34 -05:00
|
|
|
void CreateReference(TCacheEntry* other_entry)
|
2016-06-16 21:51:39 +12:00
|
|
|
{
|
2016-06-18 03:28:57 +12:00
|
|
|
// References are two-way, so they can easily be destroyed later
|
2016-06-16 21:51:39 +12:00
|
|
|
this->references.emplace(other_entry);
|
|
|
|
other_entry->references.emplace(this);
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-06-26 12:06:50 -05:00
|
|
|
void SetXfbCopy(u32 stride);
|
2015-10-29 17:48:35 +01:00
|
|
|
void SetEfbCopy(u32 stride);
|
2017-06-26 12:06:50 -05:00
|
|
|
void SetNotCopy();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-06-29 22:19:19 -03:00
|
|
|
bool OverlapsMemoryRange(u32 range_address, u32 range_size) const;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-06-29 22:19:19 -03:00
|
|
|
bool IsEfbCopy() const { return is_efb_copy; }
|
2017-06-26 12:06:50 -05:00
|
|
|
bool IsCopy() const { return is_xfb_copy || is_efb_copy; }
|
2015-06-29 22:19:19 -03:00
|
|
|
u32 NumBlocksY() const;
|
2015-10-19 12:00:29 -04:00
|
|
|
u32 BytesPerRow() const;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-06-29 22:19:19 -03:00
|
|
|
u64 CalculateHash() const;
|
2017-04-22 23:44:34 -05:00
|
|
|
|
2017-05-29 17:02:09 -05:00
|
|
|
int HashSampleSize() const;
|
2017-04-22 23:44:34 -05:00
|
|
|
u32 GetWidth() const { return texture->GetConfig().width; }
|
|
|
|
u32 GetHeight() const { return texture->GetConfig().height; }
|
|
|
|
u32 GetNumLevels() const { return texture->GetConfig().levels; }
|
|
|
|
u32 GetNumLayers() const { return texture->GetConfig().layers; }
|
2017-06-12 12:37:28 -05:00
|
|
|
AbstractTextureFormat GetFormat() const { return texture->GetConfig().format; }
|
2019-06-29 19:27:53 +10:00
|
|
|
void DoState(PointerWrap& p);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Minimal version of TCacheEntry just for TexPool
|
|
|
|
struct TexPoolEntry
|
|
|
|
{
|
|
|
|
std::unique_ptr<AbstractTexture> texture;
|
|
|
|
std::unique_ptr<AbstractFramebuffer> framebuffer;
|
|
|
|
int frameCount = FRAMECOUNT_INVALID;
|
|
|
|
|
|
|
|
TexPoolEntry(std::unique_ptr<AbstractTexture> tex, std::unique_ptr<AbstractFramebuffer> fb);
|
2012-08-10 13:13:51 +02:00
|
|
|
};
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
TextureCacheBase();
|
|
|
|
virtual ~TextureCacheBase();
|
|
|
|
|
|
|
|
bool Initialize();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-08-04 22:31:25 -04:00
|
|
|
void OnConfigChanged(const VideoConfig& config);
|
2019-08-17 14:40:58 -05:00
|
|
|
void ForceReload();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-05-16 21:57:14 +02:00
|
|
|
// Removes textures which aren't used for more than TEXTURE_KILL_THRESHOLD frames,
|
|
|
|
// frameCount is the current frame number.
|
2016-09-06 18:57:58 -04:00
|
|
|
void Cleanup(int _frameCount);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-09-06 18:57:58 -04:00
|
|
|
void Invalidate();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-04-22 23:44:34 -05:00
|
|
|
TCacheEntry* Load(const u32 stage);
|
2017-06-29 23:09:32 +02:00
|
|
|
static void InvalidateAllBindPoints() { valid_bind_points.reset(); }
|
|
|
|
static bool IsValidBindPoint(u32 i) { return valid_bind_points.test(i); }
|
2021-05-01 00:59:24 -05:00
|
|
|
TCacheEntry* GetTexture(const int textureCacheSafetyColorSampleSize, TextureInfo& texture_info);
|
2019-03-31 14:11:53 +10:00
|
|
|
TCacheEntry* GetXFBTexture(u32 address, u32 width, u32 height, u32 stride,
|
|
|
|
MathUtil::Rectangle<int>* display_rect);
|
2017-09-29 00:32:04 -05:00
|
|
|
|
2017-05-29 17:02:09 -05:00
|
|
|
virtual void BindTextures();
|
2017-12-23 12:44:01 +01:00
|
|
|
void CopyRenderTargetToTexture(u32 dstAddr, EFBCopyFormat dstFormat, u32 width, u32 height,
|
2019-04-16 00:47:46 +10:00
|
|
|
u32 dstStride, bool is_depth_copy,
|
|
|
|
const MathUtil::Rectangle<int>& srcRect, bool isIntensity,
|
|
|
|
bool scaleByHalf, float y_scale, float gamma, bool clamp_top,
|
|
|
|
bool clamp_bottom,
|
2018-04-29 18:52:30 +10:00
|
|
|
const CopyFilterCoefficients::Values& filter_coefficients);
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2017-07-02 21:24:20 -05:00
|
|
|
void ScaleTextureCacheEntryTo(TCacheEntry* entry, u32 new_width, u32 new_height);
|
|
|
|
|
2018-11-03 00:17:00 +10:00
|
|
|
// Flushes all pending EFB copies to emulated RAM.
|
|
|
|
void FlushEFBCopies();
|
|
|
|
|
2019-06-29 19:27:53 +10:00
|
|
|
// Texture Serialization
|
|
|
|
void SerializeTexture(AbstractTexture* tex, const TextureConfig& config, PointerWrap& p);
|
|
|
|
std::optional<TexPoolEntry> DeserializeTexture(PointerWrap& p);
|
|
|
|
|
|
|
|
// Save States
|
|
|
|
void DoState(PointerWrap& p);
|
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
// Returns false if the top/bottom row coefficients are zero.
|
|
|
|
static bool NeedsCopyFilterInShader(const EFBCopyFilterCoefficients& coefficients);
|
2018-11-03 00:17:00 +10:00
|
|
|
|
2010-10-19 22:24:27 +00:00
|
|
|
protected:
|
2019-02-15 11:59:50 +10:00
|
|
|
// Decodes the specified data to the GPU texture specified by entry.
|
|
|
|
// Returns false if the configuration is not supported.
|
|
|
|
// width, height are the size of the image in pixels.
|
|
|
|
// aligned_width, aligned_height are the size of the image in pixels, aligned to the block size.
|
|
|
|
// row_stride is the number of bytes for a row of blocks, not pixels.
|
|
|
|
bool DecodeTextureOnGPU(TCacheEntry* entry, u32 dst_level, const u8* data, u32 data_size,
|
|
|
|
TextureFormat format, u32 width, u32 height, u32 aligned_width,
|
|
|
|
u32 aligned_height, u32 row_stride, const u8* palette,
|
|
|
|
TLUTFormat palette_format);
|
|
|
|
|
|
|
|
virtual void CopyEFB(AbstractStagingTexture* dst, const EFBCopyParams& params, u32 native_width,
|
|
|
|
u32 bytes_per_row, u32 num_blocks_y, u32 memory_stride,
|
2019-04-16 00:47:46 +10:00
|
|
|
const MathUtil::Rectangle<int>& src_rect, bool scale_by_half,
|
|
|
|
bool linear_filter, float y_scale, float gamma, bool clamp_top,
|
|
|
|
bool clamp_bottom, const EFBCopyFilterCoefficients& filter_coefficients);
|
2019-02-15 11:59:50 +10:00
|
|
|
virtual void CopyEFBToCacheEntry(TCacheEntry* entry, bool is_depth_copy,
|
2019-04-16 00:47:46 +10:00
|
|
|
const MathUtil::Rectangle<int>& src_rect, bool scale_by_half,
|
2019-03-31 14:42:38 +10:00
|
|
|
bool linear_filter, EFBCopyFormat dst_format, bool is_intensity,
|
|
|
|
float gamma, bool clamp_top, bool clamp_bottom,
|
2019-02-15 11:59:50 +10:00
|
|
|
const EFBCopyFilterCoefficients& filter_coefficients);
|
2018-05-03 14:09:32 +10:00
|
|
|
|
2016-09-06 18:57:58 -04:00
|
|
|
alignas(16) u8* temp = nullptr;
|
|
|
|
size_t temp_size = 0;
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2017-04-22 23:44:34 -05:00
|
|
|
std::array<TCacheEntry*, 8> bound_textures{};
|
2017-06-29 23:09:32 +02:00
|
|
|
static std::bitset<8> valid_bind_points;
|
2016-01-07 19:38:00 -08:00
|
|
|
|
2010-10-19 22:24:27 +00:00
|
|
|
private:
|
2017-10-08 20:15:34 -04:00
|
|
|
using TexAddrCache = std::multimap<u32, TCacheEntry*>;
|
|
|
|
using TexHashCache = std::multimap<u64, TCacheEntry*>;
|
|
|
|
using TexPool = std::unordered_multimap<TextureConfig, TexPoolEntry>;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
bool CreateUtilityTextures();
|
|
|
|
|
2016-09-06 18:57:58 -04:00
|
|
|
void SetBackupConfig(const VideoConfig& config);
|
|
|
|
|
2021-05-06 20:55:39 -05:00
|
|
|
TCacheEntry* GetXFBFromCache(u32 address, u32 width, u32 height, u32 stride);
|
2019-03-31 14:11:53 +10:00
|
|
|
|
2021-05-01 00:59:24 -05:00
|
|
|
TCacheEntry* ApplyPaletteToEntry(TCacheEntry* entry, const u8* palette, TLUTFormat tlutfmt);
|
2016-09-06 18:57:58 -04:00
|
|
|
|
2019-07-14 15:24:12 +10:00
|
|
|
TCacheEntry* ReinterpretEntry(const TCacheEntry* existing_entry, TextureFormat new_format);
|
|
|
|
|
2021-05-01 00:59:24 -05:00
|
|
|
TCacheEntry* DoPartialTextureUpdates(TCacheEntry* entry_to_update, const u8* palette,
|
2017-07-30 12:45:55 -07:00
|
|
|
TLUTFormat tlutfmt);
|
2019-03-31 14:11:53 +10:00
|
|
|
void StitchXFBCopy(TCacheEntry* entry_to_update);
|
2016-09-06 18:57:58 -04:00
|
|
|
|
2017-10-18 02:02:56 -07:00
|
|
|
void DumpTexture(TCacheEntry* entry, std::string basename, unsigned int level, bool is_arbitrary);
|
2016-09-06 18:57:58 -04:00
|
|
|
void CheckTempSize(size_t required_size);
|
|
|
|
|
2017-04-22 23:44:34 -05:00
|
|
|
TCacheEntry* AllocateCacheEntry(const TextureConfig& config);
|
2019-02-15 11:59:50 +10:00
|
|
|
std::optional<TexPoolEntry> AllocateTexture(const TextureConfig& config);
|
2017-04-22 23:44:34 -05:00
|
|
|
TexPool::iterator FindMatchingTextureFromPool(const TextureConfig& config);
|
|
|
|
TexAddrCache::iterator GetTexCacheIter(TCacheEntry* entry);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2016-12-26 18:41:34 +01:00
|
|
|
// Return all possible overlapping textures. As addr+size of the textures is not
|
|
|
|
// indexed, this may return false positives.
|
|
|
|
std::pair<TexAddrCache::iterator, TexAddrCache::iterator>
|
|
|
|
FindOverlappingTextures(u32 addr, u32 size_in_bytes);
|
|
|
|
|
2016-06-18 03:28:57 +12:00
|
|
|
// Removes and unlinks texture from texture cache and returns it to the pool
|
2018-11-03 00:17:00 +10:00
|
|
|
TexAddrCache::iterator InvalidateTexture(TexAddrCache::iterator t_iter,
|
|
|
|
bool discard_pending_efb_copy = false);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-11-04 12:04:26 -05:00
|
|
|
void UninitializeXFBMemory(u8* dst, u32 stride, u32 bytes_per_row, u32 num_blocks_y);
|
|
|
|
|
2018-04-29 18:52:30 +10:00
|
|
|
// Precomputing the coefficients for the previous, current, and next lines for the copy filter.
|
2019-02-15 11:59:50 +10:00
|
|
|
static EFBCopyFilterCoefficients
|
|
|
|
GetRAMCopyFilterCoefficients(const CopyFilterCoefficients::Values& coefficients);
|
|
|
|
static EFBCopyFilterCoefficients
|
|
|
|
GetVRAMCopyFilterCoefficients(const CopyFilterCoefficients::Values& coefficients);
|
2018-04-29 18:52:30 +10:00
|
|
|
|
2018-11-03 00:17:00 +10:00
|
|
|
// Flushes a pending EFB copy to RAM from the host to the guest RAM.
|
|
|
|
void WriteEFBCopyToRAM(u8* dst_ptr, u32 width, u32 height, u32 stride,
|
|
|
|
std::unique_ptr<AbstractStagingTexture> staging_texture);
|
|
|
|
void FlushEFBCopy(TCacheEntry* entry);
|
|
|
|
|
|
|
|
// Returns a staging texture of the maximum EFB copy size.
|
|
|
|
std::unique_ptr<AbstractStagingTexture> GetEFBCopyStagingTexture();
|
|
|
|
|
|
|
|
// Returns an EFB copy staging texture to the pool, so it can be re-used.
|
|
|
|
void ReleaseEFBCopyStagingTexture(std::unique_ptr<AbstractStagingTexture> tex);
|
|
|
|
|
2019-06-29 19:27:53 +10:00
|
|
|
bool CheckReadbackTexture(u32 width, u32 height, AbstractTextureFormat format);
|
|
|
|
void DoSaveState(PointerWrap& p);
|
|
|
|
void DoLoadState(PointerWrap& p);
|
|
|
|
|
2016-12-25 18:48:19 +01:00
|
|
|
TexAddrCache textures_by_address;
|
|
|
|
TexHashCache textures_by_hash;
|
2016-09-06 18:57:58 -04:00
|
|
|
TexPool texture_pool;
|
2017-08-20 14:24:37 -05:00
|
|
|
u64 last_entry_id = 0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2012-05-28 11:31:37 +02:00
|
|
|
// Backup configuration values
|
2016-09-06 18:57:58 -04:00
|
|
|
struct BackupConfig
|
2012-05-28 11:31:37 +02:00
|
|
|
{
|
2016-09-06 18:57:58 -04:00
|
|
|
int color_samples;
|
|
|
|
bool texfmt_overlay;
|
|
|
|
bool texfmt_overlay_center;
|
|
|
|
bool hires_textures;
|
|
|
|
bool cache_hires_textures;
|
|
|
|
bool copy_cache_enable;
|
|
|
|
bool stereo_3d;
|
|
|
|
bool efb_mono_depth;
|
2016-11-27 18:14:59 +10:00
|
|
|
bool gpu_texture_decoding;
|
2018-05-03 14:24:44 +10:00
|
|
|
bool disable_vram_copies;
|
2018-07-03 06:19:07 -04:00
|
|
|
bool arbitrary_mipmap_detection;
|
2016-09-06 18:57:58 -04:00
|
|
|
};
|
|
|
|
BackupConfig backup_config = {};
|
2018-11-03 00:17:00 +10:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
// Encoding texture used for EFB copies to RAM.
|
|
|
|
std::unique_ptr<AbstractTexture> m_efb_encoding_texture;
|
|
|
|
std::unique_ptr<AbstractFramebuffer> m_efb_encoding_framebuffer;
|
|
|
|
|
|
|
|
// Decoding texture used for GPU texture decoding.
|
|
|
|
std::unique_ptr<AbstractTexture> m_decoding_texture;
|
|
|
|
|
2018-11-03 00:17:00 +10:00
|
|
|
// Pool of readback textures used for deferred EFB copies.
|
|
|
|
std::vector<std::unique_ptr<AbstractStagingTexture>> m_efb_copy_staging_texture_pool;
|
|
|
|
|
|
|
|
// List of pending EFB copies. It is important that the order is preserved for these,
|
|
|
|
// so that overlapping textures are written to guest RAM in the order they are issued.
|
|
|
|
std::vector<TCacheEntry*> m_pending_efb_copies;
|
2019-06-29 19:27:53 +10:00
|
|
|
|
|
|
|
// Staging texture used for readbacks.
|
|
|
|
// We store this in the class so that the same staging texture can be used for multiple
|
|
|
|
// readbacks, saving the overhead of allocating a new buffer every time.
|
|
|
|
std::unique_ptr<AbstractStagingTexture> m_readback_texture;
|
2010-10-19 22:24:27 +00:00
|
|
|
};
|
|
|
|
|
2015-12-20 21:49:49 -05:00
|
|
|
extern std::unique_ptr<TextureCacheBase> g_texture_cache;
|