1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-16 14:59:54 +00:00

Fix reading array of zero terminated strings

NIFZ and KFFZ subrecords store multiple strings separated by \0 character.
This commit is contained in:
elsid 2022-09-14 23:56:23 +02:00
parent ae812701ec
commit 6f00641c8a
No known key found for this signature in database
GPG key ID: 4DE04C198CBA7625
4 changed files with 32 additions and 25 deletions

View file

@ -121,15 +121,8 @@ void ESM4::Creature::load(ESM4::Reader& reader)
case ESM4::SUB_NAM1: reader.getZString(mBloodDecal); break; case ESM4::SUB_NAM1: reader.getZString(mBloodDecal); break;
case ESM4::SUB_NIFZ: case ESM4::SUB_NIFZ:
{ {
std::string str; if (!reader.getZeroTerminatedStringArray(mNif))
if (!reader.getZString(str))
throw std::runtime_error ("CREA NIFZ data read error"); throw std::runtime_error ("CREA NIFZ data read error");
std::stringstream ss(str);
std::string file;
while (std::getline(ss, file, '\0')) // split the strings
mNif.push_back(file);
break; break;
} }
case ESM4::SUB_NIFT: case ESM4::SUB_NIFT:
@ -149,15 +142,8 @@ void ESM4::Creature::load(ESM4::Reader& reader)
} }
case ESM4::SUB_KFFZ: case ESM4::SUB_KFFZ:
{ {
std::string str; if (!reader.getZeroTerminatedStringArray(mKf))
if (!reader.getZString(str))
throw std::runtime_error ("CREA KFFZ data read error"); throw std::runtime_error ("CREA KFFZ data read error");
std::stringstream ss(str);
std::string file;
while (std::getline(ss, file, '\0')) // split the strings
mKf.push_back(file);
break; break;
} }
case ESM4::SUB_TPLT: reader.get(mBaseTemplate); break; // FO3 case ESM4::SUB_TPLT: reader.get(mBaseTemplate); break; // FO3

View file

@ -144,19 +144,12 @@ void ESM4::Npc::load(ESM4::Reader& reader)
case ESM4::SUB_MODB: reader.get(mBoundRadius); break; case ESM4::SUB_MODB: reader.get(mBoundRadius); break;
case ESM4::SUB_KFFZ: case ESM4::SUB_KFFZ:
{ {
std::string str;
if (!reader.getZString(str))
throw std::runtime_error ("NPC_ KFFZ data read error");
// Seems to be only below 3, and only happens 3 times while loading TES4: // Seems to be only below 3, and only happens 3 times while loading TES4:
// Forward_SheogorathWithCane.kf // Forward_SheogorathWithCane.kf
// TurnLeft_SheogorathWithCane.kf // TurnLeft_SheogorathWithCane.kf
// TurnRight_SheogorathWithCane.kf // TurnRight_SheogorathWithCane.kf
std::stringstream ss(str); if (!reader.getZeroTerminatedStringArray(mKf))
std::string file; throw std::runtime_error ("NPC_ KFFZ data read error");
while (std::getline(ss, file, '\0')) // split the strings
mKf.push_back(file);
break; break;
} }
case ESM4::SUB_LNAM: reader.get(mHairLength); break; case ESM4::SUB_LNAM: reader.get(mHairLength); break;

View file

@ -693,4 +693,30 @@ bool Reader::getStringImpl(std::string& str, std::size_t size,
return false; // FIXME: throw instead? return false; // FIXME: throw instead?
} }
bool Reader::getZeroTerminatedStringArray(std::vector<std::string>& values)
{
const std::size_t size = mCtx.subRecordHeader.dataSize;
std::string input(size, '\0');
mStream->read(input.data(), size);
if (mStream->gcount() != static_cast<std::streamsize>(size))
return false;
std::string_view inputView(input.data(), input.size());
std::string buffer;
while (true)
{
std::string_view value(inputView.data());
const std::size_t next = inputView.find_first_not_of('\0', value.size());
if (mEncoder != nullptr)
value = mEncoder->getUtf8(value, ToUTF8::BufferAllocationPolicy::UseGrowFactor, buffer);
values.emplace_back(value);
if (next == std::string_view::npos)
break;
inputView = inputView.substr(next);
}
return true;
}
} }

View file

@ -293,6 +293,8 @@ namespace ESM4 {
return getStringImpl(str, mCtx.subRecordHeader.dataSize, *mStream, mEncoder); return getStringImpl(str, mCtx.subRecordHeader.dataSize, *mStream, mEncoder);
} }
bool getZeroTerminatedStringArray(std::vector<std::string>& values);
void enterGroup(); void enterGroup();
void exitGroupCheck(); void exitGroupCheck();