ADD: Texture Cache 2

This commit is contained in:
2025-11-03 18:44:42 +09:00
parent c64db86f82
commit 266b54560e
2 changed files with 53 additions and 8 deletions

View File

@@ -34,7 +34,23 @@ void TextureCache::cleanup()
TextureCache::TextureHandle TextureCache::request(const TextureKey &key, VkSampler sampler)
{
auto it = _lookup.find(key.hash);
// Ensure we have a valid, stable hash for deduplication.
TextureKey normKey = key;
if (normKey.hash == 0)
{
if (normKey.kind == TextureKey::SourceKind::FilePath)
{
std::string id = std::string("PATH:") + normKey.path + (normKey.srgb ? "#sRGB" : "#UNORM");
normKey.hash = texcache::fnv1a64(id);
}
else if (!normKey.bytes.empty())
{
uint64_t h = texcache::fnv1a64(normKey.bytes.data(), normKey.bytes.size());
normKey.hash = h ^ (normKey.srgb ? 0x9E3779B97F4A7C15ull : 0ull);
}
}
auto it = _lookup.find(normKey.hash);
if (it != _lookup.end())
{
TextureHandle h = it->second;
@@ -47,19 +63,19 @@ TextureCache::TextureHandle TextureCache::request(const TextureKey &key, VkSampl
}
TextureHandle h = static_cast<TextureHandle>(_entries.size());
_lookup.emplace(key.hash, h);
_lookup.emplace(normKey.hash, h);
Entry e{};
e.key = key;
e.key = normKey;
e.sampler = sampler;
e.state = EntryState::Unloaded;
if (key.kind == TextureKey::SourceKind::FilePath)
if (normKey.kind == TextureKey::SourceKind::FilePath)
{
e.path = key.path;
e.path = normKey.path;
}
else
{
e.bytes = key.bytes;
e.bytes = normKey.bytes;
}
_entries.push_back(std::move(e));
return h;
@@ -198,14 +214,27 @@ void TextureCache::pumpLoads(ResourceManager &rm, FrameResources &)
// Simple throttle to avoid massive spikes.
const int kMaxLoadsPerPump = 4;
int started = 0;
const uint32_t now = _context ? _context->frameIndex : 0u;
for (auto &e : _entries)
{
if (e.state == EntryState::Unloaded)
{
// Visibility-driven residency: only start uploads for textures
// that were marked used recently (current or previous frame).
// This avoids uploading assets that are not visible.
bool recentlyUsed = true;
if (_context)
{
// Schedule when first seen (previous frame) or if seen again.
recentlyUsed = (now == 0u) || (now - e.lastUsedFrame <= 1u);
}
if (recentlyUsed)
{
start_load(e, rm);
if (++started >= kMaxLoadsPerPump) break;
}
}
}
}
void TextureCache::evictToBudget(size_t budgetBytes)

View File

@@ -170,6 +170,22 @@ void GeometryPass::draw_geometry(VkCommandBuffer cmd,
}
}
// Texture visibility-driven residency
if (ctxLocal->textures && !opaque_draws.empty())
{
std::unordered_set<VkDescriptorSet> seen;
seen.reserve(opaque_draws.size());
for (uint32_t idx : opaque_draws)
{
const RenderObject &r = mainDrawContext.OpaqueSurfaces[idx];
VkDescriptorSet set = r.material ? r.material->materialSet : VK_NULL_HANDLE;
if (set != VK_NULL_HANDLE && seen.insert(set).second)
{
ctxLocal->textures->markSetUsed(set, ctxLocal->frameIndex);
}
}
}
std::sort(opaque_draws.begin(), opaque_draws.end(), [&](const auto &iA, const auto &iB)
{
const RenderObject &A = mainDrawContext.OpaqueSurfaces[iA];