2022-01-30 10:07:39 +00:00
|
|
|
/*
|
|
|
|
Copyright (C) 2015-2016, 2018, 2020-2021 cc9cii
|
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
|
|
|
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
|
|
|
|
|
|
cc9cii cc9c@iinet.net.au
|
|
|
|
|
|
|
|
Much of the information on the data structures are based on the information
|
|
|
|
from Tes4Mod:Mod_File_Format and Tes5Mod:File_Formats but also refined by
|
|
|
|
trial & error. See http://en.uesp.net/wiki for details.
|
|
|
|
|
|
|
|
*/
|
|
|
|
#include "loadtes4.hpp"
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
#include "common.hpp"
|
|
|
|
#include "reader.hpp"
|
|
|
|
//#include "writer.hpp"
|
|
|
|
|
|
|
|
void ESM4::Header::load(ESM4::Reader& reader)
|
|
|
|
{
|
|
|
|
mFlags = reader.hdr().record.flags; // 0x01 = Rec_ESM, 0x80 = Rec_Localized
|
|
|
|
|
|
|
|
while (reader.getSubRecordHeader())
|
|
|
|
{
|
|
|
|
const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader();
|
|
|
|
switch (subHdr.typeId)
|
|
|
|
{
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("HEDR"):
|
2022-01-30 10:07:39 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
if (!reader.getExact(mData.version) || !reader.getExact(mData.records)
|
|
|
|
|| !reader.getExact(mData.nextObjectId))
|
2022-01-30 10:07:39 +00:00
|
|
|
throw std::runtime_error("TES4 HEDR data read error");
|
2022-09-22 18:26:05 +00:00
|
|
|
if ((size_t)subHdr.dataSize
|
|
|
|
!= sizeof(mData.version) + sizeof(mData.records) + sizeof(mData.nextObjectId))
|
2022-05-02 21:17:24 +00:00
|
|
|
throw std::runtime_error("TES4 HEDR data size mismatch");
|
2022-01-30 10:07:39 +00:00
|
|
|
break;
|
|
|
|
}
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("CNAM"):
|
2022-09-22 18:26:05 +00:00
|
|
|
reader.getZString(mAuthor);
|
|
|
|
break;
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("SNAM"):
|
2022-09-22 18:26:05 +00:00
|
|
|
reader.getZString(mDesc);
|
|
|
|
break;
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("MAST"): // multiple
|
2022-01-30 10:07:39 +00:00
|
|
|
{
|
|
|
|
ESM::MasterData m;
|
|
|
|
if (!reader.getZString(m.name))
|
|
|
|
throw std::runtime_error("TES4 MAST data read error");
|
|
|
|
|
|
|
|
// NOTE: some mods do not have DATA following MAST so can't read DATA here
|
2022-05-01 12:46:47 +00:00
|
|
|
m.size = 0;
|
2022-09-22 18:26:05 +00:00
|
|
|
mMaster.push_back(m);
|
2022-01-30 10:07:39 +00:00
|
|
|
break;
|
|
|
|
}
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("DATA"):
|
2022-01-30 10:07:39 +00:00
|
|
|
{
|
2023-05-20 17:21:32 +00:00
|
|
|
if (mMaster.empty())
|
|
|
|
throw std::runtime_error(
|
|
|
|
"Failed to read TES4 DATA subrecord: there is no preceding MAST subrecord");
|
2022-01-30 10:07:39 +00:00
|
|
|
// WARNING: assumes DATA always follows MAST
|
|
|
|
if (!reader.getExact(mMaster.back().size))
|
|
|
|
throw std::runtime_error("TES4 DATA data read error");
|
|
|
|
break;
|
|
|
|
}
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("ONAM"):
|
2022-01-30 10:07:39 +00:00
|
|
|
{
|
2023-07-30 19:55:36 +00:00
|
|
|
mOverrides.resize(subHdr.dataSize / sizeof(ESM::FormId32));
|
|
|
|
for (ESM::FormId& mOverride : mOverrides)
|
2022-01-30 10:07:39 +00:00
|
|
|
{
|
2023-04-07 00:14:32 +00:00
|
|
|
uint32_t v;
|
|
|
|
if (!reader.getExact(v))
|
2022-01-30 10:07:39 +00:00
|
|
|
throw std::runtime_error("TES4 ONAM data read error");
|
2023-07-30 19:55:36 +00:00
|
|
|
mOverride = ESM::FormId::fromUint32(v);
|
2022-01-30 10:07:39 +00:00
|
|
|
#if 0
|
2022-05-04 20:33:39 +00:00
|
|
|
std::string padding;
|
2022-01-30 10:07:39 +00:00
|
|
|
padding.insert(0, reader.stackSize()*2, ' ');
|
2023-06-01 11:44:44 +00:00
|
|
|
std::cout << padding << "ESM4::Header::ONAM overrides: " << formIdToString(mOverride) << std::endl;
|
2022-01-30 10:07:39 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2024-03-18 00:21:57 +00:00
|
|
|
case ESM::fourCC("INTV"):
|
|
|
|
case ESM::fourCC("INCC"):
|
|
|
|
case ESM::fourCC("OFST"): // Oblivion only?
|
|
|
|
case ESM::fourCC("DELE"): // Oblivion only?
|
|
|
|
case ESM::fourCC("TNAM"): // Fallout 4 (CK only)
|
2024-01-05 01:22:31 +00:00
|
|
|
case ESM::fourCC("MMSB"): // Fallout 76
|
2022-01-30 10:07:39 +00:00
|
|
|
reader.skipSubRecordData();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw std::runtime_error("ESM4::Header::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
// void ESM4::Header::save(ESM4::Writer& writer)
|
2022-01-30 10:07:39 +00:00
|
|
|
//{
|
2022-09-22 18:26:05 +00:00
|
|
|
// }
|