2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2010 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-18 05:09:55 +02:00
|
|
|
// Refer to the license.txt file included.
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2014-02-10 19:54:46 +01:00
|
|
|
#pragma once
|
2010-10-20 00:24:27 +02:00
|
|
|
|
|
|
|
#include <map>
|
2015-12-21 03:49:49 +01:00
|
|
|
#include <memory>
|
2015-01-17 10:57:19 +01:00
|
|
|
#include <unordered_map>
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2014-02-17 11:18:15 +01:00
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
#include "VideoCommon/BPMemory.h"
|
|
|
|
#include "VideoCommon/TextureDecoder.h"
|
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2012-05-28 11:31:37 +02:00
|
|
|
struct VideoConfig;
|
|
|
|
|
2015-11-06 01:28:05 +01:00
|
|
|
class TextureCacheBase
|
2010-10-20 00:24:27 +02:00
|
|
|
{
|
|
|
|
public:
|
2015-01-11 15:03:41 +01:00
|
|
|
struct TCacheEntryConfig
|
|
|
|
{
|
|
|
|
TCacheEntryConfig() : width(0), height(0), levels(1), layers(1), rendertarget(false) {}
|
|
|
|
|
|
|
|
u32 width, height;
|
|
|
|
u32 levels, layers;
|
|
|
|
bool rendertarget;
|
2015-01-17 10:29:10 +01:00
|
|
|
|
|
|
|
bool operator == (const TCacheEntryConfig& b) const
|
|
|
|
{
|
|
|
|
return width == b.width && height == b.height && levels == b.levels && layers == b.layers && rendertarget == b.rendertarget;
|
|
|
|
}
|
2015-01-17 10:57:19 +01:00
|
|
|
|
|
|
|
struct Hasher : std::hash<u64>
|
|
|
|
{
|
2015-11-06 01:28:05 +01:00
|
|
|
size_t operator()(const TextureCacheBase::TCacheEntryConfig& c) const
|
2015-01-17 10:57:19 +01:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
};
|
2015-01-11 15:03:41 +01:00
|
|
|
};
|
2010-10-20 00:24:27 +02:00
|
|
|
struct TCacheEntryBase
|
|
|
|
{
|
2015-01-11 15:03:41 +01:00
|
|
|
const TCacheEntryConfig config;
|
|
|
|
|
2011-12-26 17:35:27 +01:00
|
|
|
// common members
|
2015-02-20 00:19:31 +01:00
|
|
|
u32 addr;
|
2010-10-20 00:24:27 +02:00
|
|
|
u32 size_in_bytes;
|
2015-09-10 22:28:59 +02:00
|
|
|
u64 base_hash;
|
|
|
|
u64 hash; // for paletted textures, hash = base_hash ^ palette_hash
|
2015-12-29 02:25:31 +01:00
|
|
|
u32 format; // bits 0-3 will contain the in-memory format.
|
2015-02-20 00:19:31 +01:00
|
|
|
bool is_efb_copy;
|
2015-03-01 13:04:48 +01:00
|
|
|
bool is_custom_tex;
|
2015-09-04 16:45:29 +02:00
|
|
|
u32 memory_stride;
|
2010-10-20 02:39:45 +02: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;
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2011-12-26 18:33:05 +01:00
|
|
|
// used to delete textures which haven't been used for TEXTURE_KILL_THRESHOLD frames
|
|
|
|
int frameCount;
|
|
|
|
|
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
|
|
|
|
std::multimap<u64, TCacheEntryBase*>::iterator textures_by_hash_iter;
|
2011-12-26 18:33:05 +01:00
|
|
|
|
2015-02-20 00:19:31 +01:00
|
|
|
void SetGeneralParameters(u32 _addr, u32 _size, u32 _format)
|
2011-12-26 18:05:01 +01:00
|
|
|
{
|
2013-01-29 23:40:15 +01:00
|
|
|
addr = _addr;
|
|
|
|
size_in_bytes = _size;
|
|
|
|
format = _format;
|
2011-12-26 18:05:01 +01: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 23:40:15 +01:00
|
|
|
native_width = _native_width;
|
|
|
|
native_height = _native_height;
|
2015-01-11 22:41:04 +01:00
|
|
|
native_levels = _native_levels;
|
2015-09-04 16:45:29 +02:00
|
|
|
memory_stride = _native_width;
|
2011-12-26 18:05:01 +01: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 23:40:15 +01:00
|
|
|
hash = _hash;
|
2011-12-26 18:05:01 +01:00
|
|
|
}
|
|
|
|
|
2015-09-04 16:45:29 +02:00
|
|
|
void SetEfbCopy(u32 stride);
|
|
|
|
|
2015-01-11 15:03:41 +01:00
|
|
|
TCacheEntryBase(const TCacheEntryConfig& c) : config(c) {}
|
2010-10-20 00:24:27 +02:00
|
|
|
virtual ~TCacheEntryBase();
|
|
|
|
|
|
|
|
virtual void Bind(unsigned int stage) = 0;
|
2014-02-23 23:03:39 +01:00
|
|
|
virtual bool Save(const std::string& filename, unsigned int level) = 0;
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2015-06-30 03:19:19 +02:00
|
|
|
virtual void CopyRectangleFromTexture(
|
|
|
|
const TCacheEntryBase* source,
|
|
|
|
const MathUtil::Rectangle<int> &srcrect,
|
|
|
|
const MathUtil::Rectangle<int> &dstrect) = 0;
|
2015-06-13 15:51:58 +02:00
|
|
|
|
2010-10-20 00:24:27 +02:00
|
|
|
virtual void Load(unsigned int width, unsigned int height,
|
2012-08-10 13:13:51 +02:00
|
|
|
unsigned int expanded_width, unsigned int level) = 0;
|
2015-10-29 18:08:04 +01:00
|
|
|
virtual void FromRenderTarget(u8* dst, PEControl::PixelFormat srcFormat, const EFBRectangle& srcRect,
|
|
|
|
bool scaleByHalf, unsigned int cbufid, const float *colmat) = 0;
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2014-05-16 21:57:14 +02:00
|
|
|
bool OverlapsMemoryRange(u32 range_address, u32 range_size) const;
|
2012-01-29 20:24:23 +01:00
|
|
|
|
2015-04-15 06:38:21 +02:00
|
|
|
bool IsEfbCopy() const { return is_efb_copy; }
|
2015-09-04 16:45:29 +02:00
|
|
|
|
|
|
|
u32 NumBlocksY() const;
|
2015-10-19 18:00:29 +02:00
|
|
|
u32 BytesPerRow() const;
|
2015-09-04 16:45:29 +02:00
|
|
|
|
2015-09-13 13:30:56 +02:00
|
|
|
u64 CalculateHash() const;
|
2010-10-20 00:24:27 +02:00
|
|
|
};
|
|
|
|
|
2015-11-06 01:28:05 +01:00
|
|
|
virtual ~TextureCacheBase(); // needs virtual for DX11 dtor
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2012-05-28 11:31:37 +02:00
|
|
|
static void OnConfigChanged(VideoConfig& config);
|
2014-05-19 18:31:38 +02:00
|
|
|
|
|
|
|
// Removes textures which aren't used for more than TEXTURE_KILL_THRESHOLD frames,
|
|
|
|
// frameCount is the current frame number.
|
2015-03-16 03:28:47 +01:00
|
|
|
static void Cleanup(int _frameCount);
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2012-05-28 11:37:14 +02:00
|
|
|
static void Invalidate();
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2015-01-17 10:01:41 +01:00
|
|
|
virtual TCacheEntryBase* CreateTexture(const TCacheEntryConfig& config) = 0;
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2015-10-29 17:48:35 +01:00
|
|
|
virtual void CopyEFB(u8* dst, u32 format, u32 native_width, u32 bytes_per_row, u32 num_blocks_y, u32 memory_stride,
|
|
|
|
PEControl::PixelFormat srcFormat, const EFBRectangle& srcRect,
|
|
|
|
bool isIntensity, bool scaleByHalf) = 0;
|
|
|
|
|
2014-11-04 00:53:14 +01:00
|
|
|
virtual void CompileShaders() = 0; // currently only implemented by OGL
|
|
|
|
virtual void DeleteShaders() = 0; // currently only implemented by OGL
|
|
|
|
|
2015-01-11 12:48:04 +01:00
|
|
|
static TCacheEntryBase* Load(const u32 stage);
|
2015-01-27 00:33:23 +01:00
|
|
|
static void UnbindTextures();
|
|
|
|
static void BindTextures();
|
2015-08-31 16:41:16 +02:00
|
|
|
static void CopyRenderTargetToTexture(u32 dstAddr, unsigned int dstFormat, u32 dstStride,
|
|
|
|
PEControl::PixelFormat srcFormat, const EFBRectangle& srcRect, bool isIntensity, bool scaleByHalf);
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2015-01-27 00:33:23 +01:00
|
|
|
virtual void ConvertTexture(TCacheEntryBase* entry, TCacheEntryBase* unconverted, void* palette, TlutFormat format) = 0;
|
|
|
|
|
2010-10-20 00:24:27 +02:00
|
|
|
protected:
|
2015-11-06 01:28:05 +01:00
|
|
|
TextureCacheBase();
|
2010-10-20 00:24:27 +02:00
|
|
|
|
2015-09-05 17:44:21 +02:00
|
|
|
alignas(16) static u8* temp;
|
2014-12-22 12:53:03 +01:00
|
|
|
static size_t temp_size;
|
2010-10-20 00:24:27 +02:00
|
|
|
|
|
|
|
private:
|
2015-02-22 18:42:19 +01:00
|
|
|
typedef std::multimap<u64, TCacheEntryBase*> TexCache;
|
|
|
|
typedef std::unordered_multimap<TCacheEntryConfig, TCacheEntryBase*, TCacheEntryConfig::Hasher> TexPool;
|
2015-06-30 03:19:19 +02:00
|
|
|
static TCacheEntryBase* DoPartialTextureUpdates(TexCache::iterator iter);
|
2014-12-22 22:33:38 +01:00
|
|
|
static void DumpTexture(TCacheEntryBase* entry, std::string basename, unsigned int level);
|
2014-12-22 12:53:03 +01:00
|
|
|
static void CheckTempSize(size_t required_size);
|
2012-05-12 13:25:13 +02:00
|
|
|
|
2015-01-17 10:29:10 +01:00
|
|
|
static TCacheEntryBase* AllocateTexture(const TCacheEntryConfig& config);
|
2015-06-09 21:45:09 +02:00
|
|
|
static TexCache::iterator FreeTexture(TexCache::iterator t_iter);
|
2014-06-12 01:04:42 +02:00
|
|
|
|
2015-01-27 00:33:23 +01:00
|
|
|
static TCacheEntryBase* ReturnEntry(unsigned int stage, TCacheEntryBase* entry);
|
|
|
|
|
2015-02-22 18:42:19 +01:00
|
|
|
static TexCache textures_by_address;
|
|
|
|
static TexCache textures_by_hash;
|
2015-01-17 10:57:19 +01:00
|
|
|
static TexPool texture_pool;
|
2015-01-27 00:33:23 +01:00
|
|
|
static TCacheEntryBase* bound_textures[8];
|
2012-05-28 11:31:37 +02:00
|
|
|
|
|
|
|
// Backup configuration values
|
2013-04-24 15:21:54 +02:00
|
|
|
static struct BackupConfig
|
|
|
|
{
|
2012-05-28 11:31:37 +02:00
|
|
|
int s_colorsamples;
|
|
|
|
bool s_texfmt_overlay;
|
|
|
|
bool s_texfmt_overlay_center;
|
|
|
|
bool s_hires_textures;
|
2015-03-01 23:53:15 +01:00
|
|
|
bool s_cache_hires_textures;
|
2012-05-28 11:31:37 +02:00
|
|
|
bool s_copy_cache_enable;
|
2014-11-04 00:53:14 +01:00
|
|
|
bool s_stereo_3d;
|
2014-12-24 23:06:44 +01:00
|
|
|
bool s_efb_mono_depth;
|
2012-05-28 11:31:37 +02:00
|
|
|
} backup_config;
|
2010-10-20 00:24:27 +02:00
|
|
|
};
|
|
|
|
|
2015-12-21 03:49:49 +01:00
|
|
|
extern std::unique_ptr<TextureCacheBase> g_texture_cache;
|