Update terminology entirely for packedstore.h/.cpp

VPK tooling also works on the R1 game (Titanfall 1)
This commit is contained in:
Kawe Mazidjatari 2022-06-06 14:54:22 +02:00
parent 6e6e791c10
commit 461920a970
3 changed files with 189 additions and 187 deletions

View File

@ -1,11 +1,11 @@
/***********************************************************************
* *
* *
* *
* *
* *
* *
***********************************************************************/
/*******************************************************************
* *
* *
* *
* *
* *
* *
*******************************************************************/
#include "core/stdafx.h"
#include "tier1/cvar.h"
@ -38,11 +38,11 @@ void CPackedStore::InitLzDecompParams(void)
}
//-----------------------------------------------------------------------------
// Purpose: returns populated pack dir struct for specified pack dir file
// Purpose: gets a directory structure for sepcified file
// Input : svPackDirFile -
// Output : VPKDir_t
//-----------------------------------------------------------------------------
VPKDir_t CPackedStore::GetPackDirFile(string svPackDirFile) const
VPKDir_t CPackedStore::GetDirectoryFile(string svPackDirFile) const
{
/*| PACKDIRFILE |||||||||||||||||||||||||||||||||||||||||||||||||||||||||*/
std::regex rgArchiveRegex("pak000_([0-9]{3})");
@ -71,17 +71,17 @@ VPKDir_t CPackedStore::GetPackDirFile(string svPackDirFile) const
}escape:;
}
VPKDir_t vpk_dir(svPackDirFile);
return vpk_dir;
VPKDir_t vDir(svPackDirFile);
return vDir;
}
//-----------------------------------------------------------------------------
// Purpose: obtains archive chunk path for specific file
// Purpose: formats pack file path for specific directory file
// Input : &svPackDirFile -
// iArchiveIndex -
// output : string
//-----------------------------------------------------------------------------
string CPackedStore::GetPackChunkFile(const string& svPackDirFile, uint16_t iArchiveIndex) const
string CPackedStore::GetPackFile(const string& svPackDirFile, uint16_t iArchiveIndex) const
{
/*| ARCHIVES ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||*/
string svPackChunkFile = StripLocalePrefix(svPackDirFile);
@ -110,7 +110,7 @@ vector<VPKEntryBlock_t> CPackedStore::GetEntryBlocks(CIOStream* pReader) const
{
while (!(svName = pReader->ReadString()).empty())
{
string svFilePath = FormatBlockPath(svPath, svName, svExtension);
string svFilePath = FormatEntryPath(svPath, svName, svExtension);
vBlocks.push_back(VPKEntryBlock_t(pReader, svFilePath));
}
}
@ -123,7 +123,7 @@ vector<VPKEntryBlock_t> CPackedStore::GetEntryBlocks(CIOStream* pReader) const
// Input : &svPathIn -
// Output : vector<string>
//-----------------------------------------------------------------------------
vector<string> CPackedStore::GetBlockPaths(const string& svPathIn) const
vector<string> CPackedStore::GetEntryPaths(const string& svPathIn) const
{
vector<string> vPaths;
fs::recursive_directory_iterator dir(svPathIn), end;
@ -131,7 +131,7 @@ vector<string> CPackedStore::GetBlockPaths(const string& svPathIn) const
{
if (dir->path().filename() == "manifest")
{
dir.disable_recursion_pending(); // Don't recurse into this directory (manifest files only).
dir.disable_recursion_pending(); // Manifest files are not packed.
}
if (!GetExtension(dir->path().u8string()).empty())
{
@ -148,7 +148,7 @@ vector<string> CPackedStore::GetBlockPaths(const string& svPathIn) const
// &jManifest -
// Output : vector<string>
//-----------------------------------------------------------------------------
vector<string> CPackedStore::GetBlockPaths(const string& svPathIn, const nlohmann::json& jManifest) const
vector<string> CPackedStore::GetEntryPaths(const string& svPathIn, const nlohmann::json& jManifest) const
{
vector<string> vPaths;
fs::recursive_directory_iterator dir(svPathIn), end;
@ -156,7 +156,7 @@ vector<string> CPackedStore::GetBlockPaths(const string& svPathIn, const nlohman
{
if (dir->path().filename() == "manifest")
{
dir.disable_recursion_pending(); // Don't recurse into this directory (manifest files only).
dir.disable_recursion_pending(); // Manifest files are not packed.
}
if (!GetExtension(dir->path().u8string()).empty())
{
@ -241,13 +241,13 @@ nlohmann::json CPackedStore::GetManifest(const string& svWorkSpace, const string
}
//-----------------------------------------------------------------------------
// Purpose: formats the entry block path
// Purpose: formats the file entry path
// Input : svPath -
// &svName -
// &svExtension -
// Output : string
//-----------------------------------------------------------------------------
string CPackedStore::FormatBlockPath(string svPath, const string& svName, const string& svExtension) const
string CPackedStore::FormatEntryPath(string svPath, const string& svName, const string& svExtension) const
{
if (!svPath.empty())
{
@ -315,12 +315,12 @@ void CPackedStore::BuildManifest(const vector<VPKEntryBlock_t>& vBlock, const st
for (size_t i = 0; i < vBlock.size(); i++)
{
jEntry[vBlock[i].m_svBlockPath] =
jEntry[vBlock[i].m_svEntryPath] =
{
{ "preloadData", vBlock[i].m_nPreloadData },
{ "entryFlags", vBlock[i].m_vvEntries[0].m_nEntryFlags },
{ "textureFlags", vBlock[i].m_vvEntries[0].m_nTextureFlags },
{ "useCompression", vBlock[i].m_vvEntries[0].m_nCompressedSize != vBlock[i].m_vvEntries[0].m_nUncompressedSize },
{ "preloadSize", vBlock[i].m_iPreloadSize },
{ "LoadFlags", vBlock[i].m_vChunks[0].m_nLoadFlags },
{ "textureFlags", vBlock[i].m_vChunks[0].m_nTextureFlags },
{ "useCompression", vBlock[i].m_vChunks[0].m_nCompressedSize != vBlock[i].m_vChunks[0].m_nUncompressedSize },
{ "useDataSharing", true }
};
}
@ -383,11 +383,11 @@ void CPackedStore::PackAll(const VPKPair_t& vPair, const string& svPathIn, const
if (bManifestOnly)
{
vPaths = GetBlockPaths(svPathIn, jManifest);
vPaths = GetEntryPaths(svPathIn, jManifest);
}
else // Pack all files in workspace.
{
vPaths = GetBlockPaths(svPathIn);
vPaths = GetEntryPaths(svPathIn);
}
uint64_t nSharedTotal = 0i64;
@ -399,8 +399,8 @@ void CPackedStore::PackAll(const VPKPair_t& vPair, const string& svPathIn, const
if (reader.IsReadable())
{
string svDestPath = StringReplaceC(vPaths[i], svPathIn, "");
uint16_t nPreloadData = 0i16;
uint32_t nEntryFlags = static_cast<uint32_t>(EPackedEntryFlags::ENTRY_VISIBLE) | static_cast<uint32_t>(EPackedEntryFlags::ENTRY_CACHE);
uint16_t nPreloadSize = 0i16;
uint32_t nLoadFlags = static_cast<uint32_t>(EPackedLoadFlags::LOAD_VISIBLE) | static_cast<uint32_t>(EPackedLoadFlags::LOAD_CACHE);
uint16_t nTextureFlags = static_cast<short>(EPackedTextureFlags::TEXTURE_DEFAULT); // !TODO: Reverse these.
bool bUseCompression = true;
bool bUseDataSharing = true;
@ -412,8 +412,8 @@ void CPackedStore::PackAll(const VPKPair_t& vPair, const string& svPathIn, const
nlohmann::json jEntry = jManifest[svDestPath];
if (!jEntry.is_null())
{
nPreloadData = jEntry.at("preloadData").get<uint32_t>();
nEntryFlags = jEntry.at("entryFlags").get<uint32_t>();
nPreloadSize = jEntry.at("preloadSize").get<uint32_t>();
nLoadFlags = jEntry.at("loadFlags").get<uint32_t>();
nTextureFlags = jEntry.at("textureFlags").get<uint16_t>();
bUseCompression = jEntry.at("useCompression").get<bool>();
bUseDataSharing = jEntry.at("useDataSharing").get<bool>();
@ -424,63 +424,63 @@ void CPackedStore::PackAll(const VPKPair_t& vPair, const string& svPathIn, const
Warning(eDLL_T::FS, "Exception while reading VPK manifest file: '%s'\n", ex.what());
}
}
DevMsg(eDLL_T::FS, "Packing block '%llu' ('%s')\n", i, svDestPath.c_str());
DevMsg(eDLL_T::FS, "Packing entry '%llu' ('%s')\n", i, svDestPath.c_str());
vEntryBlocks.push_back(VPKEntryBlock_t(reader.GetVector(), writer.GetPosition(), nPreloadData, 0, nEntryFlags, nTextureFlags, svDestPath));
for (size_t j = 0; j < vEntryBlocks[i].m_vvEntries.size(); j++)
vEntryBlocks.push_back(VPKEntryBlock_t(reader.GetVector(), writer.GetPosition(), nPreloadSize, 0, nLoadFlags, nTextureFlags, svDestPath));
for (size_t j = 0; j < vEntryBlocks[i].m_vChunks.size(); j++)
{
uint8_t* pSrc = new uint8_t[vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize];
uint8_t* pDest = new uint8_t[vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize];
uint8_t* pSrc = new uint8_t[vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize];
uint8_t* pDest = new uint8_t[vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize];
bool bShared = false;
bool bCompressed = bUseCompression;
reader.Read(*pSrc, vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize);
vEntryBlocks[i].m_vvEntries[j].m_nArchiveOffset = writer.GetPosition();
reader.Read(*pSrc, vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize);
vEntryBlocks[i].m_vChunks[j].m_nArchiveOffset = writer.GetPosition();
if (bUseCompression)
{
m_lzCompStatus = lzham_compress_memory(&m_lzCompParams, pDest,
&vEntryBlocks[i].m_vvEntries[j].m_nCompressedSize, pSrc,
vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize, &m_nAdler32_Internal, &m_nCrc32_Internal);
&vEntryBlocks[i].m_vChunks[j].m_nCompressedSize, pSrc,
vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize, &m_nAdler32_Internal, &m_nCrc32_Internal);
if (m_lzCompStatus != lzham_compress_status_t::LZHAM_COMP_STATUS_SUCCESS)
{
Warning(eDLL_T::FS, "Status '%d' for entry '%llu' within block '%llu' for chunk '%hu' (entry packed without compression)\n",
m_lzCompStatus, j, i, vEntryBlocks[i].m_iArchiveIndex);
Warning(eDLL_T::FS, "Status '%d' for chunk '%llu' within entry '%llu' in block '%hu' (chunk packed without compression)\n",
m_lzCompStatus, j, i, vEntryBlocks[i].m_iPackFileIndex);
vEntryBlocks[i].m_vvEntries[j].m_nCompressedSize = vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize;
memmove(pDest, pSrc, vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize);
vEntryBlocks[i].m_vChunks[j].m_nCompressedSize = vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize;
memmove(pDest, pSrc, vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize);
}
}
else // Write data uncompressed.
{
vEntryBlocks[i].m_vvEntries[j].m_nCompressedSize = vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize;
memmove(pDest, pSrc, vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize);
vEntryBlocks[i].m_vChunks[j].m_nCompressedSize = vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize;
memmove(pDest, pSrc, vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize);
}
vEntryBlocks[i].m_vvEntries[j].m_bIsCompressed = vEntryBlocks[i].m_vvEntries[j].m_nCompressedSize != vEntryBlocks[i].m_vvEntries[j].m_nUncompressedSize;
vEntryBlocks[i].m_vChunks[j].m_bIsCompressed = vEntryBlocks[i].m_vChunks[j].m_nCompressedSize != vEntryBlocks[i].m_vChunks[j].m_nUncompressedSize;
if (bUseDataSharing)
{
string svEntryHash = sha1(string(reinterpret_cast<char*>(pDest), vEntryBlocks[i].m_vvEntries[j].m_nCompressedSize));
string svEntryHash = sha1(string(reinterpret_cast<char*>(pDest), vEntryBlocks[i].m_vChunks[j].m_nCompressedSize));
if (auto it{ m_mEntryHashMap.find(svEntryHash) }; it != std::end(m_mEntryHashMap))
if (auto it{ m_mChunkHashMap.find(svEntryHash) }; it != std::end(m_mChunkHashMap))
{
vEntryBlocks[i].m_vvEntries[j].m_nArchiveOffset = it->second.m_nArchiveOffset;
vEntryBlocks[i].m_vChunks[j].m_nArchiveOffset = it->second.m_nArchiveOffset;
nSharedTotal += it->second.m_nCompressedSize;
nSharedCount++;
bShared = true;
DevMsg(eDLL_T::FS, "Mapping entry '%lld' ('%s') to existing entry at '0x%llx'\n", j, svEntryHash.c_str(), it->second.m_nArchiveOffset);
DevMsg(eDLL_T::FS, "Mapping chunk '%lld' ('%s') to existing chunk at '0x%llx'\n", j, svEntryHash.c_str(), it->second.m_nArchiveOffset);
}
else // Add entry to hashmap.
{
m_mEntryHashMap.insert({ svEntryHash, vEntryBlocks[i].m_vvEntries[j] });
m_mChunkHashMap.insert({ svEntryHash, vEntryBlocks[i].m_vChunks[j] });
bShared = false;
}
}
if (!bShared)
{
writer.Write(pDest, vEntryBlocks[i].m_vvEntries[j].m_nCompressedSize);
writer.Write(pDest, vEntryBlocks[i].m_vChunks[j].m_nCompressedSize);
}
delete[] pDest;
@ -488,8 +488,8 @@ void CPackedStore::PackAll(const VPKPair_t& vPair, const string& svPathIn, const
}
}
}
DevMsg(eDLL_T::FS, "*** Build chunk totalling '%llu' bytes with '%llu' shared bytes among '%lu' entries\n", writer.GetPosition(), nSharedTotal, nSharedCount);
m_mEntryHashMap.clear();
DevMsg(eDLL_T::FS, "*** Build block totalling '%llu' bytes with '%llu' shared bytes among '%lu' chunks\n", writer.GetPosition(), nSharedTotal, nSharedCount);
m_mChunkHashMap.clear();
VPKDir_t vDir = VPKDir_t();
vDir.Build(svPathOut + vPair.m_svDirectoryName, vEntryBlocks);
@ -497,35 +497,35 @@ void CPackedStore::PackAll(const VPKPair_t& vPair, const string& svPathIn, const
//-----------------------------------------------------------------------------
// Purpose: extracts all files from specified VPK file
// Input : &vpkDir -
// Input : &vDir -
// &svPathOut -
//-----------------------------------------------------------------------------
void CPackedStore::UnpackAll(const VPKDir_t& vpkDir, const string& svPathOut)
void CPackedStore::UnpackAll(const VPKDir_t& vDir, const string& svPathOut)
{
if (vpkDir.m_vHeader.m_nHeaderMarker != VPK_HEADER_MARKER ||
vpkDir.m_vHeader.m_nMajorVersion != VPK_MAJOR_VERSION ||
vpkDir.m_vHeader.m_nMinorVersion != VPK_MINOR_VERSION)
if (vDir.m_vHeader.m_nHeaderMarker != VPK_HEADER_MARKER ||
vDir.m_vHeader.m_nMajorVersion != VPK_MAJOR_VERSION ||
vDir.m_vHeader.m_nMinorVersion != VPK_MINOR_VERSION)
{
Error(eDLL_T::FS, "Invalid VPK directory file (header doesn't match criteria)\n");
return;
}
BuildManifest(vpkDir.m_vvEntryBlocks, svPathOut, GetSourceName(vpkDir.m_svDirPath));
BuildManifest(vDir.m_vEntryBlocks, svPathOut, GetSourceName(vDir.m_svDirPath));
for (size_t i = 0; i < vpkDir.m_vsvArchives.size(); i++)
for (size_t i = 0; i < vDir.m_vPackFile.size(); i++)
{
fs::path fspVpkPath(vpkDir.m_svDirPath);
string svPath = fspVpkPath.parent_path().u8string() + '\\' + vpkDir.m_vsvArchives[i];
fs::path fspVpkPath(vDir.m_svDirPath);
string svPath = fspVpkPath.parent_path().u8string() + '\\' + vDir.m_vPackFile[i];
CIOStream iStream(svPath, CIOStream::Mode_t::READ); // Create stream to read from each archive.
for ( size_t j = 0; j < vpkDir.m_vvEntryBlocks.size(); j++)
for ( size_t j = 0; j < vDir.m_vEntryBlocks.size(); j++)
{
if (vpkDir.m_vvEntryBlocks[j].m_iArchiveIndex != static_cast<uint16_t>(i))
if (vDir.m_vEntryBlocks[j].m_iPackFileIndex != static_cast<uint16_t>(i))
{
goto escape;
}
else // Chunk belongs to this block.
{
string svFilePath = CreateDirectories(svPathOut + vpkDir.m_vvEntryBlocks[j].m_svBlockPath);
string svFilePath = CreateDirectories(svPathOut + vDir.m_vEntryBlocks[j].m_svEntryPath);
CIOStream oStream(svFilePath, CIOStream::Mode_t::WRITE);
if (!oStream.IsWritable())
@ -533,46 +533,46 @@ void CPackedStore::UnpackAll(const VPKDir_t& vpkDir, const string& svPathOut)
Error(eDLL_T::FS, "Unable to write file '%s'\n", svFilePath.c_str());
continue;
}
DevMsg(eDLL_T::FS, "Unpacking block '%llu' from chunk '%llu' ('%s')\n", j, i, vpkDir.m_vvEntryBlocks[j].m_svBlockPath.c_str());
DevMsg(eDLL_T::FS, "Unpacking entry '%llu' from block '%llu' ('%s')\n", j, i, vDir.m_vEntryBlocks[j].m_svEntryPath.c_str());
for (VPKEntryDescriptor_t vEntry : vpkDir.m_vvEntryBlocks[j].m_vvEntries)
for (VPKChunkDescriptor_t vChunk : vDir.m_vEntryBlocks[j].m_vChunks)
{
m_nEntryCount++;
m_nChunkCount++;
uint8_t* pCompressedData = new uint8_t[vEntry.m_nCompressedSize];
uint8_t* pCompressedData = new uint8_t[vChunk.m_nCompressedSize];
iStream.SetPosition(vEntry.m_nArchiveOffset);
iStream.Read(*pCompressedData, vEntry.m_nCompressedSize);
iStream.SetPosition(vChunk.m_nArchiveOffset);
iStream.Read(*pCompressedData, vChunk.m_nCompressedSize);
if (vEntry.m_bIsCompressed)
if (vChunk.m_bIsCompressed)
{
uint8_t* pLzOutputBuf = new uint8_t[vEntry.m_nUncompressedSize];
uint8_t* pLzOutputBuf = new uint8_t[vChunk.m_nUncompressedSize];
m_lzDecompStatus = lzham_decompress_memory(&m_lzDecompParams, pLzOutputBuf,
&vEntry.m_nUncompressedSize, pCompressedData,
vEntry.m_nCompressedSize, &m_nAdler32_Internal, &m_nCrc32_Internal);
&vChunk.m_nUncompressedSize, pCompressedData,
vChunk.m_nCompressedSize, &m_nAdler32_Internal, &m_nCrc32_Internal);
if (m_lzDecompStatus != lzham_decompress_status_t::LZHAM_DECOMP_STATUS_SUCCESS)
{
Error(eDLL_T::FS, "Status '%d' for entry '%llu' within block '%llu' for chunk '%hu' (entry not decompressed)\n",
m_lzDecompStatus, m_nEntryCount, i, vpkDir.m_vvEntryBlocks[j].m_iArchiveIndex);
Error(eDLL_T::FS, "Status '%d' for chunk '%llu' within entry '%llu' in block '%hu' (- not decompressed)\n",
m_lzDecompStatus, m_nChunkCount, i, vDir.m_vEntryBlocks[j].m_iPackFileIndex);
}
else // If successfully decompressed, write to file.
{
oStream.Write(pLzOutputBuf, vEntry.m_nUncompressedSize);
oStream.Write(pLzOutputBuf, vChunk.m_nUncompressedSize);
}
delete[] pLzOutputBuf;
}
else // If not compressed, write raw data into output file.
{
oStream.Write(pCompressedData, vEntry.m_nUncompressedSize);
oStream.Write(pCompressedData, vChunk.m_nUncompressedSize);
}
delete[] pCompressedData;
}
if (m_nEntryCount == vpkDir.m_vvEntryBlocks[j].m_vvEntries.size()) // Only validate after last entry in block had been written.
if (m_nChunkCount == vDir.m_vEntryBlocks[j].m_vChunks.size()) // Only validate after last entry in block had been written.
{
m_nEntryCount = 0;
m_nCrc32_Internal = vpkDir.m_vvEntryBlocks[j].m_nCrc32;
m_nChunkCount = 0;
m_nCrc32_Internal = vDir.m_vEntryBlocks[j].m_nFileCRC;
oStream.Flush();
ValidateCRC32PostDecomp(svFilePath);
@ -588,20 +588,20 @@ void CPackedStore::UnpackAll(const VPKDir_t& vpkDir, const string& svPathOut)
// Input : *pReader -
// svBlockPath -
//-----------------------------------------------------------------------------
VPKEntryBlock_t::VPKEntryBlock_t(CIOStream* pReader, string svBlockPath)
VPKEntryBlock_t::VPKEntryBlock_t(CIOStream* pReader, string svEntryPath)
{
StringReplace(svBlockPath, "\\", "/"); // Flip windows-style backslash to forward slash.
StringReplace(svBlockPath, " /", "" ); // Remove space character representing VPK root.
StringReplace(svEntryPath, "\\", "/"); // Flip windows-style backslash to forward slash.
StringReplace(svEntryPath, " /", "" ); // Remove space character representing VPK root.
this->m_svBlockPath = svBlockPath; // Set path of block.
pReader->Read<uint32_t>(this->m_nCrc32); //
pReader->Read<uint16_t>(this->m_nPreloadData); //
pReader->Read<uint16_t>(this->m_iArchiveIndex); //
this->m_svEntryPath = svEntryPath; // Set the entry path.
pReader->Read<uint32_t>(this->m_nFileCRC); //
pReader->Read<uint16_t>(this->m_iPreloadSize); //
pReader->Read<uint16_t>(this->m_iPackFileIndex); //
do // Loop through all entries in the block and push them to the vector.
do // Loop through all chunks in the entry and push them to the vector.
{
VPKEntryDescriptor_t entry(pReader);
this->m_vvEntries.push_back(entry);
VPKChunkDescriptor_t entry(pReader);
this->m_vChunks.push_back(entry);
} while (pReader->Read<uint16_t>() != UINT16_MAX);
}
@ -615,33 +615,33 @@ VPKEntryBlock_t::VPKEntryBlock_t(CIOStream* pReader, string svBlockPath)
// nTextureFlags -
// svBlockPath -
//-----------------------------------------------------------------------------
VPKEntryBlock_t::VPKEntryBlock_t(const vector<uint8_t> &vData, int64_t nOffset, uint16_t nPreloadData, uint16_t nArchiveIndex, uint32_t nEntryFlags, uint16_t nTextureFlags, const string& svBlockPath)
VPKEntryBlock_t::VPKEntryBlock_t(const vector<uint8_t> &vData, int64_t nOffset, uint16_t nPreloadData, uint16_t nArchiveIndex, uint32_t nEntryFlags, uint16_t nTextureFlags, const string& svEntryPath)
{
m_nCrc32 = crc32::update(m_nCrc32, vData.data(), vData.size());
m_nPreloadData = nPreloadData;
m_iArchiveIndex = nArchiveIndex;
m_svBlockPath = svBlockPath;
m_nFileCRC = crc32::update(m_nFileCRC, vData.data(), vData.size());
m_iPreloadSize = nPreloadData;
m_iPackFileIndex = nArchiveIndex;
m_svEntryPath = svEntryPath;
size_t nEntryCount = (vData.size() + ENTRY_MAX_LEN - 1) / ENTRY_MAX_LEN;
size_t nDataSize = vData.size();
int64_t nCurrentOffset = nOffset;
for (int i = 0; i < nEntryCount; i++)
for (size_t i = 0; i < nEntryCount; i++) // Fragment data into 1MiB chunks
{
size_t nSize = std::min<uint64_t>(ENTRY_MAX_LEN, nDataSize);
nDataSize -= nSize;
m_vvEntries.push_back(VPKEntryDescriptor_t(nEntryFlags, nTextureFlags, nCurrentOffset, nSize, nSize));
m_vChunks.push_back(VPKChunkDescriptor_t(nEntryFlags, nTextureFlags, nCurrentOffset, nSize, nSize));
nCurrentOffset += nSize;
}
}
//-----------------------------------------------------------------------------
// Purpose: 'VPKDir_t' file constructor
// Purpose: 'VPKChunkDescriptor_t' file constructor
// Input : *pReader -
//-----------------------------------------------------------------------------
VPKEntryDescriptor_t::VPKEntryDescriptor_t(CIOStream* pReader)
VPKChunkDescriptor_t::VPKChunkDescriptor_t(CIOStream* pReader)
{
pReader->Read<uint32_t>(this->m_nEntryFlags); //
pReader->Read<uint32_t>(this->m_nLoadFlags); //
pReader->Read<uint16_t>(this->m_nTextureFlags); //
pReader->Read<uint64_t>(this->m_nArchiveOffset); //
pReader->Read<uint64_t>(this->m_nCompressedSize); //
@ -650,16 +650,16 @@ VPKEntryDescriptor_t::VPKEntryDescriptor_t(CIOStream* pReader)
}
//-----------------------------------------------------------------------------
// Purpose: 'VPKEntryDescriptor_t' memory constructor
// Purpose: 'VPKChunkDescriptor_t' memory constructor
// Input : &nEntryFlags -
// &nTextureFlags -
// &nArchiveOffset -
// &nCompressedSize -
// &nUncompressedSize -
//-----------------------------------------------------------------------------
VPKEntryDescriptor_t::VPKEntryDescriptor_t(uint32_t nEntryFlags, uint16_t nTextureFlags, uint64_t nArchiveOffset, uint64_t nCompressedSize, uint64_t nUncompressedSize)
VPKChunkDescriptor_t::VPKChunkDescriptor_t(uint32_t nLoadFlags, uint16_t nTextureFlags, uint64_t nArchiveOffset, uint64_t nCompressedSize, uint64_t nUncompressedSize)
{
m_nEntryFlags = nEntryFlags;
m_nLoadFlags = nLoadFlags;
m_nTextureFlags = nTextureFlags;
m_nArchiveOffset = nArchiveOffset;
@ -681,21 +681,21 @@ VPKDir_t::VPKDir_t(const string& svPath)
reader.Read<uint32_t>(this->m_vHeader.m_nDirectorySize); //
reader.Read<uint32_t>(this->m_nFileDataSize); //
this->m_vvEntryBlocks = g_pPackedStore->GetEntryBlocks(&reader);
this->m_svDirPath = svPath; // Set path to vpk_dir file.
this->m_vEntryBlocks = g_pPackedStore->GetEntryBlocks(&reader);
this->m_svDirPath = svPath; // Set path to vpk directory file.
for (VPKEntryBlock_t block : this->m_vvEntryBlocks)
for (VPKEntryBlock_t vEntry : this->m_vEntryBlocks)
{
if (block.m_iArchiveIndex > this->m_iArchiveCount)
if (vEntry.m_iPackFileIndex > this->m_iPackFileCount)
{
this->m_iArchiveCount = block.m_iArchiveIndex;
this->m_iPackFileCount = vEntry.m_iPackFileIndex;
}
}
for (uint16_t i = 0; i < this->m_iArchiveCount + 1; i++)
for (uint16_t i = 0; i < this->m_iPackFileCount + 1; i++)
{
string svArchivePath = g_pPackedStore->GetPackChunkFile(svPath, i);
this->m_vsvArchives.push_back(svArchivePath);
string svArchivePath = g_pPackedStore->GetPackFile(svPath, i);
this->m_vPackFile.push_back(svArchivePath);
}
}
@ -718,8 +718,8 @@ void VPKDir_t::Build(const string& svDirectoryFile, const vector<VPKEntryBlock_t
for (VPKEntryBlock_t vBlock : vEntryBlocks)
{
string svExtension = GetExtension(vBlock.m_svBlockPath);
string svFilePath = RemoveFileName(vBlock.m_svBlockPath);
string svExtension = GetExtension(vBlock.m_svEntryPath);
string svFilePath = RemoveFileName(vBlock.m_svEntryPath);
if (svFilePath.empty())
{
@ -744,23 +744,23 @@ void VPKDir_t::Build(const string& svDirectoryFile, const vector<VPKEntryBlock_t
writer.WriteString(jKeyValue.first);
for (auto& vEntry : jKeyValue.second)
{
writer.WriteString(GetFileName(vEntry.m_svBlockPath, true));
writer.WriteString(GetFileName(vEntry.m_svEntryPath, true));
{/*Write entry block*/
writer.Write(vEntry.m_nCrc32);
writer.Write(vEntry.m_nPreloadData);
writer.Write(vEntry.m_iArchiveIndex);
writer.Write(vEntry.m_nFileCRC);
writer.Write(vEntry.m_iPreloadSize);
writer.Write(vEntry.m_iPackFileIndex);
for (size_t i = 0; i < vEntry.m_vvEntries.size(); i++)
for (size_t i = 0; i < vEntry.m_vChunks.size(); i++)
{
{/*Write entry descriptor*/
writer.Write(vEntry.m_vvEntries[i].m_nEntryFlags);
writer.Write(vEntry.m_vvEntries[i].m_nTextureFlags);
writer.Write(vEntry.m_vvEntries[i].m_nArchiveOffset);
writer.Write(vEntry.m_vvEntries[i].m_nCompressedSize);
writer.Write(vEntry.m_vvEntries[i].m_nUncompressedSize);
{/*Write chunk descriptor*/
writer.Write(vEntry.m_vChunks[i].m_nLoadFlags);
writer.Write(vEntry.m_vChunks[i].m_nTextureFlags);
writer.Write(vEntry.m_vChunks[i].m_nArchiveOffset);
writer.Write(vEntry.m_vChunks[i].m_nCompressedSize);
writer.Write(vEntry.m_vChunks[i].m_nUncompressedSize);
}
if (i != (vEntry.m_vvEntries.size() - 1))
if (i != (vEntry.m_vChunks.size() - 1))
{
const ushort s = 0;
writer.Write(s);
@ -785,7 +785,7 @@ void VPKDir_t::Build(const string& svDirectoryFile, const vector<VPKEntryBlock_t
writer.Write(this->m_vHeader.m_nDirectorySize);
writer.Write(0);
DevMsg(eDLL_T::FS, "*** Build directory totalling '%llu' bytes with '%llu' blocks and '%llu' descriptors\n",
DevMsg(eDLL_T::FS, "*** Build directory totalling '%llu' bytes with '%llu' entries and '%llu' descriptors\n",
sizeof(VPKDirHeader_t) + m_vHeader.m_nDirectorySize, vEntryBlocks.size(), nDescriptors);
}
///////////////////////////////////////////////////////////////////////////////

View File

@ -11,15 +11,14 @@ constexpr int ENTRY_MAX_LEN = 1024 * 1024;
const vector<string> DIR_CONTEXT = { "server", "client" };
const vector<string> DIR_LOCALE = { "english", "french", "german", "italian", "japanese", "korean", "polish", "portuguese", "russian", "spanish", "tchinese" };
enum class EPackedEntryFlags : int
enum class EPackedLoadFlags : int
{
ENTRY_NONE,
ENTRY_VISIBLE = 1 << 0, // FileSystem visibility?
ENTRY_CACHE = 1 << 8, // Only set for assets not stored in the depot directory.
ENTRY_TEXTURE_UNK0 = 1 << 18,
ENTRY_TEXTURE_UNK1 = 1 << 19,
ENTRY_TEXTURE_UNK2 = 1 << 20,
LOAD_NONE,
LOAD_VISIBLE = 1 << 0, // FileSystem visibility?
LOAD_CACHE = 1 << 8, // Only set for assets not stored in the depot directory.
LOAD_TEXTURE_UNK0 = 1 << 18,
LOAD_TEXTURE_UNK1 = 1 << 19,
LOAD_TEXTURE_UNK2 = 1 << 20,
};
enum class EPackedTextureFlags : short
@ -50,30 +49,30 @@ struct VPKData_t
};
#pragma pack(pop)
struct VPKEntryDescriptor_t
struct VPKChunkDescriptor_t
{
uint32_t m_nEntryFlags {}; // Entry flags.
uint32_t m_nLoadFlags {}; // Load flags.
uint16_t m_nTextureFlags {}; // Texture flags (only used if the entry is a vtf).
uint64_t m_nArchiveOffset {}; // Offset in archive.
uint64_t m_nCompressedSize {}; // Compressed size of entry.
uint64_t m_nUncompressedSize{}; // Uncompressed size of entry.
uint64_t m_nCompressedSize {}; // Compressed size of chunk.
uint64_t m_nUncompressedSize{}; // Uncompressed size of chunk.
bool m_bIsCompressed = false;
VPKEntryDescriptor_t(){};
VPKEntryDescriptor_t(CIOStream* reader);
VPKEntryDescriptor_t(uint32_t nEntryFlags, uint16_t nTextureFlags, uint64_t nArchiveOffset, uint64_t nCompressedSize, uint64_t nUncompressedSize);
VPKChunkDescriptor_t(){};
VPKChunkDescriptor_t(CIOStream* pReader);
VPKChunkDescriptor_t(uint32_t nEntryFlags, uint16_t nTextureFlags, uint64_t nArchiveOffset, uint64_t nCompressedSize, uint64_t nUncompressedSize);
};
struct VPKEntryBlock_t
{
uint32_t m_nCrc32 {}; // Crc32 for the uncompressed block.
uint16_t m_nPreloadData{}; // Preload bytes.
uint16_t m_iArchiveIndex{}; // Index of the archive that contains this block.
vector<VPKEntryDescriptor_t> m_vvEntries {}; // Vector of all the entries of a given block (entries have a size limit of 1 MiB, so anything over is split into separate entries within the same block).
string m_svBlockPath {}; // Path to block within vpk.
uint32_t m_nFileCRC {}; // Crc32 for the uncompressed entry.
uint16_t m_iPreloadSize {}; // Preload bytes.
uint16_t m_iPackFileIndex{}; // Index of the pack file that contains this entry.
vector<VPKChunkDescriptor_t> m_vChunks {}; // Vector of all the chunks of a given entry (chunks have a size limit of 1 MiB, anything over this limit is fragmented into smaller chunks).
string m_svEntryPath {}; // Path to entry within vpk.
VPKEntryBlock_t(CIOStream* pReader, string svBlockPath);
VPKEntryBlock_t(const vector<uint8_t>& vData, int64_t nOffset, uint16_t nPreloadData, uint16_t nArchiveIndex, uint32_t nEntryFlags, uint16_t nTextureFlags, const string& svBlockPath);
VPKEntryBlock_t(CIOStream* pReader, string svEntryPath);
VPKEntryBlock_t(const vector<uint8_t>& vData, int64_t nOffset, uint16_t nPreloadData, uint16_t nArchiveIndex, uint32_t nEntryFlags, uint16_t nTextureFlags, const string& svEntryPath);
};
struct VPKDirHeader_t
@ -87,12 +86,12 @@ struct VPKDirHeader_t
struct VPKDir_t
{
VPKDirHeader_t m_vHeader {}; // Dir header.
uint32_t m_nFileDataSize{}; // File data section size.
vector<VPKEntryBlock_t> m_vvEntryBlocks{}; // Vector of entry blocks.
uint16_t m_iArchiveCount{}; // Highest archive index (archive count-1).
vector<string> m_vsvArchives {}; // Vector of archive file names.
string m_svDirPath {}; // Path to vpk_dir file.
VPKDirHeader_t m_vHeader {}; // Dir header.
uint32_t m_nFileDataSize {}; // File data section size.
vector<VPKEntryBlock_t> m_vEntryBlocks {}; // Vector of entry blocks.
uint16_t m_iPackFileCount{}; // Highest archive index (archive count-1).
vector<string> m_vPackFile {}; // Vector of archive file names.
string m_svDirPath {}; // Path to vpk_dir file.
VPKDir_t(const string& svPath);
VPKDir_t() { m_vHeader.m_nHeaderMarker = VPK_HEADER_MARKER; m_vHeader.m_nMajorVersion = VPK_MAJOR_VERSION; m_vHeader.m_nMinorVersion = VPK_MINOR_VERSION; };
@ -108,7 +107,35 @@ struct VPKPair_t
class CPackedStore
{
size_t m_nEntryCount {}; // Entry per-block incrementor.
public:
void InitLzCompParams(void);
void InitLzDecompParams(void);
VPKDir_t GetDirectoryFile(string svDirectoryFile) const;
string GetPackFile(const string& svPackDirFile, uint16_t iArchiveIndex) const;
vector<VPKEntryBlock_t> GetEntryBlocks(CIOStream* pReader) const;
vector<string> GetEntryPaths(const string& svPathIn) const;
vector<string> GetEntryPaths(const string& svPathIn, const nlohmann::json& jManifest) const;
string GetNameParts(const string& svDirectoryName, int nCaptureGroup) const;
string GetSourceName(const string& svDirectoryName) const;
nlohmann::json GetManifest(const string& svWorkSpace, const string& svManifestName) const;
string FormatEntryPath(string svName, const string& svPath, const string& svExtension) const;
string StripLocalePrefix(const string& svDirectoryFile) const;
VPKPair_t BuildFileName(string svLanguage, string svContext, const string& svPakName, int nPatch) const;
void BuildManifest(const vector<VPKEntryBlock_t>& vBlock, const string& svWorkSpace, const string& svManifestName) const;
void PackAll(const VPKPair_t& vPair, const string& svPathIn, const string& svPathOut, bool bManifestOnly);
void UnpackAll(const VPKDir_t& vDir, const string& svPathOut = "");
void ValidateAdler32PostDecomp(const string& svDirAsset);
void ValidateCRC32PostDecomp(const string& svDirAsset);
private:
size_t m_nChunkCount {}; // Entry per-block incrementor.
lzham_uint32 m_nAdler32_Internal{}; // Internal operation Adler32 file checksum.
lzham_uint32 m_nAdler32 {}; // Pre/post operation Adler32 file checksum.
lzham_uint32 m_nCrc32_Internal {}; // Internal operation Crc32 file checksum.
@ -117,32 +144,7 @@ class CPackedStore
lzham_compress_status_t m_lzCompStatus {}; // LZham compression status.
lzham_decompress_params m_lzDecompParams {}; // LZham decompression parameters.
lzham_decompress_status_t m_lzDecompStatus {}; // LZham decompression status.
std::unordered_map<string, VPKEntryDescriptor_t> m_mEntryHashMap{};
public:
void InitLzCompParams(void);
void InitLzDecompParams(void);
VPKDir_t GetPackDirFile(string svDirectoryFile) const;
string GetPackChunkFile(const string& svPackDirFile, uint16_t iArchiveIndex) const;
vector<VPKEntryBlock_t> GetEntryBlocks(CIOStream* reader) const;
vector<string> GetBlockPaths(const string& svPathIn) const;
vector<string> GetBlockPaths(const string& svPathIn, const nlohmann::json& jManifest) const;
string GetNameParts(const string& svDirectoryName, int nCaptureGroup) const;
string GetSourceName(const string& svDirectoryName) const;
nlohmann::json GetManifest(const string& svWorkSpace, const string& svManifestName) const;
string FormatBlockPath(string svName, const string& svPath, const string& svExtension) const;
string StripLocalePrefix(const string& svPackDirFile) const;
VPKPair_t BuildFileName(string svLanguage, string svContext, const string& svPakName, int nPatch) const;
void BuildManifest(const vector<VPKEntryBlock_t>& vBlock, const string& svWorkSpace, const string& svManifestName) const;
void PackAll(const VPKPair_t& vPair, const string& svPathIn, const string& svPathOut, bool bManifestOnly);
void UnpackAll(const VPKDir_t& vpkDir, const string& svPathOut = "");
void ValidateAdler32PostDecomp(const string& svDirAsset);
void ValidateCRC32PostDecomp(const string& svDirAsset);
std::unordered_map<string, VPKChunkDescriptor_t> m_mChunkHashMap{};
};
///////////////////////////////////////////////////////////////////////////////
extern CPackedStore* g_pPackedStore;

View File

@ -659,7 +659,7 @@ void VPK_Unpack_f(const CCommand& args)
DevMsg(eDLL_T::FS, "*** Starting VPK extraction command for: '%s'\n", args.Arg(1));
VPKDir_t vpk = g_pPackedStore->GetPackDirFile(args.Arg(1));
VPKDir_t vpk = g_pPackedStore->GetDirectoryFile(args.Arg(1));
g_pPackedStore->InitLzDecompParams();
std::thread th([&] { g_pPackedStore->UnpackAll(vpk, ConvertToWinPath(fs_packedstore_workspace->GetString())); });