Merge pull request #2160 from degasus/texfilter
TextureCache: load all mipmap levels from custom textures
This commit is contained in:
commit
1b3e3f8141
|
@ -1154,7 +1154,7 @@ void Renderer::SetDitherMode()
|
||||||
// TODO: Set dither mode to bpmem.blendmode.dither
|
// TODO: Set dither mode to bpmem.blendmode.dither
|
||||||
}
|
}
|
||||||
|
|
||||||
void Renderer::SetSamplerState(int stage, int texindex)
|
void Renderer::SetSamplerState(int stage, int texindex, bool custom_tex)
|
||||||
{
|
{
|
||||||
const FourTexUnits &tex = bpmem.tex[texindex];
|
const FourTexUnits &tex = bpmem.tex[texindex];
|
||||||
const TexMode0 &tm0 = tex.texMode0[stage];
|
const TexMode0 &tm0 = tex.texMode0[stage];
|
||||||
|
@ -1179,6 +1179,12 @@ void Renderer::SetSamplerState(int stage, int texindex)
|
||||||
gx_state.sampler[stage].max_lod = (u32)tm1.max_lod;
|
gx_state.sampler[stage].max_lod = (u32)tm1.max_lod;
|
||||||
gx_state.sampler[stage].min_lod = (u32)tm1.min_lod;
|
gx_state.sampler[stage].min_lod = (u32)tm1.min_lod;
|
||||||
gx_state.sampler[stage].lod_bias = (s32)tm0.lod_bias;
|
gx_state.sampler[stage].lod_bias = (s32)tm0.lod_bias;
|
||||||
|
|
||||||
|
// custom textures may have higher resolution, so disable the max_lod
|
||||||
|
if (custom_tex)
|
||||||
|
{
|
||||||
|
gx_state.sampler[stage].max_lod = 255;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Renderer::SetInterlacingMode()
|
void Renderer::SetInterlacingMode()
|
||||||
|
|
|
@ -19,7 +19,7 @@ public:
|
||||||
void SetDepthMode() override;
|
void SetDepthMode() override;
|
||||||
void SetLogicOpMode() override;
|
void SetLogicOpMode() override;
|
||||||
void SetDitherMode() override;
|
void SetDitherMode() override;
|
||||||
void SetSamplerState(int stage,int texindex) override;
|
void SetSamplerState(int stage, int texindex, bool custom_tex) override;
|
||||||
void SetInterlacingMode() override;
|
void SetInterlacingMode() override;
|
||||||
void SetViewport() override;
|
void SetViewport() override;
|
||||||
|
|
||||||
|
|
|
@ -1868,13 +1868,13 @@ void Renderer::SetDitherMode()
|
||||||
glDisable(GL_DITHER);
|
glDisable(GL_DITHER);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Renderer::SetSamplerState(int stage, int texindex)
|
void Renderer::SetSamplerState(int stage, int texindex, bool custom_tex)
|
||||||
{
|
{
|
||||||
auto const& tex = bpmem.tex[texindex];
|
auto const& tex = bpmem.tex[texindex];
|
||||||
auto const& tm0 = tex.texMode0[stage];
|
auto const& tm0 = tex.texMode0[stage];
|
||||||
auto const& tm1 = tex.texMode1[stage];
|
auto const& tm1 = tex.texMode1[stage];
|
||||||
|
|
||||||
g_sampler_cache->SetSamplerState((texindex * 4) + stage, tm0, tm1);
|
g_sampler_cache->SetSamplerState((texindex * 4) + stage, tm0, tm1, custom_tex);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Renderer::SetInterlacingMode()
|
void Renderer::SetInterlacingMode()
|
||||||
|
|
|
@ -58,7 +58,7 @@ public:
|
||||||
void SetDepthMode() override;
|
void SetDepthMode() override;
|
||||||
void SetLogicOpMode() override;
|
void SetLogicOpMode() override;
|
||||||
void SetDitherMode() override;
|
void SetDitherMode() override;
|
||||||
void SetSamplerState(int stage,int texindex) override;
|
void SetSamplerState(int stage, int texindex, bool custom_tex) override;
|
||||||
void SetInterlacingMode() override;
|
void SetInterlacingMode() override;
|
||||||
void SetViewport() override;
|
void SetViewport() override;
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,7 @@ SamplerCache::~SamplerCache()
|
||||||
Clear();
|
Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void SamplerCache::SetSamplerState(int stage, const TexMode0& tm0, const TexMode1& tm1)
|
void SamplerCache::SetSamplerState(int stage, const TexMode0& tm0, const TexMode1& tm1, bool custom_tex)
|
||||||
{
|
{
|
||||||
// TODO: can this go somewhere else?
|
// TODO: can this go somewhere else?
|
||||||
if (m_last_max_anisotropy != g_ActiveConfig.iMaxAnisotropy)
|
if (m_last_max_anisotropy != g_ActiveConfig.iMaxAnisotropy)
|
||||||
|
@ -38,6 +38,12 @@ void SamplerCache::SetSamplerState(int stage, const TexMode0& tm0, const TexMode
|
||||||
params.tm0.mag_filter |= 0x1;
|
params.tm0.mag_filter |= 0x1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// custom textures may have higher resolution, so disable the max_lod
|
||||||
|
if (custom_tex)
|
||||||
|
{
|
||||||
|
params.tm1.max_lod = 255;
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: Should keep a circular buffer for each stage of recently used samplers.
|
// TODO: Should keep a circular buffer for each stage of recently used samplers.
|
||||||
|
|
||||||
auto& active_sampler = m_active_samplers[stage];
|
auto& active_sampler = m_active_samplers[stage];
|
||||||
|
|
|
@ -14,7 +14,7 @@ public:
|
||||||
SamplerCache();
|
SamplerCache();
|
||||||
~SamplerCache();
|
~SamplerCache();
|
||||||
|
|
||||||
void SetSamplerState(int stage, const TexMode0& tm0, const TexMode1& tm1);
|
void SetSamplerState(int stage, const TexMode0& tm0, const TexMode1& tm1, bool custom_tex);
|
||||||
void Clear();
|
void Clear();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -134,7 +134,7 @@ void PixelShaderManager::SetDestAlpha()
|
||||||
dirty = true;
|
dirty = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PixelShaderManager::SetTexDims(int texmapid, u32 width, u32 height, u32 wraps, u32 wrapt)
|
void PixelShaderManager::SetTexDims(int texmapid, u32 width, u32 height)
|
||||||
{
|
{
|
||||||
// TODO: move this check out to callee. There we could just call this function on texture changes
|
// TODO: move this check out to callee. There we could just call this function on texture changes
|
||||||
// or better, use textureSize() in glsl
|
// or better, use textureSize() in glsl
|
||||||
|
|
|
@ -33,7 +33,7 @@ public:
|
||||||
static void SetTevKonstColor(int index, int component, s32 value);
|
static void SetTevKonstColor(int index, int component, s32 value);
|
||||||
static void SetAlpha();
|
static void SetAlpha();
|
||||||
static void SetDestAlpha();
|
static void SetDestAlpha();
|
||||||
static void SetTexDims(int texmapid, u32 width, u32 height, u32 wraps, u32 wrapt);
|
static void SetTexDims(int texmapid, u32 width, u32 height);
|
||||||
static void SetZTextureBias();
|
static void SetZTextureBias();
|
||||||
static void SetViewportChanged();
|
static void SetViewportChanged();
|
||||||
static void SetEfbScaleChanged();
|
static void SetEfbScaleChanged();
|
||||||
|
|
|
@ -57,7 +57,7 @@ public:
|
||||||
virtual void SetDepthMode() = 0;
|
virtual void SetDepthMode() = 0;
|
||||||
virtual void SetLogicOpMode() = 0;
|
virtual void SetLogicOpMode() = 0;
|
||||||
virtual void SetDitherMode() = 0;
|
virtual void SetDitherMode() = 0;
|
||||||
virtual void SetSamplerState(int stage,int texindex) = 0;
|
virtual void SetSamplerState(int stage, int texindex, bool custom_tex) = 0;
|
||||||
virtual void SetInterlacingMode() = 0;
|
virtual void SetInterlacingMode() = 0;
|
||||||
virtual void SetViewport() = 0;
|
virtual void SetViewport() = 0;
|
||||||
|
|
||||||
|
|
|
@ -266,8 +266,8 @@ TextureCache::TCacheEntryBase* TextureCache::Load(const u32 stage)
|
||||||
const int texformat = tex.texImage0[id].format;
|
const int texformat = tex.texImage0[id].format;
|
||||||
const u32 tlutaddr = tex.texTlut[id].tmem_offset << 9;
|
const u32 tlutaddr = tex.texTlut[id].tmem_offset << 9;
|
||||||
const u32 tlutfmt = tex.texTlut[id].tlut_format;
|
const u32 tlutfmt = tex.texTlut[id].tlut_format;
|
||||||
u32 tex_levels = (tex.texMode1[id].max_lod + 0xf) / 0x10 + 1;
|
const bool use_mipmaps = (tex.texMode0[id].min_filter & 3) != 0;
|
||||||
const bool use_mipmaps = (tex.texMode0[id].min_filter & 3) != 0 && tex_levels > 0;
|
u32 tex_levels = use_mipmaps ? ((tex.texMode1[id].max_lod + 0xf) / 0x10 + 1) : 1;
|
||||||
const bool from_tmem = tex.texImage1[id].image_type != 0;
|
const bool from_tmem = tex.texImage1[id].image_type != 0;
|
||||||
|
|
||||||
if (0 == address)
|
if (0 == address)
|
||||||
|
@ -472,11 +472,8 @@ TextureCache::TCacheEntryBase* TextureCache::Load(const u32 stage)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 texLevels = use_mipmaps ? tex_levels : 1;
|
// how many levels the allocated texture shall have
|
||||||
const bool using_custom_lods = hires_tex && hires_tex->m_levels.size() >= texLevels;
|
const u32 texLevels = hires_tex ? (u32)hires_tex->m_levels.size() : tex_levels;
|
||||||
// Only load native mips if their dimensions fit to our virtual texture dimensions
|
|
||||||
const bool use_native_mips = use_mipmaps && !using_custom_lods && (width == nativeW && height == nativeH);
|
|
||||||
texLevels = (use_native_mips || using_custom_lods) ? texLevels : 1; // TODO: Should be forced to 1 for non-pow2 textures (e.g. efb copies with automatically adjusted IR)
|
|
||||||
|
|
||||||
// create the entry/texture
|
// create the entry/texture
|
||||||
TCacheEntryConfig config;
|
TCacheEntryConfig config;
|
||||||
|
@ -493,6 +490,7 @@ TextureCache::TCacheEntryBase* TextureCache::Load(const u32 stage)
|
||||||
entry->SetDimensions(nativeW, nativeH, tex_levels);
|
entry->SetDimensions(nativeW, nativeH, tex_levels);
|
||||||
entry->hash = tex_hash ^ tlut_hash;
|
entry->hash = tex_hash ^ tlut_hash;
|
||||||
entry->is_efb_copy = false;
|
entry->is_efb_copy = false;
|
||||||
|
entry->is_custom_tex = hires_tex != nullptr;
|
||||||
|
|
||||||
// load texture
|
// load texture
|
||||||
entry->Load(width, height, expandedWidth, 0);
|
entry->Load(width, height, expandedWidth, 0);
|
||||||
|
@ -510,10 +508,19 @@ TextureCache::TCacheEntryBase* TextureCache::Load(const u32 stage)
|
||||||
DumpTexture(entry, basename, 0);
|
DumpTexture(entry, basename, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 level = 1;
|
if (hires_tex)
|
||||||
// load mips - TODO: Loading mipmaps from tmem is untested!
|
|
||||||
if (use_native_mips)
|
|
||||||
{
|
{
|
||||||
|
for (u32 level = 1; level != texLevels; ++level)
|
||||||
|
{
|
||||||
|
auto& l = hires_tex->m_levels[level];
|
||||||
|
CheckTempSize(l.data_size);
|
||||||
|
memcpy(temp, l.data, l.data_size);
|
||||||
|
entry->Load(l.width, l.height, l.width, level);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// load mips - TODO: Loading mipmaps from tmem is untested!
|
||||||
src_data += texture_size;
|
src_data += texture_size;
|
||||||
|
|
||||||
const u8* ptr_even = nullptr;
|
const u8* ptr_even = nullptr;
|
||||||
|
@ -524,7 +531,7 @@ TextureCache::TCacheEntryBase* TextureCache::Load(const u32 stage)
|
||||||
ptr_odd = &texMem[bpmem.tex[stage / 4].texImage2[stage % 4].tmem_odd * TMEM_LINE_SIZE];
|
ptr_odd = &texMem[bpmem.tex[stage / 4].texImage2[stage % 4].tmem_odd * TMEM_LINE_SIZE];
|
||||||
}
|
}
|
||||||
|
|
||||||
for (; level != texLevels; ++level)
|
for (u32 level = 1; level != texLevels; ++level)
|
||||||
{
|
{
|
||||||
const u32 mip_width = CalculateLevelSize(width, level);
|
const u32 mip_width = CalculateLevelSize(width, level);
|
||||||
const u32 mip_height = CalculateLevelSize(height, level);
|
const u32 mip_height = CalculateLevelSize(height, level);
|
||||||
|
@ -544,16 +551,6 @@ TextureCache::TCacheEntryBase* TextureCache::Load(const u32 stage)
|
||||||
DumpTexture(entry, basename, level);
|
DumpTexture(entry, basename, level);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (using_custom_lods)
|
|
||||||
{
|
|
||||||
for (; level != texLevels; ++level)
|
|
||||||
{
|
|
||||||
auto& l = hires_tex->m_levels[level];
|
|
||||||
CheckTempSize(l.data_size);
|
|
||||||
memcpy(temp, l.data, l.data_size);
|
|
||||||
entry->Load(l.width, l.height, l.width, level);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
INCSTAT(stats.numTexturesUploaded);
|
INCSTAT(stats.numTexturesUploaded);
|
||||||
SETSTAT(stats.numTexturesAlive, textures.size());
|
SETSTAT(stats.numTexturesAlive, textures.size());
|
||||||
|
@ -872,6 +869,7 @@ void TextureCache::CopyRenderTargetToTexture(u32 dstAddr, unsigned int dstFormat
|
||||||
|
|
||||||
entry->frameCount = FRAMECOUNT_INVALID;
|
entry->frameCount = FRAMECOUNT_INVALID;
|
||||||
entry->is_efb_copy = true;
|
entry->is_efb_copy = true;
|
||||||
|
entry->is_custom_tex = false;
|
||||||
|
|
||||||
entry->FromRenderTarget(dstAddr, dstFormat, srcFormat, srcRect, isIntensity, scaleByHalf, cbufid, colmat);
|
entry->FromRenderTarget(dstAddr, dstFormat, srcFormat, srcRect, isIntensity, scaleByHalf, cbufid, colmat);
|
||||||
|
|
||||||
|
|
|
@ -52,6 +52,7 @@ public:
|
||||||
u64 hash;
|
u64 hash;
|
||||||
u32 format;
|
u32 format;
|
||||||
bool is_efb_copy;
|
bool is_efb_copy;
|
||||||
|
bool is_custom_tex;
|
||||||
|
|
||||||
unsigned int native_width, native_height; // Texture dimensions from the GameCube's point of view
|
unsigned int native_width, native_height; // Texture dimensions from the GameCube's point of view
|
||||||
unsigned int native_levels;
|
unsigned int native_levels;
|
||||||
|
|
|
@ -210,13 +210,12 @@ void VertexManager::Flush()
|
||||||
TextureCache::UnbindTextures();
|
TextureCache::UnbindTextures();
|
||||||
for (unsigned int i : usedtextures)
|
for (unsigned int i : usedtextures)
|
||||||
{
|
{
|
||||||
g_renderer->SetSamplerState(i & 3, i >> 2);
|
|
||||||
const TextureCache::TCacheEntryBase* tentry = TextureCache::Load(i);
|
const TextureCache::TCacheEntryBase* tentry = TextureCache::Load(i);
|
||||||
|
|
||||||
if (tentry)
|
if (tentry)
|
||||||
{
|
{
|
||||||
// 0s are probably for no manual wrapping needed.
|
g_renderer->SetSamplerState(i & 3, i >> 2, tentry->is_custom_tex);
|
||||||
PixelShaderManager::SetTexDims(i, tentry->native_width, tentry->native_height, 0, 0);
|
PixelShaderManager::SetTexDims(i, tentry->native_width, tentry->native_height);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue