mirror of
https://github.com/AxioDL/metaforce.git
synced 2025-12-10 22:27:42 +00:00
Initial ShaderCache implementation
This commit is contained in:
256
hecl/lib/Runtime/ShaderCacheManager.cpp
Normal file
256
hecl/lib/Runtime/ShaderCacheManager.cpp
Normal file
@@ -0,0 +1,256 @@
|
||||
#include "HECL/Runtime.hpp"
|
||||
#include <Athena/FileReader.hpp>
|
||||
#include <Athena/FileWriter.hpp>
|
||||
#include <zlib.h>
|
||||
#include <algorithm>
|
||||
|
||||
namespace HECL
|
||||
{
|
||||
namespace Runtime
|
||||
{
|
||||
static LogVisor::LogModule Log("ShaderCacheManager");
|
||||
static uint64_t IDX_MAGIC = SBIG(0xDEADFEEDC001D00D);
|
||||
static uint64_t DAT_MAGIC = SBIG(0xC001D00DDEADBABE);
|
||||
static uint64_t ZERO64 = 0;
|
||||
|
||||
static uint64_t Random64()
|
||||
{
|
||||
uint64_t ret;
|
||||
#if _WIN32
|
||||
#else
|
||||
FILE* fp = fopen("/dev/urandom", "rb");
|
||||
fread(&ret, 1, 8, fp);
|
||||
fclose(fp);
|
||||
#endif
|
||||
return ret;
|
||||
}
|
||||
|
||||
void ShaderCacheManager::BootstrapIndex()
|
||||
{
|
||||
m_loadedRand = Random64();
|
||||
m_idxFr.close();
|
||||
m_datFr.close();
|
||||
|
||||
FILE* idxFp = HECL::Fopen(m_idxFr.filename().c_str(), _S("wb"));
|
||||
if (!idxFp)
|
||||
Log.report(LogVisor::FatalError, _S("unable to write shader cache index at %s"),
|
||||
m_idxFr.filename().c_str());
|
||||
fwrite(&IDX_MAGIC, 1, 8, idxFp);
|
||||
fwrite(&m_loadedRand, 1, 8, idxFp);
|
||||
fwrite(&ZERO64, 1, 8, idxFp);
|
||||
fwrite(&ZERO64, 1, 8, idxFp);
|
||||
fclose(idxFp);
|
||||
|
||||
FILE* datFp = HECL::Fopen(m_datFr.filename().c_str(), _S("wb"));
|
||||
if (!datFp)
|
||||
Log.report(LogVisor::FatalError, _S("unable to write shader cache data at %s"),
|
||||
m_datFr.filename().c_str());
|
||||
fwrite(&DAT_MAGIC, 1, 8, datFp);
|
||||
fwrite(&m_loadedRand, 1, 8, datFp);
|
||||
fclose(datFp);
|
||||
|
||||
m_idxFr.open();
|
||||
m_datFr.open();
|
||||
}
|
||||
|
||||
void ShaderCacheManager::reload()
|
||||
{
|
||||
m_entries.clear();
|
||||
m_entryLookup.clear();
|
||||
m_loadedRand = 0;
|
||||
|
||||
/* Attempt to open existing index */
|
||||
m_idxFr.seek(0, Athena::Begin);
|
||||
m_datFr.seek(0, Athena::Begin);
|
||||
if (m_idxFr.hasError() || m_datFr.hasError())
|
||||
{
|
||||
BootstrapIndex();
|
||||
return;
|
||||
}
|
||||
else
|
||||
{
|
||||
uint64_t idxMagic;
|
||||
size_t rb = m_idxFr.readUBytesToBuf(&idxMagic, 8);
|
||||
if (rb != 8 || idxMagic != IDX_MAGIC)
|
||||
{
|
||||
BootstrapIndex();
|
||||
return;
|
||||
}
|
||||
|
||||
uint64_t datMagic;
|
||||
rb = m_datFr.readUBytesToBuf(&datMagic, 8);
|
||||
if (rb != 8 || datMagic != DAT_MAGIC)
|
||||
{
|
||||
BootstrapIndex();
|
||||
return;
|
||||
}
|
||||
|
||||
uint64_t idxRand, datRand;
|
||||
rb = m_idxFr.readUBytesToBuf(&idxRand, 8);
|
||||
size_t rb2 = m_datFr.readUBytesToBuf(&datRand, 8);
|
||||
if (rb != 8 || rb2 != 8 || idxRand != datRand)
|
||||
{
|
||||
BootstrapIndex();
|
||||
return;
|
||||
}
|
||||
m_loadedRand = idxRand;
|
||||
}
|
||||
|
||||
/* Read existing entries */
|
||||
atUint64 idxCount = m_idxFr.readUint64Big();
|
||||
if (m_idxFr.position() != 24)
|
||||
{
|
||||
BootstrapIndex();
|
||||
return;
|
||||
}
|
||||
|
||||
m_entries.reserve(idxCount);
|
||||
m_entryLookup.reserve(idxCount);
|
||||
for (atUint64 i=0 ; i<idxCount ; ++i)
|
||||
{
|
||||
m_entries.emplace_back();
|
||||
IndexEntry& ent = m_entries.back();
|
||||
ent.read(m_idxFr);
|
||||
m_entryLookup[ent.m_hash] = m_entries.size() - 1;
|
||||
}
|
||||
}
|
||||
|
||||
ShaderCacheManager::CachedData ShaderCacheManager::lookupData(const Hash& hash)
|
||||
{
|
||||
auto search = m_entryLookup.find(hash);
|
||||
if (search == m_entryLookup.cend())
|
||||
return CachedData();
|
||||
|
||||
const IndexEntry& ent = m_entries[search->second];
|
||||
if (ent.m_compOffset + ent.m_compSize > m_datFr.length())
|
||||
{
|
||||
Log.report(LogVisor::Warning, "shader cache not long enough to read entry, might be corrupt");
|
||||
return CachedData();
|
||||
}
|
||||
|
||||
/* File-streamed decompression */
|
||||
m_datFr.seek(ent.m_compOffset, Athena::Begin);
|
||||
CachedData ret(ent.m_hash, ent.m_meta, ent.m_decompSize);
|
||||
uint8_t compDat[2048];
|
||||
z_stream z = {};
|
||||
inflateInit(&z);
|
||||
z.avail_out = ent.m_decompSize;
|
||||
z.next_out = ret.m_data.get();
|
||||
while (z.avail_out)
|
||||
{
|
||||
z.avail_in = std::min(size_t(2048), size_t(ent.m_compSize - z.total_in));
|
||||
m_datFr.readUBytesToBuf(compDat, z.avail_in);
|
||||
z.next_in = compDat;
|
||||
inflate(&z, Z_NO_FLUSH);
|
||||
}
|
||||
inflateEnd(&z);
|
||||
return ret;
|
||||
}
|
||||
|
||||
bool ShaderCacheManager::addData(const ShaderTag& tag, const void* data, size_t sz)
|
||||
{
|
||||
m_idxFr.close();
|
||||
m_datFr.close();
|
||||
|
||||
/* Perform one-shot buffer compression */
|
||||
uLong cBound = compressBound(sz);
|
||||
void* compBuf = malloc(cBound);
|
||||
if (compress((Bytef*)compBuf, &cBound, (Bytef*)data, sz) != Z_OK)
|
||||
Log.report(LogVisor::FatalError, "unable to deflate data");
|
||||
|
||||
/* Open index for writing (non overwriting) */
|
||||
Athena::io::FileWriter idxFw(m_idxFr.filename(), false);
|
||||
if (idxFw.hasError())
|
||||
Log.report(LogVisor::FatalError, _S("unable to append shader cache index at %s"),
|
||||
m_idxFr.filename().c_str());
|
||||
|
||||
/* Open data for writing (non overwriting) */
|
||||
Athena::io::FileWriter datFw(m_datFr.filename(), false);
|
||||
if (datFw.hasError())
|
||||
Log.report(LogVisor::FatalError, _S("unable to append shader cache data at %s"),
|
||||
m_datFr.filename().c_str());
|
||||
|
||||
size_t targetOffset = 0;
|
||||
auto search = m_entryLookup.find(tag);
|
||||
if (search != m_entryLookup.cend())
|
||||
{
|
||||
/* Hash already present, attempt to replace data */
|
||||
IndexEntry& ent = m_entries[search->second];
|
||||
if (search->second == m_entries.size() - 1)
|
||||
{
|
||||
/* Replacing final entry; simply write-over */
|
||||
ent.m_meta = tag.getMetaData();
|
||||
ent.m_compSize = cBound;
|
||||
ent.m_decompSize = sz;
|
||||
targetOffset = ent.m_compOffset;
|
||||
idxFw.seek(search->second * 32 + 32);
|
||||
ent.write(idxFw);
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Replacing non-final entry; write into available space */
|
||||
IndexEntry& nent = m_entries[search->second+1];
|
||||
size_t space = nent.m_compOffset - ent.m_compOffset;
|
||||
if (cBound <= space)
|
||||
{
|
||||
ent.m_meta = tag.getMetaData();
|
||||
ent.m_compSize = cBound;
|
||||
ent.m_decompSize = sz;
|
||||
targetOffset = ent.m_compOffset;
|
||||
idxFw.seek(search->second * 32 + 32);
|
||||
ent.write(idxFw);
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Not enough space; null-entry and add to end */
|
||||
ent.m_hash = 0;
|
||||
ent.m_meta = 0;
|
||||
ent.m_compOffset = 0;
|
||||
ent.m_compSize = 0;
|
||||
ent.m_decompSize = 0;
|
||||
idxFw.seek(search->second * 32 + 32);
|
||||
ent.write(idxFw);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!targetOffset)
|
||||
{
|
||||
/* New index entry at end */
|
||||
idxFw.seek(16, Athena::Begin);
|
||||
idxFw.writeUint64Big(m_entries.size() + 1);
|
||||
idxFw.seek(m_entries.size() * 32 + 32, Athena::Begin);
|
||||
datFw.seek(0, Athena::End);
|
||||
m_entryLookup[tag] = m_entries.size();
|
||||
m_entries.emplace_back();
|
||||
|
||||
IndexEntry& ent = m_entries.back();
|
||||
ent.m_hash = tag.val64();
|
||||
ent.m_meta = tag.getMetaData();
|
||||
ent.m_compOffset = datFw.position();
|
||||
ent.m_compSize = cBound;
|
||||
ent.m_decompSize = sz;
|
||||
ent.write(idxFw);
|
||||
|
||||
datFw.writeUBytes((atUint8*)compBuf, cBound);
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Reusing index entry and data space */
|
||||
datFw.seek(targetOffset, Athena::Begin);
|
||||
datFw.writeUBytes((atUint8*)compBuf, cBound);
|
||||
}
|
||||
|
||||
free(compBuf);
|
||||
|
||||
idxFw.close();
|
||||
datFw.close();
|
||||
|
||||
m_idxFr.open();
|
||||
m_datFr.open();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user