mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-16 15:29:55 +00:00
Make loading and saving script record more robust
* Check the header presence before using it. * Write the header based on the actual content.
This commit is contained in:
parent
447c4bb6a0
commit
5c3ae3d94c
6 changed files with 139 additions and 127 deletions
|
@ -279,11 +279,11 @@ namespace ESM
|
||||||
Script record;
|
Script record;
|
||||||
record.blank();
|
record.blank();
|
||||||
record.mId = generateRandomRefId(33);
|
record.mId = generateRandomRefId(33);
|
||||||
record.mData.mNumShorts = 42;
|
record.mNumShorts = 42;
|
||||||
Script result;
|
Script result;
|
||||||
saveAndLoadRecord(record, CurrentSaveGameFormatVersion, result);
|
saveAndLoadRecord(record, CurrentSaveGameFormatVersion, result);
|
||||||
EXPECT_EQ(result.mId, record.mId);
|
EXPECT_EQ(result.mId, record.mId);
|
||||||
EXPECT_EQ(result.mData.mNumShorts, record.mData.mNumShorts);
|
EXPECT_EQ(result.mNumShorts, record.mNumShorts);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(Esm3SaveLoadRecordTest, playerShouldNotChange)
|
TEST_P(Esm3SaveLoadRecordTest, playerShouldNotChange)
|
||||||
|
@ -417,26 +417,21 @@ namespace ESM
|
||||||
Script record;
|
Script record;
|
||||||
record.blank();
|
record.blank();
|
||||||
record.mId = generateRandomRefId(32);
|
record.mId = generateRandomRefId(32);
|
||||||
record.mData.mNumShorts = 3;
|
record.mNumShorts = 3;
|
||||||
record.mData.mNumFloats = 4;
|
record.mNumFloats = 4;
|
||||||
record.mData.mNumLongs = 5;
|
record.mNumLongs = 5;
|
||||||
record.mData.mScriptDataSize = 13;
|
generateStrings(
|
||||||
generateStrings(std::back_inserter(record.mVarNames),
|
std::back_inserter(record.mVarNames), record.mNumShorts + record.mNumFloats + record.mNumLongs);
|
||||||
record.mData.mNumShorts + record.mData.mNumFloats + record.mData.mNumLongs);
|
generateBytes(std::back_inserter(record.mScriptData), 13);
|
||||||
record.mData.mStringTableSize = std::accumulate(record.mVarNames.begin(), record.mVarNames.end(), 0,
|
|
||||||
[](std::size_t r, const std::string& v) { return r + v.size() + 1; });
|
|
||||||
generateBytes(std::back_inserter(record.mScriptData), record.mData.mScriptDataSize);
|
|
||||||
record.mScriptText = generateRandomString(17);
|
record.mScriptText = generateRandomString(17);
|
||||||
|
|
||||||
Script result;
|
Script result;
|
||||||
saveAndLoadRecord(record, GetParam(), result);
|
saveAndLoadRecord(record, GetParam(), result);
|
||||||
|
|
||||||
EXPECT_EQ(result.mId, record.mId);
|
EXPECT_EQ(result.mId, record.mId);
|
||||||
EXPECT_EQ(result.mData.mNumShorts, record.mData.mNumShorts);
|
EXPECT_EQ(result.mNumShorts, record.mNumShorts);
|
||||||
EXPECT_EQ(result.mData.mNumFloats, record.mData.mNumFloats);
|
EXPECT_EQ(result.mNumFloats, record.mNumFloats);
|
||||||
EXPECT_EQ(result.mData.mNumShorts, record.mData.mNumShorts);
|
EXPECT_EQ(result.mNumShorts, record.mNumShorts);
|
||||||
EXPECT_EQ(result.mData.mScriptDataSize, record.mData.mScriptDataSize);
|
|
||||||
EXPECT_EQ(result.mData.mStringTableSize, record.mData.mStringTableSize);
|
|
||||||
EXPECT_EQ(result.mVarNames, record.mVarNames);
|
EXPECT_EQ(result.mVarNames, record.mVarNames);
|
||||||
EXPECT_EQ(result.mScriptData, record.mScriptData);
|
EXPECT_EQ(result.mScriptData, record.mScriptData);
|
||||||
EXPECT_EQ(result.mScriptText, record.mScriptText);
|
EXPECT_EQ(result.mScriptText, record.mScriptText);
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
#include "labels.hpp"
|
#include "labels.hpp"
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
#include <numeric>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
#include <components/esm3/cellstate.hpp>
|
#include <components/esm3/cellstate.hpp>
|
||||||
|
@ -1181,11 +1182,11 @@ namespace EsmTool
|
||||||
{
|
{
|
||||||
std::cout << " Name: " << mData.mId << std::endl;
|
std::cout << " Name: " << mData.mId << std::endl;
|
||||||
|
|
||||||
std::cout << " Num Shorts: " << mData.mData.mNumShorts << std::endl;
|
std::cout << " Num Shorts: " << mData.mNumShorts << std::endl;
|
||||||
std::cout << " Num Longs: " << mData.mData.mNumLongs << std::endl;
|
std::cout << " Num Longs: " << mData.mNumLongs << std::endl;
|
||||||
std::cout << " Num Floats: " << mData.mData.mNumFloats << std::endl;
|
std::cout << " Num Floats: " << mData.mNumFloats << std::endl;
|
||||||
std::cout << " Script Data Size: " << mData.mData.mScriptDataSize << std::endl;
|
std::cout << " Script Data Size: " << mData.mScriptData.size() << std::endl;
|
||||||
std::cout << " Table Size: " << mData.mData.mStringTableSize << std::endl;
|
std::cout << " Table Size: " << ESM::computeScriptStringTableSize(mData.mVarNames) << std::endl;
|
||||||
|
|
||||||
for (const std::string& variable : mData.mVarNames)
|
for (const std::string& variable : mData.mVarNames)
|
||||||
std::cout << " Variable: " << variable << std::endl;
|
std::cout << " Variable: " << variable << std::endl;
|
||||||
|
|
|
@ -7,8 +7,8 @@ namespace ESSImport
|
||||||
|
|
||||||
void SCPT::load(ESM::ESMReader& esm)
|
void SCPT::load(ESM::ESMReader& esm)
|
||||||
{
|
{
|
||||||
esm.getHNT("SCHD", mSCHD.mName.mData, mSCHD.mData.mNumShorts, mSCHD.mData.mNumLongs, mSCHD.mData.mNumFloats,
|
esm.getHNT("SCHD", mSCHD.mName.mData, mSCHD.mNumShorts, mSCHD.mNumLongs, mSCHD.mNumFloats,
|
||||||
mSCHD.mData.mScriptDataSize, mSCHD.mData.mStringTableSize);
|
mSCHD.mScriptDataSize, mSCHD.mStringTableSize);
|
||||||
|
|
||||||
mSCRI.load(esm);
|
mSCRI.load(esm);
|
||||||
|
|
||||||
|
|
|
@ -19,7 +19,11 @@ namespace ESSImport
|
||||||
struct SCHD
|
struct SCHD
|
||||||
{
|
{
|
||||||
ESM::NAME32 mName;
|
ESM::NAME32 mName;
|
||||||
ESM::Script::SCHD mData;
|
std::uint32_t mNumShorts;
|
||||||
|
std::uint32_t mNumLongs;
|
||||||
|
std::uint32_t mNumFloats;
|
||||||
|
std::uint32_t mScriptDataSize;
|
||||||
|
std::uint32_t mStringTableSize;
|
||||||
};
|
};
|
||||||
|
|
||||||
// A running global script
|
// A running global script
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
#include "loadscpt.hpp"
|
#include "loadscpt.hpp"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
#include <numeric>
|
||||||
|
#include <optional>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
#include <components/debug/debuglog.hpp>
|
#include <components/debug/debuglog.hpp>
|
||||||
|
@ -11,80 +13,93 @@
|
||||||
|
|
||||||
namespace ESM
|
namespace ESM
|
||||||
{
|
{
|
||||||
template <Misc::SameAsWithoutCvref<Script::SCHD> T>
|
namespace
|
||||||
void decompose(T&& v, const auto& f)
|
|
||||||
{
|
{
|
||||||
f(v.mNumShorts, v.mNumLongs, v.mNumFloats, v.mScriptDataSize, v.mStringTableSize);
|
struct SCHD
|
||||||
}
|
|
||||||
|
|
||||||
void Script::loadSCVR(ESMReader& esm)
|
|
||||||
{
|
|
||||||
uint32_t s = mData.mStringTableSize;
|
|
||||||
|
|
||||||
std::vector<char> tmp(s);
|
|
||||||
// not using getHExact, vanilla doesn't seem to mind unused bytes at the end
|
|
||||||
esm.getSubHeader();
|
|
||||||
uint32_t left = esm.getSubSize();
|
|
||||||
if (left < s)
|
|
||||||
esm.fail("SCVR string list is smaller than specified");
|
|
||||||
esm.getExact(tmp.data(), s);
|
|
||||||
if (left > s)
|
|
||||||
esm.skip(left - s); // skip the leftover junk
|
|
||||||
|
|
||||||
// Set up the list of variable names
|
|
||||||
mVarNames.resize(mData.mNumShorts + mData.mNumLongs + mData.mNumFloats);
|
|
||||||
|
|
||||||
// The tmp buffer is a null-byte separated string list, we
|
|
||||||
// just have to pick out one string at a time.
|
|
||||||
char* str = tmp.data();
|
|
||||||
if (tmp.empty())
|
|
||||||
{
|
{
|
||||||
if (mVarNames.size() > 0)
|
/// Data from script-precompling in the editor.
|
||||||
Log(Debug::Warning) << "SCVR with no variable names";
|
/// \warning Do not use them. OpenCS currently does not precompile scripts.
|
||||||
|
std::uint32_t mNumShorts = 0;
|
||||||
|
std::uint32_t mNumLongs = 0;
|
||||||
|
std::uint32_t mNumFloats = 0;
|
||||||
|
std::uint32_t mScriptDataSize = 0;
|
||||||
|
std::uint32_t mStringTableSize = 0;
|
||||||
|
};
|
||||||
|
|
||||||
return;
|
template <Misc::SameAsWithoutCvref<SCHD> T>
|
||||||
|
void decompose(T&& v, const auto& f)
|
||||||
|
{
|
||||||
|
f(v.mNumShorts, v.mNumLongs, v.mNumFloats, v.mScriptDataSize, v.mStringTableSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Support '\r' terminated strings like vanilla. See Bug #1324.
|
void loadVarNames(const SCHD& header, std::vector<std::string>& varNames, ESMReader& esm)
|
||||||
std::replace(tmp.begin(), tmp.end(), '\r', '\0');
|
|
||||||
// Avoid heap corruption
|
|
||||||
if (tmp.back() != '\0')
|
|
||||||
{
|
{
|
||||||
tmp.emplace_back('\0');
|
uint32_t s = header.mStringTableSize;
|
||||||
std::stringstream ss;
|
|
||||||
ss << "Malformed string table";
|
|
||||||
ss << "\n File: " << esm.getName();
|
|
||||||
ss << "\n Record: " << esm.getContext().recName.toStringView();
|
|
||||||
ss << "\n Subrecord: "
|
|
||||||
<< "SCVR";
|
|
||||||
ss << "\n Offset: 0x" << std::hex << esm.getFileOffset();
|
|
||||||
Log(Debug::Verbose) << ss.str();
|
|
||||||
str = tmp.data();
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto tmpEnd = tmp.data() + tmp.size();
|
std::vector<char> tmp(s);
|
||||||
for (size_t i = 0; i < mVarNames.size(); i++)
|
// not using getHExact, vanilla doesn't seem to mind unused bytes at the end
|
||||||
{
|
esm.getSubHeader();
|
||||||
mVarNames[i] = std::string(str);
|
uint32_t left = esm.getSubSize();
|
||||||
str += mVarNames[i].size() + 1;
|
if (left < s)
|
||||||
if (str >= tmpEnd)
|
esm.fail("SCVR string list is smaller than specified");
|
||||||
|
esm.getExact(tmp.data(), s);
|
||||||
|
if (left > s)
|
||||||
|
esm.skip(left - s); // skip the leftover junk
|
||||||
|
|
||||||
|
// Set up the list of variable names
|
||||||
|
varNames.resize(header.mNumShorts + header.mNumLongs + header.mNumFloats);
|
||||||
|
|
||||||
|
// The tmp buffer is a null-byte separated string list, we
|
||||||
|
// just have to pick out one string at a time.
|
||||||
|
if (tmp.empty())
|
||||||
{
|
{
|
||||||
if (str > tmpEnd)
|
if (!varNames.empty())
|
||||||
|
Log(Debug::Warning) << "SCVR with no variable names";
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Support '\r' terminated strings like vanilla. See Bug #1324.
|
||||||
|
std::replace(tmp.begin(), tmp.end(), '\r', '\0');
|
||||||
|
// Avoid heap corruption
|
||||||
|
if (tmp.back() != '\0')
|
||||||
|
{
|
||||||
|
tmp.push_back('\0');
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << "Malformed string table";
|
||||||
|
ss << "\n File: " << esm.getName();
|
||||||
|
ss << "\n Record: " << esm.getContext().recName.toStringView();
|
||||||
|
ss << "\n Subrecord: "
|
||||||
|
<< "SCVR";
|
||||||
|
ss << "\n Offset: 0x" << std::hex << esm.getFileOffset();
|
||||||
|
Log(Debug::Verbose) << ss.str();
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* str = tmp.data();
|
||||||
|
const char* const tmpEnd = tmp.data() + tmp.size();
|
||||||
|
for (size_t i = 0; i < varNames.size(); i++)
|
||||||
|
{
|
||||||
|
varNames[i] = std::string(str);
|
||||||
|
str += varNames[i].size() + 1;
|
||||||
|
if (str >= tmpEnd)
|
||||||
{
|
{
|
||||||
// SCVR subrecord is unused and variable names are determined
|
if (str > tmpEnd)
|
||||||
// from the script source, so an overflow is not fatal.
|
{
|
||||||
std::stringstream ss;
|
// SCVR subrecord is unused and variable names are determined
|
||||||
ss << "String table overflow";
|
// from the script source, so an overflow is not fatal.
|
||||||
ss << "\n File: " << esm.getName();
|
std::stringstream ss;
|
||||||
ss << "\n Record: " << esm.getContext().recName.toStringView();
|
ss << "String table overflow";
|
||||||
ss << "\n Subrecord: "
|
ss << "\n File: " << esm.getName();
|
||||||
<< "SCVR";
|
ss << "\n Record: " << esm.getContext().recName.toStringView();
|
||||||
ss << "\n Offset: 0x" << std::hex << esm.getFileOffset();
|
ss << "\n Subrecord: "
|
||||||
Log(Debug::Verbose) << ss.str();
|
<< "SCVR";
|
||||||
|
ss << "\n Offset: 0x" << std::hex << esm.getFileOffset();
|
||||||
|
Log(Debug::Verbose) << ss.str();
|
||||||
|
}
|
||||||
|
// Get rid of empty strings in the list.
|
||||||
|
varNames.resize(i + 1);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
// Get rid of empty strings in the list.
|
|
||||||
mVarNames.resize(i + 1);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -96,7 +111,7 @@ namespace ESM
|
||||||
|
|
||||||
mVarNames.clear();
|
mVarNames.clear();
|
||||||
|
|
||||||
bool hasHeader = false;
|
std::optional<SCHD> header;
|
||||||
while (esm.hasMoreSubs())
|
while (esm.hasMoreSubs())
|
||||||
{
|
{
|
||||||
esm.getSubName();
|
esm.getSubName();
|
||||||
|
@ -106,22 +121,27 @@ namespace ESM
|
||||||
{
|
{
|
||||||
esm.getSubHeader();
|
esm.getSubHeader();
|
||||||
mId = esm.getMaybeFixedRefIdSize(32);
|
mId = esm.getMaybeFixedRefIdSize(32);
|
||||||
esm.getComposite(mData);
|
esm.getComposite(header.emplace());
|
||||||
|
mNumShorts = header->mNumShorts;
|
||||||
hasHeader = true;
|
mNumLongs = header->mNumLongs;
|
||||||
|
mNumFloats = header->mNumFloats;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case fourCC("SCVR"):
|
case fourCC("SCVR"):
|
||||||
// list of local variables
|
if (!header.has_value())
|
||||||
loadSCVR(esm);
|
esm.fail("SCVR is placed before SCHD record");
|
||||||
|
loadVarNames(*header, mVarNames, esm);
|
||||||
break;
|
break;
|
||||||
case fourCC("SCDT"):
|
case fourCC("SCDT"):
|
||||||
{
|
{
|
||||||
|
if (!header.has_value())
|
||||||
|
esm.fail("SCDT is placed before SCHD record");
|
||||||
|
|
||||||
// compiled script
|
// compiled script
|
||||||
esm.getSubHeader();
|
esm.getSubHeader();
|
||||||
uint32_t subSize = esm.getSubSize();
|
uint32_t subSize = esm.getSubSize();
|
||||||
|
|
||||||
if (subSize != mData.mScriptDataSize)
|
if (subSize != header->mScriptDataSize)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Script data size defined in SCHD subrecord does not match size of SCDT subrecord";
|
ss << "Script data size defined in SCHD subrecord does not match size of SCDT subrecord";
|
||||||
|
@ -147,22 +167,21 @@ namespace ESM
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!hasHeader)
|
if (!header.has_value())
|
||||||
esm.fail("Missing SCHD subrecord");
|
esm.fail("Missing SCHD subrecord");
|
||||||
// Reported script data size is not always trustworthy, so override it with actual data size
|
|
||||||
mData.mScriptDataSize = static_cast<uint32_t>(mScriptData.size());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Script::save(ESMWriter& esm, bool isDeleted) const
|
void Script::save(ESMWriter& esm, bool isDeleted) const
|
||||||
{
|
{
|
||||||
std::string varNameString;
|
|
||||||
if (!mVarNames.empty())
|
|
||||||
for (std::vector<std::string>::const_iterator it = mVarNames.begin(); it != mVarNames.end(); ++it)
|
|
||||||
varNameString.append(*it);
|
|
||||||
|
|
||||||
esm.startSubRecord("SCHD");
|
esm.startSubRecord("SCHD");
|
||||||
esm.writeMaybeFixedSizeRefId(mId, 32);
|
esm.writeMaybeFixedSizeRefId(mId, 32);
|
||||||
esm.writeComposite(mData);
|
esm.writeComposite(SCHD{
|
||||||
|
.mNumShorts = mNumShorts,
|
||||||
|
.mNumLongs = mNumLongs,
|
||||||
|
.mNumFloats = mNumFloats,
|
||||||
|
.mScriptDataSize = static_cast<std::uint32_t>(mScriptData.size()),
|
||||||
|
.mStringTableSize = computeScriptStringTableSize(mVarNames),
|
||||||
|
});
|
||||||
esm.endRecord("SCHD");
|
esm.endRecord("SCHD");
|
||||||
|
|
||||||
if (isDeleted)
|
if (isDeleted)
|
||||||
|
@ -174,15 +193,13 @@ namespace ESM
|
||||||
if (!mVarNames.empty())
|
if (!mVarNames.empty())
|
||||||
{
|
{
|
||||||
esm.startSubRecord("SCVR");
|
esm.startSubRecord("SCVR");
|
||||||
for (std::vector<std::string>::const_iterator it = mVarNames.begin(); it != mVarNames.end(); ++it)
|
for (const std::string& v : mVarNames)
|
||||||
{
|
esm.writeHCString(v);
|
||||||
esm.writeHCString(*it);
|
|
||||||
}
|
|
||||||
esm.endRecord("SCVR");
|
esm.endRecord("SCVR");
|
||||||
}
|
}
|
||||||
|
|
||||||
esm.startSubRecord("SCDT");
|
esm.startSubRecord("SCDT");
|
||||||
esm.write(reinterpret_cast<const char*>(mScriptData.data()), mData.mScriptDataSize);
|
esm.write(reinterpret_cast<const char*>(mScriptData.data()), mScriptData.size());
|
||||||
esm.endRecord("SCDT");
|
esm.endRecord("SCDT");
|
||||||
|
|
||||||
esm.writeHNOString("SCTX", mScriptText);
|
esm.writeHNOString("SCTX", mScriptText);
|
||||||
|
@ -191,9 +208,9 @@ namespace ESM
|
||||||
void Script::blank()
|
void Script::blank()
|
||||||
{
|
{
|
||||||
mRecordFlags = 0;
|
mRecordFlags = 0;
|
||||||
mData.mNumShorts = mData.mNumLongs = mData.mNumFloats = 0;
|
mNumShorts = 0;
|
||||||
mData.mScriptDataSize = 0;
|
mNumLongs = 0;
|
||||||
mData.mStringTableSize = 0;
|
mNumFloats = 0;
|
||||||
|
|
||||||
mVarNames.clear();
|
mVarNames.clear();
|
||||||
mScriptData.clear();
|
mScriptData.clear();
|
||||||
|
@ -204,4 +221,9 @@ namespace ESM
|
||||||
mScriptText = "Begin " + stringId + "\n\nEnd " + stringId + "\n";
|
mScriptText = "Begin " + stringId + "\n\nEnd " + stringId + "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::uint32_t computeScriptStringTableSize(const std::vector<std::string>& varNames)
|
||||||
|
{
|
||||||
|
return std::accumulate(varNames.begin(), varNames.end(), static_cast<std::uint32_t>(0),
|
||||||
|
[](std::uint32_t r, const std::string& v) { return r + 1 + static_cast<std::uint32_t>(v.size()); });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,21 +26,12 @@ namespace ESM
|
||||||
/// Return a string descriptor for this record type. Currently used for debugging / error logs only.
|
/// Return a string descriptor for this record type. Currently used for debugging / error logs only.
|
||||||
static std::string_view getRecordType() { return "Script"; }
|
static std::string_view getRecordType() { return "Script"; }
|
||||||
|
|
||||||
struct SCHD
|
|
||||||
{
|
|
||||||
/// Data from script-precompling in the editor.
|
|
||||||
/// \warning Do not use them. OpenCS currently does not precompile scripts.
|
|
||||||
std::uint32_t mNumShorts;
|
|
||||||
std::uint32_t mNumLongs;
|
|
||||||
std::uint32_t mNumFloats;
|
|
||||||
std::uint32_t mScriptDataSize;
|
|
||||||
std::uint32_t mStringTableSize;
|
|
||||||
};
|
|
||||||
|
|
||||||
uint32_t mRecordFlags;
|
uint32_t mRecordFlags;
|
||||||
RefId mId;
|
RefId mId;
|
||||||
|
|
||||||
SCHD mData;
|
std::uint32_t mNumShorts;
|
||||||
|
std::uint32_t mNumLongs;
|
||||||
|
std::uint32_t mNumFloats;
|
||||||
|
|
||||||
/// Variable names generated by script-precompiling in the editor.
|
/// Variable names generated by script-precompiling in the editor.
|
||||||
/// \warning Do not use this field. OpenCS currently does not precompile scripts.
|
/// \warning Do not use this field. OpenCS currently does not precompile scripts.
|
||||||
|
@ -58,9 +49,8 @@ namespace ESM
|
||||||
|
|
||||||
void blank();
|
void blank();
|
||||||
///< Set record to default state (does not touch the ID/index).
|
///< Set record to default state (does not touch the ID/index).
|
||||||
|
|
||||||
private:
|
|
||||||
void loadSCVR(ESMReader& esm);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
std::uint32_t computeScriptStringTableSize(const std::vector<std::string>& varNames);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in a new issue