2015-05-24 04:55:12 +00:00
|
|
|
// Copyright 2010 Dolphin Emulator Project
|
2015-05-17 23:08:10 +00:00
|
|
|
// Licensed under GPLv2+
|
2013-04-18 03:09:55 +00:00
|
|
|
// Refer to the license.txt file included.
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2014-02-10 18:54:46 +00:00
|
|
|
#pragma once
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2017-03-25 18:45:55 +00:00
|
|
|
#include <array>
|
2010-10-19 22:24:27 +00:00
|
|
|
#include <map>
|
2015-12-21 02:49:49 +00:00
|
|
|
#include <memory>
|
2016-01-24 03:43:01 +00:00
|
|
|
#include <tuple>
|
2015-01-17 09:57:19 +00:00
|
|
|
#include <unordered_map>
|
2016-06-16 09:51:39 +00:00
|
|
|
#include <unordered_set>
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2014-02-17 10:18:15 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
#include "VideoCommon/BPMemory.h"
|
|
|
|
#include "VideoCommon/TextureDecoder.h"
|
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2012-05-28 09:31:37 +00:00
|
|
|
struct VideoConfig;
|
|
|
|
|
2015-11-06 00:28:05 +00:00
|
|
|
class TextureCacheBase
|
2010-10-19 22:24:27 +00:00
|
|
|
{
|
|
|
|
public:
|
2016-06-24 08:43:46 +00:00
|
|
|
struct TCacheEntryConfig
|
|
|
|
{
|
|
|
|
constexpr TCacheEntryConfig() = default;
|
|
|
|
|
|
|
|
bool operator==(const TCacheEntryConfig& o) const
|
|
|
|
{
|
|
|
|
return std::tie(width, height, levels, layers, rendertarget) ==
|
|
|
|
std::tie(o.width, o.height, o.levels, o.layers, o.rendertarget);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Hasher : std::hash<u64>
|
|
|
|
{
|
|
|
|
size_t operator()(const TCacheEntryConfig& c) const
|
|
|
|
{
|
|
|
|
u64 id = (u64)c.rendertarget << 63 | (u64)c.layers << 48 | (u64)c.levels << 32 |
|
|
|
|
(u64)c.height << 16 | (u64)c.width;
|
|
|
|
return std::hash<u64>::operator()(id);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
u32 width = 0;
|
|
|
|
u32 height = 0;
|
|
|
|
u32 levels = 1;
|
|
|
|
u32 layers = 1;
|
|
|
|
bool rendertarget = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TCacheEntryBase
|
|
|
|
{
|
|
|
|
const TCacheEntryConfig config;
|
|
|
|
|
|
|
|
// common members
|
|
|
|
u32 addr;
|
|
|
|
u32 size_in_bytes;
|
|
|
|
u64 base_hash;
|
|
|
|
u64 hash; // for paletted textures, hash = base_hash ^ palette_hash
|
|
|
|
u32 format; // bits 0-3 will contain the in-memory format.
|
2016-12-26 17:41:34 +00:00
|
|
|
u32 memory_stride;
|
2016-06-24 08:43:46 +00:00
|
|
|
bool is_efb_copy;
|
|
|
|
bool is_custom_tex;
|
2016-12-26 17:41:34 +00:00
|
|
|
bool may_have_overlapping_textures;
|
2016-06-24 08:43:46 +00:00
|
|
|
|
|
|
|
unsigned int native_width,
|
|
|
|
native_height; // Texture dimensions from the GameCube's point of view
|
|
|
|
unsigned int native_levels;
|
|
|
|
|
|
|
|
// used to delete textures which haven't been used for TEXTURE_KILL_THRESHOLD frames
|
|
|
|
int frameCount;
|
|
|
|
|
|
|
|
// Keep an iterator to the entry in textures_by_hash, so it does not need to be searched when
|
|
|
|
// removing the cache entry
|
|
|
|
std::multimap<u64, TCacheEntryBase*>::iterator textures_by_hash_iter;
|
|
|
|
|
|
|
|
// 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
|
|
|
|
std::unordered_set<TCacheEntryBase*> references;
|
|
|
|
|
|
|
|
void SetGeneralParameters(u32 _addr, u32 _size, u32 _format)
|
|
|
|
{
|
|
|
|
addr = _addr;
|
|
|
|
size_in_bytes = _size;
|
|
|
|
format = _format;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetDimensions(unsigned int _native_width, unsigned int _native_height,
|
|
|
|
unsigned int _native_levels)
|
|
|
|
{
|
|
|
|
native_width = _native_width;
|
|
|
|
native_height = _native_height;
|
|
|
|
native_levels = _native_levels;
|
|
|
|
memory_stride = _native_width;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetHashes(u64 _base_hash, u64 _hash)
|
|
|
|
{
|
|
|
|
base_hash = _base_hash;
|
|
|
|
hash = _hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This texture entry is used by the other entry as a sub-texture
|
|
|
|
void CreateReference(TCacheEntryBase* other_entry)
|
|
|
|
{
|
|
|
|
// References are two-way, so they can easily be destroyed later
|
|
|
|
this->references.emplace(other_entry);
|
|
|
|
other_entry->references.emplace(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DestroyAllReferences()
|
|
|
|
{
|
|
|
|
for (auto& reference : references)
|
|
|
|
reference->references.erase(this);
|
|
|
|
|
|
|
|
references.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetEfbCopy(u32 stride);
|
|
|
|
|
|
|
|
TCacheEntryBase(const TCacheEntryConfig& c) : config(c) {}
|
|
|
|
virtual ~TCacheEntryBase();
|
|
|
|
|
|
|
|
virtual void Bind(unsigned int stage) = 0;
|
|
|
|
virtual bool Save(const std::string& filename, unsigned int level) = 0;
|
|
|
|
|
|
|
|
virtual void CopyRectangleFromTexture(const TCacheEntryBase* source,
|
|
|
|
const MathUtil::Rectangle<int>& srcrect,
|
|
|
|
const MathUtil::Rectangle<int>& dstrect) = 0;
|
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
virtual void Load(const u8* buffer, u32 width, u32 height, u32 expanded_width, u32 level) = 0;
|
2016-12-26 20:33:57 +00:00
|
|
|
virtual void FromRenderTarget(bool is_depth_copy, const EFBRectangle& srcRect, bool scaleByHalf,
|
|
|
|
unsigned int cbufid, const float* colmat) = 0;
|
2016-06-24 08:43:46 +00:00
|
|
|
|
|
|
|
bool OverlapsMemoryRange(u32 range_address, u32 range_size) const;
|
|
|
|
|
|
|
|
bool IsEfbCopy() const { return is_efb_copy; }
|
|
|
|
u32 NumBlocksY() const;
|
|
|
|
u32 BytesPerRow() const;
|
|
|
|
|
|
|
|
u64 CalculateHash() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
virtual ~TextureCacheBase(); // needs virtual for DX11 dtor
|
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
void OnConfigChanged(VideoConfig& config);
|
2016-06-24 08:43:46 +00:00
|
|
|
|
|
|
|
// Removes textures which aren't used for more than TEXTURE_KILL_THRESHOLD frames,
|
|
|
|
// frameCount is the current frame number.
|
2016-09-06 22:57:58 +00:00
|
|
|
void Cleanup(int _frameCount);
|
2016-06-24 08:43:46 +00:00
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
void Invalidate();
|
2016-06-24 08:43:46 +00:00
|
|
|
|
|
|
|
virtual TCacheEntryBase* CreateTexture(const TCacheEntryConfig& config) = 0;
|
2015-09-04 14:45:29 +00:00
|
|
|
|
2017-04-04 13:55:36 +00:00
|
|
|
virtual void CopyEFB(u8* dst, const EFBCopyFormat& format, u32 native_width, u32 bytes_per_row,
|
|
|
|
u32 num_blocks_y, u32 memory_stride, bool is_depth_copy,
|
|
|
|
const EFBRectangle& src_rect, bool scale_by_half) = 0;
|
2015-09-04 14:45:29 +00:00
|
|
|
|
2016-08-13 12:56:01 +00:00
|
|
|
virtual bool CompileShaders() = 0;
|
|
|
|
virtual void DeleteShaders() = 0;
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
TCacheEntryBase* Load(const u32 stage);
|
|
|
|
void UnbindTextures();
|
2016-06-24 08:43:46 +00:00
|
|
|
virtual void BindTextures();
|
2016-09-06 22:57:58 +00:00
|
|
|
void CopyRenderTargetToTexture(u32 dstAddr, unsigned int dstFormat, u32 dstStride,
|
2016-12-26 19:54:37 +00:00
|
|
|
bool is_depth_copy, const EFBRectangle& srcRect, bool isIntensity,
|
|
|
|
bool scaleByHalf);
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2016-06-24 08:43:46 +00:00
|
|
|
virtual void ConvertTexture(TCacheEntryBase* entry, TCacheEntryBase* unconverted, void* palette,
|
|
|
|
TlutFormat format) = 0;
|
2015-01-26 23:33:23 +00:00
|
|
|
|
2016-11-27 08:14:59 +00:00
|
|
|
// Returns true if the texture data and palette formats are supported by the GPU decoder.
|
|
|
|
virtual bool SupportsGPUTextureDecode(TextureFormat format, TlutFormat palette_format)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decodes the specified data to the GPU texture specified by entry.
|
|
|
|
// 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.
|
|
|
|
virtual void DecodeTextureOnGPU(TCacheEntryBase* entry, u32 dst_level, const u8* data,
|
|
|
|
size_t data_size, TextureFormat format, u32 width, u32 height,
|
|
|
|
u32 aligned_width, u32 aligned_height, u32 row_stride,
|
|
|
|
const u8* palette, TlutFormat palette_format)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-10-19 22:24:27 +00:00
|
|
|
protected:
|
2016-06-24 08:43:46 +00:00
|
|
|
TextureCacheBase();
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
alignas(16) u8* temp = nullptr;
|
|
|
|
size_t temp_size = 0;
|
2010-10-19 22:24:27 +00:00
|
|
|
|
2017-03-25 18:45:55 +00:00
|
|
|
std::array<TCacheEntryBase*, 8> bound_textures{};
|
2016-01-08 03:38:00 +00:00
|
|
|
|
2010-10-19 22:24:27 +00:00
|
|
|
private:
|
2016-12-25 17:48:19 +00:00
|
|
|
typedef std::multimap<u32, TCacheEntryBase*> TexAddrCache;
|
|
|
|
typedef std::multimap<u64, TCacheEntryBase*> TexHashCache;
|
2016-06-24 08:43:46 +00:00
|
|
|
typedef std::unordered_multimap<TCacheEntryConfig, TCacheEntryBase*, TCacheEntryConfig::Hasher>
|
|
|
|
TexPool;
|
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
void SetBackupConfig(const VideoConfig& config);
|
|
|
|
|
|
|
|
TCacheEntryBase* ApplyPaletteToEntry(TCacheEntryBase* entry, u8* palette, u32 tlutfmt);
|
|
|
|
|
|
|
|
void ScaleTextureCacheEntryTo(TCacheEntryBase** entry, u32 new_width, u32 new_height);
|
2016-12-25 17:48:19 +00:00
|
|
|
TCacheEntryBase* DoPartialTextureUpdates(TCacheEntryBase* entry_to_update, u8* palette,
|
|
|
|
u32 tlutfmt);
|
2016-09-06 22:57:58 +00:00
|
|
|
|
|
|
|
void DumpTexture(TCacheEntryBase* entry, std::string basename, unsigned int level);
|
|
|
|
void CheckTempSize(size_t required_size);
|
|
|
|
|
|
|
|
TCacheEntryBase* AllocateTexture(const TCacheEntryConfig& config);
|
|
|
|
TexPool::iterator FindMatchingTextureFromPool(const TCacheEntryConfig& config);
|
2016-12-25 17:48:19 +00:00
|
|
|
TexAddrCache::iterator GetTexCacheIter(TCacheEntryBase* entry);
|
2016-06-24 08:43:46 +00:00
|
|
|
|
2016-12-26 17:41:34 +00: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-24 08:43:46 +00:00
|
|
|
// Removes and unlinks texture from texture cache and returns it to the pool
|
2016-12-25 17:48:19 +00:00
|
|
|
TexAddrCache::iterator InvalidateTexture(TexAddrCache::iterator t_iter);
|
2016-06-24 08:43:46 +00:00
|
|
|
|
2016-09-06 22:57:58 +00:00
|
|
|
TCacheEntryBase* ReturnEntry(unsigned int stage, TCacheEntryBase* entry);
|
2016-06-24 08:43:46 +00:00
|
|
|
|
2016-12-25 17:48:19 +00:00
|
|
|
TexAddrCache textures_by_address;
|
|
|
|
TexHashCache textures_by_hash;
|
2016-09-06 22:57:58 +00:00
|
|
|
TexPool texture_pool;
|
2016-06-24 08:43:46 +00:00
|
|
|
|
|
|
|
// Backup configuration values
|
2016-09-06 22:57:58 +00:00
|
|
|
struct BackupConfig
|
2016-06-24 08:43:46 +00:00
|
|
|
{
|
2016-09-06 22:57:58 +00: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 08:14:59 +00:00
|
|
|
bool gpu_texture_decoding;
|
2016-09-06 22:57:58 +00:00
|
|
|
};
|
|
|
|
BackupConfig backup_config = {};
|
2010-10-19 22:24:27 +00:00
|
|
|
};
|
|
|
|
|
2015-12-21 02:49:49 +00:00
|
|
|
extern std::unique_ptr<TextureCacheBase> g_texture_cache;
|