TextureReplacements: Namespace-ify

This commit is contained in:
Stenzek 2024-06-03 00:45:36 +10:00
parent 82b5772d27
commit 87f2995f3d
No known key found for this signature in database
7 changed files with 129 additions and 176 deletions

View file

@ -532,7 +532,7 @@ void GPU::FinishVRAMWrite()
if (g_settings.texture_replacements.ShouldDumpVRAMWrite(m_vram_transfer.width, m_vram_transfer.height)) if (g_settings.texture_replacements.ShouldDumpVRAMWrite(m_vram_transfer.width, m_vram_transfer.height))
{ {
g_texture_replacements.DumpVRAMWrite(m_vram_transfer.width, m_vram_transfer.height, TextureReplacements::DumpVRAMWrite(m_vram_transfer.width, m_vram_transfer.height,
reinterpret_cast<const u16*>(m_blit_buffer.data())); reinterpret_cast<const u16*>(m_blit_buffer.data()));
} }

View file

@ -2578,8 +2578,8 @@ void GPU_HW::LoadVertices()
} }
} }
bool GPU_HW::BlitVRAMReplacementTexture(const TextureReplacementTexture* tex, u32 dst_x, u32 dst_y, u32 width, bool GPU_HW::BlitVRAMReplacementTexture(const TextureReplacements::ReplacementImage* tex, u32 dst_x, u32 dst_y,
u32 height) u32 width, u32 height)
{ {
if (!m_vram_replacement_texture || m_vram_replacement_texture->GetWidth() < tex->GetWidth() || if (!m_vram_replacement_texture || m_vram_replacement_texture->GetWidth() < tex->GetWidth() ||
m_vram_replacement_texture->GetHeight() < tex->GetHeight() || g_gpu_device->GetFeatures().prefer_unused_textures) m_vram_replacement_texture->GetHeight() < tex->GetHeight() || g_gpu_device->GetFeatures().prefer_unused_textures)
@ -2977,7 +2977,7 @@ void GPU_HW::UpdateVRAM(u32 x, u32 y, u32 width, u32 height, const void* data, b
} }
else else
{ {
const TextureReplacementTexture* rtex = g_texture_replacements.GetVRAMWriteReplacement(width, height, data); const TextureReplacements::ReplacementImage* rtex = TextureReplacements::GetVRAMReplacement(width, height, data);
if (rtex && BlitVRAMReplacementTexture(rtex, x * m_resolution_scale, y * m_resolution_scale, if (rtex && BlitVRAMReplacementTexture(rtex, x * m_resolution_scale, y * m_resolution_scale,
width * m_resolution_scale, height * m_resolution_scale)) width * m_resolution_scale, height * m_resolution_scale))
{ {

View file

@ -213,7 +213,8 @@ private:
void UpdateVRAMOnGPU(u32 x, u32 y, u32 width, u32 height, const void* data, u32 data_pitch, bool set_mask, void UpdateVRAMOnGPU(u32 x, u32 y, u32 width, u32 height, const void* data, u32 data_pitch, bool set_mask,
bool check_mask, const GSVector4i bounds); bool check_mask, const GSVector4i bounds);
bool BlitVRAMReplacementTexture(const TextureReplacementTexture* tex, u32 dst_x, u32 dst_y, u32 width, u32 height); bool BlitVRAMReplacementTexture(const TextureReplacements::ReplacementImage* tex, u32 dst_x, u32 dst_y, u32 width,
u32 height);
/// Expands a line into two triangles. /// Expands a line into two triangles.
void DrawLine(float x0, float y0, u32 col0, float x1, float y1, u32 col1, float depth); void DrawLine(float x0, float y0, u32 col0, float x1, float y1, u32 col1, float depth);

View file

@ -427,7 +427,7 @@ DEFINE_HOTKEY("ReloadTextureReplacements", TRANSLATE_NOOP("Hotkeys", "Graphics")
{ {
Host::AddKeyedOSDMessage("ReloadTextureReplacements", Host::AddKeyedOSDMessage("ReloadTextureReplacements",
TRANSLATE_STR("OSDMessage", "Texture replacements reloaded."), 10.0f); TRANSLATE_STR("OSDMessage", "Texture replacements reloaded."), 10.0f);
g_texture_replacements.Reload(); TextureReplacements::Reload();
} }
}) })

View file

@ -1718,7 +1718,7 @@ bool System::BootSystem(SystemBootParameters parameters, Error* error)
// Texture replacement preloading. // Texture replacement preloading.
// TODO: Move this and everything else below OnSystemStarted(). // TODO: Move this and everything else below OnSystemStarted().
g_texture_replacements.SetGameID(s_running_game_serial); TextureReplacements::SetGameID(s_running_game_serial);
// Good to go. // Good to go.
s_state = State::Running; s_state = State::Running;
@ -1901,7 +1901,7 @@ void System::DestroySystem()
ClearMemorySaveStates(); ClearMemorySaveStates();
g_texture_replacements.Shutdown(); TextureReplacements::Shutdown();
PCDrv::Shutdown(); PCDrv::Shutdown();
SIO::Shutdown(); SIO::Shutdown();
@ -3765,7 +3765,7 @@ void System::UpdateRunningGame(const char* path, CDImage* image, bool booting)
} }
if (!booting) if (!booting)
g_texture_replacements.SetGameID(s_running_game_serial); TextureReplacements::SetGameID(s_running_game_serial);
if (booting) if (booting)
Achievements::ResetHardcoreMode(true); Achievements::ResetHardcoreMode(true);
@ -4146,7 +4146,7 @@ void System::CheckForSettingsChanges(const Settings& old_settings)
old_settings.texture_replacements.enable_vram_write_replacements || old_settings.texture_replacements.enable_vram_write_replacements ||
g_settings.texture_replacements.preload_textures != old_settings.texture_replacements.preload_textures) g_settings.texture_replacements.preload_textures != old_settings.texture_replacements.preload_textures)
{ {
g_texture_replacements.Reload(); TextureReplacements::Reload();
} }
DMA::SetMaxSliceTicks(g_settings.dma_max_slice_ticks); DMA::SetMaxSliceTicks(g_settings.dma_max_slice_ticks);

View file

@ -1,12 +1,14 @@
// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin <stenzek@gmail.com> // SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
#include "texture_replacements.h" #include "texture_replacements.h"
#include "gpu_types.h"
#include "host.h" #include "host.h"
#include "settings.h" #include "settings.h"
#include "common/bitutils.h" #include "common/bitutils.h"
#include "common/file_system.h" #include "common/file_system.h"
#include "common/hash_combine.h"
#include "common/log.h" #include "common/log.h"
#include "common/path.h" #include "common/path.h"
#include "common/string_util.h" #include "common/string_util.h"
@ -19,33 +21,72 @@
#endif #endif
#include <cinttypes> #include <cinttypes>
#include <tuple>
#include <unordered_map>
#include <vector>
Log_SetChannel(TextureReplacements); Log_SetChannel(TextureReplacements);
TextureReplacements g_texture_replacements; namespace TextureReplacements {
namespace {
static constexpr u32 VRAMRGBA5551ToRGBA8888(u16 color) struct VRAMReplacementHash
{ {
u8 r = Truncate8(color & 31); u64 low;
u8 g = Truncate8((color >> 5) & 31); u64 high;
u8 b = Truncate8((color >> 10) & 31);
u8 a = Truncate8((color >> 15) & 1);
// 00012345 -> 1234545 TinyString ToString() const;
b = (b << 3) | (b & 0b111); bool ParseString(std::string_view sv);
g = (g << 3) | (g & 0b111);
r = (r << 3) | (r & 0b111);
a = a ? 255 : 0;
return ZeroExtend32(r) | (ZeroExtend32(g) << 8) | (ZeroExtend32(b) << 16) | (ZeroExtend32(a) << 24); bool operator<(const VRAMReplacementHash& rhs) const { return std::tie(low, high) < std::tie(rhs.low, rhs.high); }
bool operator==(const VRAMReplacementHash& rhs) const { return low == rhs.low && high == rhs.high; }
bool operator!=(const VRAMReplacementHash& rhs) const { return low != rhs.low || high != rhs.high; }
};
struct VRAMReplacementHashMapHash
{
size_t operator()(const VRAMReplacementHash& hash) const;
};
} // namespace
using VRAMWriteReplacementMap = std::unordered_map<VRAMReplacementHash, std::string, VRAMReplacementHashMapHash>;
using TextureCache = std::unordered_map<std::string, ReplacementImage>;
static bool ParseReplacementFilename(const std::string& filename, VRAMReplacementHash* replacement_hash,
ReplacmentType* replacement_type);
static std::string GetSourceDirectory();
static std::string GetDumpDirectory();
static VRAMReplacementHash GetVRAMWriteHash(u32 width, u32 height, const void* pixels);
static std::string GetVRAMWriteDumpFilename(u32 width, u32 height, const void* pixels);
static void FindTextures(const std::string& dir);
static const ReplacementImage* LoadTexture(const std::string& filename);
static void PreloadTextures();
static void PurgeUnreferencedTexturesFromCache();
static std::string s_game_id;
// TODO: Check the size, purge some when it gets too large.
static TextureCache s_texture_cache;
static VRAMWriteReplacementMap s_vram_write_replacements;
} // namespace TextureReplacements
size_t TextureReplacements::VRAMReplacementHashMapHash::operator()(const VRAMReplacementHash& hash) const
{
size_t hash_hash = std::hash<u64>{}(hash.low);
hash_combine(hash_hash, hash.high);
return hash_hash;
} }
std::string TextureReplacementHash::ToString() const TinyString TextureReplacements::VRAMReplacementHash::ToString() const
{ {
return StringUtil::StdStringFromFormat("%" PRIx64 "%" PRIx64, high, low); return TinyString::from_format("{:08X}{:08X}", high, low);
} }
bool TextureReplacementHash::ParseString(std::string_view sv) bool TextureReplacements::VRAMReplacementHash::ParseString(std::string_view sv)
{ {
if (sv.length() != 32) if (sv.length() != 32)
return false; return false;
@ -60,25 +101,22 @@ bool TextureReplacementHash::ParseString(std::string_view sv)
return true; return true;
} }
TextureReplacements::TextureReplacements() = default;
TextureReplacements::~TextureReplacements() = default;
void TextureReplacements::SetGameID(std::string game_id) void TextureReplacements::SetGameID(std::string game_id)
{ {
if (m_game_id == game_id) if (s_game_id == game_id)
return; return;
m_game_id = game_id; s_game_id = game_id;
Reload(); Reload();
} }
const TextureReplacementTexture* TextureReplacements::GetVRAMWriteReplacement(u32 width, u32 height, const void* pixels) const TextureReplacements::ReplacementImage* TextureReplacements::GetVRAMReplacement(u32 width, u32 height,
const void* pixels)
{ {
const TextureReplacementHash hash = GetVRAMWriteHash(width, height, pixels); const VRAMReplacementHash hash = GetVRAMWriteHash(width, height, pixels);
const auto it = m_vram_write_replacements.find(hash); const auto it = s_vram_write_replacements.find(hash);
if (it == m_vram_write_replacements.end()) if (it == s_vram_write_replacements.end())
return nullptr; return nullptr;
return LoadTexture(it->second); return LoadTexture(it->second);
@ -86,7 +124,7 @@ const TextureReplacementTexture* TextureReplacements::GetVRAMWriteReplacement(u3
void TextureReplacements::DumpVRAMWrite(u32 width, u32 height, const void* pixels) void TextureReplacements::DumpVRAMWrite(u32 width, u32 height, const void* pixels)
{ {
std::string filename = GetVRAMWriteDumpFilename(width, height, pixels); const std::string filename = GetVRAMWriteDumpFilename(width, height, pixels);
if (filename.empty()) if (filename.empty())
return; return;
@ -120,33 +158,35 @@ void TextureReplacements::DumpVRAMWrite(u32 width, u32 height, const void* pixel
void TextureReplacements::Shutdown() void TextureReplacements::Shutdown()
{ {
m_texture_cache.clear(); s_texture_cache.clear();
m_vram_write_replacements.clear(); s_vram_write_replacements.clear();
m_game_id.clear(); s_game_id.clear();
} }
std::string TextureReplacements::GetSourceDirectory() const // TODO: Organize into PCSX2-style.
std::string TextureReplacements::GetSourceDirectory()
{ {
return Path::Combine(EmuFolders::Textures, m_game_id); return Path::Combine(EmuFolders::Textures, s_game_id);
} }
std::string TextureReplacements::GetDumpDirectory() const std::string TextureReplacements::GetDumpDirectory()
{ {
return Path::Combine(EmuFolders::Dumps, Path::Combine("textures", m_game_id)); return Path::Combine(EmuFolders::Dumps, Path::Combine("textures", s_game_id));
} }
TextureReplacementHash TextureReplacements::GetVRAMWriteHash(u32 width, u32 height, const void* pixels) const TextureReplacements::VRAMReplacementHash TextureReplacements::GetVRAMWriteHash(u32 width, u32 height,
const void* pixels)
{ {
XXH128_hash_t hash = XXH3_128bits(pixels, width * height * sizeof(u16)); XXH128_hash_t hash = XXH3_128bits(pixels, width * height * sizeof(u16));
return {hash.low64, hash.high64}; return {hash.low64, hash.high64};
} }
std::string TextureReplacements::GetVRAMWriteDumpFilename(u32 width, u32 height, const void* pixels) const std::string TextureReplacements::GetVRAMWriteDumpFilename(u32 width, u32 height, const void* pixels)
{ {
if (m_game_id.empty()) if (s_game_id.empty())
return {}; return {};
const TextureReplacementHash hash = GetVRAMWriteHash(width, height, pixels); const VRAMReplacementHash hash = GetVRAMWriteHash(width, height, pixels);
const std::string dump_directory(GetDumpDirectory()); const std::string dump_directory(GetDumpDirectory());
std::string filename(Path::Combine(dump_directory, fmt::format("vram-write-{}.png", hash.ToString()))); std::string filename(Path::Combine(dump_directory, fmt::format("vram-write-{}.png", hash.ToString())));
@ -161,7 +201,7 @@ std::string TextureReplacements::GetVRAMWriteDumpFilename(u32 width, u32 height,
void TextureReplacements::Reload() void TextureReplacements::Reload()
{ {
m_vram_write_replacements.clear(); s_vram_write_replacements.clear();
if (g_settings.texture_replacements.AnyReplacementsEnabled()) if (g_settings.texture_replacements.AnyReplacementsEnabled())
FindTextures(GetSourceDirectory()); FindTextures(GetSourceDirectory());
@ -174,62 +214,41 @@ void TextureReplacements::Reload()
void TextureReplacements::PurgeUnreferencedTexturesFromCache() void TextureReplacements::PurgeUnreferencedTexturesFromCache()
{ {
TextureCache old_map = std::move(m_texture_cache); TextureCache old_map = std::move(s_texture_cache);
for (const auto& it : m_vram_write_replacements) for (const auto& it : s_vram_write_replacements)
{ {
auto it2 = old_map.find(it.second); auto it2 = old_map.find(it.second);
if (it2 != old_map.end()) if (it2 != old_map.end())
{ {
m_texture_cache[it.second] = std::move(it2->second); s_texture_cache[it.second] = std::move(it2->second);
old_map.erase(it2); old_map.erase(it2);
} }
} }
} }
bool TextureReplacements::ParseReplacementFilename(const std::string& filename, bool TextureReplacements::ParseReplacementFilename(const std::string& filename, VRAMReplacementHash* replacement_hash,
TextureReplacementHash* replacement_hash,
ReplacmentType* replacement_type) ReplacmentType* replacement_type)
{ {
const char* extension = std::strrchr(filename.c_str(), '.'); const std::string_view file_title = Path::GetFileTitle(filename);
const char* title = std::strrchr(filename.c_str(), '/'); if (!file_title.starts_with("vram-write-"))
#ifdef _WIN32
const char* title2 = std::strrchr(filename.c_str(), '\\');
if (title2 && (!title || title2 > title))
title = title2;
#endif
if (!title || !extension)
return false; return false;
title++; const std::string_view hashpart = file_title.substr(11);
if (!replacement_hash->ParseString(hashpart))
const char* hashpart;
if (StringUtil::Strncasecmp(title, "vram-write-", 11) == 0)
{
hashpart = title + 11;
*replacement_type = ReplacmentType::VRAMWrite;
}
else
{
return false;
}
if (!replacement_hash->ParseString(std::string_view(hashpart, static_cast<size_t>(extension - hashpart))))
return false; return false;
extension++; const std::string_view file_extension = Path::GetExtension(filename);
bool valid_extension = false; bool valid_extension = false;
for (const char* test_extension : {"png", "jpg", "tga", "bmp"}) for (const char* test_extension : {"png", "jpg", "webp"})
{ {
if (StringUtil::Strcasecmp(extension, test_extension) == 0) if (StringUtil::EqualNoCase(file_extension, test_extension))
{ {
valid_extension = true; valid_extension = true;
break; break;
} }
} }
*replacement_type = ReplacmentType::VRAMWrite;
return valid_extension; return valid_extension;
} }
@ -243,7 +262,7 @@ void TextureReplacements::FindTextures(const std::string& dir)
if (fd.Attributes & FILESYSTEM_FILE_ATTRIBUTE_DIRECTORY) if (fd.Attributes & FILESYSTEM_FILE_ATTRIBUTE_DIRECTORY)
continue; continue;
TextureReplacementHash hash; VRAMReplacementHash hash;
ReplacmentType type; ReplacmentType type;
if (!ParseReplacementFilename(fd.FileName, &hash, &type)) if (!ParseReplacementFilename(fd.FileName, &hash, &type))
continue; continue;
@ -252,26 +271,26 @@ void TextureReplacements::FindTextures(const std::string& dir)
{ {
case ReplacmentType::VRAMWrite: case ReplacmentType::VRAMWrite:
{ {
auto it = m_vram_write_replacements.find(hash); auto it = s_vram_write_replacements.find(hash);
if (it != m_vram_write_replacements.end()) if (it != s_vram_write_replacements.end())
{ {
WARNING_LOG("Duplicate VRAM write replacement: '{}' and '{}'", it->second, fd.FileName); WARNING_LOG("Duplicate VRAM write replacement: '{}' and '{}'", it->second, fd.FileName);
continue; continue;
} }
m_vram_write_replacements.emplace(hash, std::move(fd.FileName)); s_vram_write_replacements.emplace(hash, std::move(fd.FileName));
} }
break; break;
} }
} }
INFO_LOG("Found {} replacement VRAM writes for '{}'", m_vram_write_replacements.size(), m_game_id); INFO_LOG("Found {} replacement VRAM writes for '{}'", s_vram_write_replacements.size(), s_game_id);
} }
const TextureReplacementTexture* TextureReplacements::LoadTexture(const std::string& filename) const TextureReplacements::ReplacementImage* TextureReplacements::LoadTexture(const std::string& filename)
{ {
auto it = m_texture_cache.find(filename); auto it = s_texture_cache.find(filename);
if (it != m_texture_cache.end()) if (it != s_texture_cache.end())
return &it->second; return &it->second;
RGBA8Image image; RGBA8Image image;
@ -282,7 +301,7 @@ const TextureReplacementTexture* TextureReplacements::LoadTexture(const std::str
} }
INFO_LOG("Loaded '{}': {}x{}", Path::GetFileName(filename), image.GetWidth(), image.GetHeight()); INFO_LOG("Loaded '{}': {}x{}", Path::GetFileName(filename), image.GetWidth(), image.GetHeight());
it = m_texture_cache.emplace(filename, std::move(image)).first; it = s_texture_cache.emplace(filename, std::move(image)).first;
return &it->second; return &it->second;
} }
@ -292,7 +311,7 @@ void TextureReplacements::PreloadTextures()
Common::Timer last_update_time; Common::Timer last_update_time;
u32 num_textures_loaded = 0; u32 num_textures_loaded = 0;
const u32 total_textures = static_cast<u32>(m_vram_write_replacements.size()); const u32 total_textures = static_cast<u32>(s_vram_write_replacements.size());
#define UPDATE_PROGRESS() \ #define UPDATE_PROGRESS() \
if (last_update_time.GetTimeSeconds() >= UPDATE_INTERVAL) \ if (last_update_time.GetTimeSeconds() >= UPDATE_INTERVAL) \
@ -302,7 +321,7 @@ void TextureReplacements::PreloadTextures()
last_update_time.Reset(); \ last_update_time.Reset(); \
} }
for (const auto& it : m_vram_write_replacements) for (const auto& it : s_vram_write_replacements)
{ {
UPDATE_PROGRESS(); UPDATE_PROGRESS();

View file

@ -3,95 +3,28 @@
#pragma once #pragma once
#include "util/image.h"
#include "common/hash_combine.h"
#include "types.h" #include "types.h"
#include "util/image.h"
#include <string> #include <string>
#include <tuple>
#include <unordered_map>
#include <vector>
struct TextureReplacementHash namespace TextureReplacements {
{
u64 low;
u64 high;
std::string ToString() const; using ReplacementImage = RGBA8Image;
bool ParseString(std::string_view sv);
bool operator<(const TextureReplacementHash& rhs) const { return std::tie(low, high) < std::tie(rhs.low, rhs.high); }
bool operator==(const TextureReplacementHash& rhs) const { return low == rhs.low && high == rhs.high; }
bool operator!=(const TextureReplacementHash& rhs) const { return low != rhs.low || high != rhs.high; }
};
namespace std {
template<>
struct hash<TextureReplacementHash>
{
size_t operator()(const TextureReplacementHash& h) const
{
size_t hash_hash = std::hash<u64>{}(h.low);
hash_combine(hash_hash, h.high);
return hash_hash;
}
};
} // namespace std
using TextureReplacementTexture = RGBA8Image;
class TextureReplacements
{
public:
enum class ReplacmentType enum class ReplacmentType
{ {
VRAMWrite VRAMWrite,
}; };
TextureReplacements();
~TextureReplacements();
const std::string GetGameID() const { return m_game_id; }
void SetGameID(std::string game_id); void SetGameID(std::string game_id);
void Reload(); void Reload();
const TextureReplacementTexture* GetVRAMWriteReplacement(u32 width, u32 height, const void* pixels); const ReplacementImage* GetVRAMReplacement(u32 width, u32 height, const void* pixels);
void DumpVRAMWrite(u32 width, u32 height, const void* pixels); void DumpVRAMWrite(u32 width, u32 height, const void* pixels);
void Shutdown(); void Shutdown();
private: } // namespace TextureReplacements
struct ReplacementHashMapHash
{
size_t operator()(const TextureReplacementHash& hash);
};
using VRAMWriteReplacementMap = std::unordered_map<TextureReplacementHash, std::string>;
using TextureCache = std::unordered_map<std::string, TextureReplacementTexture>;
static bool ParseReplacementFilename(const std::string& filename, TextureReplacementHash* replacement_hash,
ReplacmentType* replacement_type);
std::string GetSourceDirectory() const;
std::string GetDumpDirectory() const;
TextureReplacementHash GetVRAMWriteHash(u32 width, u32 height, const void* pixels) const;
std::string GetVRAMWriteDumpFilename(u32 width, u32 height, const void* pixels) const;
void FindTextures(const std::string& dir);
const TextureReplacementTexture* LoadTexture(const std::string& filename);
void PreloadTextures();
void PurgeUnreferencedTexturesFromCache();
std::string m_game_id;
TextureCache m_texture_cache;
VRAMWriteReplacementMap m_vram_write_replacements;
};
extern TextureReplacements g_texture_replacements;