/* Copyright (C) 2016, 2018-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 "loadrace.hpp" #include #include #include "reader.hpp" //#include "writer.hpp" void ESM4::Race::load(ESM4::Reader& reader) { mId = reader.getFormIdFromHeader(); mFlags = reader.hdr().record.flags; std::uint32_t esmVer = reader.esmVersion(); bool isTES4 = (esmVer == ESM::VER_080 || esmVer == ESM::VER_100) && !reader.hasFormVersion(); bool isFONV = esmVer == ESM::VER_132 || esmVer == ESM::VER_133 || esmVer == ESM::VER_134; bool isFO3 = false; bool isMale = false; int curr_part = -1; // 0 = head, 1 = body, 2 = egt, 3 = hkx std::uint32_t currentIndex = 0xffffffff; while (reader.getSubRecordHeader()) { const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader(); // std::cout << "RACE " << ESM::printName(subHdr.typeId) << std::endl; switch (subHdr.typeId) { case ESM::fourCC("EDID"): { reader.getZString(mEditorId); // TES4 // Sheogorath 0x0005308E // GoldenSaint 0x0001208F // DarkSeducer 0x0001208E // VampireRace 0x00000019 // Dremora 0x00038010 // Argonian 0x00023FE9 // Nord 0x000224FD // Breton 0x000224FC // WoodElf 0x000223C8 // Khajiit 0x000223C7 // DarkElf 0x000191C1 // Orc 0x000191C0 // HighElf 0x00019204 // Redguard 0x00000D43 // Imperial 0x00000907 break; } case ESM::fourCC("FULL"): reader.getLocalizedString(mFullName); break; case ESM::fourCC("DESC"): { if (subHdr.dataSize == 1) // FO3? { reader.skipSubRecordData(); break; } reader.getLocalizedString(mDesc); break; } case ESM::fourCC("SPLO"): // bonus spell formid (TES5 may have SPCT and multiple SPLO) reader.getFormId(mBonusSpells.emplace_back()); break; case ESM::fourCC("DATA"): // ?? different length for TES5 { // DATA:size 128 // 0f 0f ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 00 00 // 9a 99 99 3f 00 00 80 3f 00 00 80 3f 00 00 80 3f // 48 89 10 00 00 00 40 41 00 00 00 00 00 00 48 43 // 00 00 48 43 00 00 80 3f 9a 99 19 3f 00 00 00 40 // 01 00 00 00 ff ff ff ff ff ff ff ff 00 00 00 00 // ff ff ff ff 00 00 00 00 00 00 20 41 00 00 a0 40 // 00 00 a0 40 00 00 80 42 ff ff ff ff 00 00 00 00 // 00 00 00 00 9a 99 99 3e 00 00 a0 40 02 00 00 00 #if 0 unsigned char mDataBuf[256/*bufSize*/]; reader.get(mDataBuf, subHdr.dataSize); std::ostringstream ss; ss << ESM::printName(subHdr.typeId) << ":size " << subHdr.dataSize << "\n"; for (unsigned int i = 0; i < subHdr.dataSize; ++i) { //if (mDataBuf[i] > 64 && mDataBuf[i] < 91) //ss << (char)(mDataBuf[i]) << " "; //else ss << std::setfill('0') << std::setw(2) << std::hex << (int)(mDataBuf[i]); if ((i & 0x000f) == 0xf) ss << "\n"; else if (i < 256/*bufSize*/-1) ss << " "; } std::cout << ss.str() << std::endl; #else if (subHdr.dataSize == 36) // TES4/FO3/FONV { if (!isTES4 && !isFONV && !mIsTES5) isFO3 = true; std::uint8_t skill; std::uint8_t bonus; for (unsigned int i = 0; i < 8; ++i) { reader.get(skill); reader.get(bonus); mSkillBonus[static_cast(skill)] = bonus; } reader.get(mHeightMale); reader.get(mHeightFemale); reader.get(mWeightMale); reader.get(mWeightFemale); reader.get(mRaceFlags); } else if (subHdr.dataSize == 128 || subHdr.dataSize == 164) // TES5 { mIsTES5 = true; std::uint8_t skill; std::uint8_t bonus; for (unsigned int i = 0; i < 7; ++i) { reader.get(skill); reader.get(bonus); mSkillBonus[static_cast(skill)] = bonus; } std::uint16_t unknown; reader.get(unknown); reader.get(mHeightMale); reader.get(mHeightFemale); reader.get(mWeightMale); reader.get(mWeightFemale); reader.get(mRaceFlags); // FIXME float dummy; reader.get(dummy); // starting health reader.get(dummy); // starting magicka reader.get(dummy); // starting stamina reader.get(dummy); // base carry weight reader.get(dummy); // base mass reader.get(dummy); // accleration rate reader.get(dummy); // decleration rate uint32_t dummy2; reader.get(dummy2); // size reader.get(dummy2); // head biped object reader.get(dummy2); // hair biped object reader.get(dummy); // injured health % (0.f..1.f) reader.get(dummy2); // shield biped object reader.get(dummy); // health regen reader.get(dummy); // magicka regen reader.get(dummy); // stamina regen reader.get(dummy); // unarmed damage reader.get(dummy); // unarmed reach reader.get(dummy2); // body biped object reader.get(dummy); // aim angle tolerance reader.get(dummy2); // unknown reader.get(dummy); // angular accleration rate reader.get(dummy); // angular tolerance reader.get(dummy2); // flags if (subHdr.dataSize == 164) { reader.get(dummy2); // unknown 1 reader.get(dummy2); // unknown 2 reader.get(dummy2); // unknown 3 reader.get(dummy2); // unknown 4 reader.get(dummy2); // unknown 5 reader.get(dummy2); // unknown 6 reader.get(dummy2); // unknown 7 reader.get(dummy2); // unknown 8 reader.get(dummy2); // unknown 9 } } else { reader.skipSubRecordData(); // std::cout << "RACE " << ESM::printName(subHdr.typeId) << " skipping..." << subHdr.dataSize << // std::endl; } #endif break; } case ESM::fourCC("DNAM"): { reader.getFormId(mDefaultHair[0]); // male reader.getFormId(mDefaultHair[1]); // female break; } case ESM::fourCC("CNAM"): // CNAM SNAM VNAM // Sheogorath 0x0 0000 98 2b 10011000 00101011 // Golden Saint 0x3 0011 26 46 00100110 01000110 // Dark Seducer 0xC 1100 df 55 11011111 01010101 // Vampire Race 0x0 0000 77 44 01110111 10001000 // Dremora 0x7 0111 bf 32 10111111 00110010 // Argonian 0x0 0000 dc 3c 11011100 00111100 // Nord 0x5 0101 b6 03 10110110 00000011 // Breton 0x5 0101 48 1d 01001000 00011101 00000000 00000907 (Imperial) // Wood Elf 0xD 1101 2e 4a 00101110 01001010 00019204 00019204 (HighElf) // khajiit 0x5 0101 54 5b 01010100 01011011 00023FE9 00023FE9 (Argonian) // Dark Elf 0x0 0000 72 54 01110010 01010100 00019204 00019204 (HighElf) // Orc 0xC 1100 74 09 01110100 00001001 000224FD 000224FD (Nord) // High Elf 0xF 1111 e6 21 11100110 00100001 // Redguard 0xD 1101 a9 61 10101001 01100001 // Imperial 0xD 1101 8e 35 10001110 00110101 { reader.skipSubRecordData(); break; } case ESM::fourCC("PNAM"): reader.get(mFaceGenMainClamp); break; // 0x40A00000 = 5.f case ESM::fourCC("UNAM"): reader.get(mFaceGenFaceClamp); break; // 0x40400000 = 3.f case ESM::fourCC("ATTR"): // Only in TES4? { if (subHdr.dataSize == 2) // FO3? { reader.skipSubRecordData(); break; } reader.get(mAttribMale.strength); reader.get(mAttribMale.intelligence); reader.get(mAttribMale.willpower); reader.get(mAttribMale.agility); reader.get(mAttribMale.speed); reader.get(mAttribMale.endurance); reader.get(mAttribMale.personality); reader.get(mAttribMale.luck); reader.get(mAttribFemale.strength); reader.get(mAttribFemale.intelligence); reader.get(mAttribFemale.willpower); reader.get(mAttribFemale.agility); reader.get(mAttribFemale.speed); reader.get(mAttribFemale.endurance); reader.get(mAttribFemale.personality); reader.get(mAttribFemale.luck); break; } // [0..9]-> ICON // NAM0 -> INDX -> MODL --+ // ^ -> MODB | // | | // +-------------+ // case ESM::fourCC("NAM0"): // start marker head data /* 1 */ { curr_part = 0; // head part if (isFO3 || isFONV) { mHeadParts.resize(8); mHeadPartsFemale.resize(8); } else if (isTES4) mHeadParts.resize(9); // assumed based on Construction Set else // Optimized for TES5 { mHeadPartIdsMale.resize(5); mHeadPartIdsFemale.resize(5); } currentIndex = 0xffffffff; break; } case ESM::fourCC("INDX"): { reader.get(currentIndex); // FIXME: below check is rather useless // if (headpart) //{ // if (currentIndex > 8) // throw std::runtime_error("ESM4::RACE::load - too many head part " + currentIndex); //} // else // bodypart //{ // if (currentIndex > 4) // throw std::runtime_error("ESM4::RACE::load - too many body part " + currentIndex); //} break; } case ESM::fourCC("MODL"): { if (currentIndex == 0xffffffff) { reader.skipSubRecordData(); } else if (curr_part == 0) // head part { if (isMale || isTES4) reader.getZString(mHeadParts[currentIndex].mesh); else reader.getZString(mHeadPartsFemale[currentIndex].mesh); // TODO: check TES4 // TES5 keeps head part formid in mHeadPartIdsMale and mHeadPartIdsFemale } else if (curr_part == 1) // body part { if (isMale) reader.getZString(mBodyPartsMale[currentIndex].mesh); else reader.getZString(mBodyPartsFemale[currentIndex].mesh); // TES5 seems to have no body parts at all, instead keep EGT models } // else if (curr_part == 2) // egt // { // // std::cout << mEditorId << " egt " << currentIndex << std::endl; // FIXME // reader.skipSubRecordData(); // FIXME TES5 egt // } else { // std::cout << mEditorId << " hkx " << currentIndex << std::endl; // FIXME reader.skipSubRecordData(); // FIXME TES5 hkx } break; } case ESM::fourCC("MODB"): reader.skipSubRecordData(); break; // always 0x0000? case ESM::fourCC("ICON"): { if (currentIndex == 0xffffffff) { reader.skipSubRecordData(); } else if (curr_part == 0) // head part { if (isMale || isTES4) reader.getZString(mHeadParts[currentIndex].texture); else reader.getZString(mHeadPartsFemale[currentIndex].texture); // TODO: check TES4 } else if (curr_part == 1) // body part { if (isMale) reader.getZString(mBodyPartsMale[currentIndex].texture); else reader.getZString(mBodyPartsFemale[currentIndex].texture); } else reader.skipSubRecordData(); // FIXME TES5 break; } // case ESM::fourCC("NAM1"): // start marker body data /* 4 */ { if (isFO3 || isFONV) { curr_part = 1; // body part mBodyPartsMale.resize(4); mBodyPartsFemale.resize(4); } else if (isTES4) { curr_part = 1; // body part mBodyPartsMale.resize(5); // 0 = upper body, 1 = legs, 2 = hands, 3 = feet, 4 = tail mBodyPartsFemale.resize(5); // 0 = upper body, 1 = legs, 2 = hands, 3 = feet, 4 = tail } else // TES5 curr_part = 2; // for TES5 NAM1 indicates the start of EGT model if (isTES4) currentIndex = 4; // FIXME: argonian tail mesh without preceeding INDX else currentIndex = 0xffffffff; break; } case ESM::fourCC("MNAM"): isMale = true; break; /* 2, 5, 7 */ case ESM::fourCC("FNAM"): isMale = false; break; /* 3, 6, 8 */ // case ESM::fourCC("HNAM"): { // FIXME: this is a texture name in FO4 if (subHdr.dataSize % sizeof(ESM::FormId32) != 0) reader.skipSubRecordData(); else { std::size_t numHairChoices = subHdr.dataSize / sizeof(ESM::FormId32); mHairChoices.resize(numHairChoices); for (unsigned int i = 0; i < numHairChoices; ++i) reader.getFormId(mHairChoices.at(i)); } break; } case ESM::fourCC("ENAM"): { std::size_t numEyeChoices = subHdr.dataSize / sizeof(ESM::FormId32); mEyeChoices.resize(numEyeChoices); for (unsigned int i = 0; i < numEyeChoices; ++i) reader.getFormId(mEyeChoices.at(i)); break; } case ESM::fourCC("FGGS"): { if (isMale || isTES4) { mSymShapeModeCoefficients.resize(50); for (std::size_t i = 0; i < 50; ++i) reader.get(mSymShapeModeCoefficients.at(i)); } else { mSymShapeModeCoeffFemale.resize(50); for (std::size_t i = 0; i < 50; ++i) reader.get(mSymShapeModeCoeffFemale.at(i)); } break; } case ESM::fourCC("FGGA"): { if (isMale || isTES4) { mAsymShapeModeCoefficients.resize(30); for (std::size_t i = 0; i < 30; ++i) reader.get(mAsymShapeModeCoefficients.at(i)); } else { mAsymShapeModeCoeffFemale.resize(30); for (std::size_t i = 0; i < 30; ++i) reader.get(mAsymShapeModeCoeffFemale.at(i)); } break; } case ESM::fourCC("FGTS"): { if (isMale || isTES4) { mSymTextureModeCoefficients.resize(50); for (std::size_t i = 0; i < 50; ++i) reader.get(mSymTextureModeCoefficients.at(i)); } else { mSymTextureModeCoeffFemale.resize(50); for (std::size_t i = 0; i < 50; ++i) reader.get(mSymTextureModeCoeffFemale.at(i)); } break; } // case ESM::fourCC("SNAM"): // skipping...2 // only in TES4? { reader.skipSubRecordData(); break; } case ESM::fourCC("XNAM"): { ESM::FormId race; std::int32_t adjustment; reader.getFormId(race); reader.get(adjustment); mDisposition[race] = adjustment; break; } case ESM::fourCC("VNAM"): { if (subHdr.dataSize == 8) // TES4 { reader.getFormId(mVNAM[0]); // For TES4 seems to be 2 race formids reader.getFormId(mVNAM[1]); } else if (subHdr.dataSize == 4) // TES5 { // equipment type flags meant to be uint32 ??? GLOB reference? shows up in // SCRO in sript records and CTDA in INFO records uint32_t dummy; reader.get(dummy); } else { reader.skipSubRecordData(); // std::cout << "RACE " << ESM::printName(subHdr.typeId) << " skipping..." << subHdr.dataSize << // std::endl; } break; } // case ESM::fourCC("ANAM"): // TES5 { if (isMale) reader.getZString(mModelMale); else reader.getZString(mModelFemale); break; } case ESM::fourCC("KSIZ"): reader.get(mNumKeywords); break; case ESM::fourCC("KWDA"): { ESM::FormId formid; for (unsigned int i = 0; i < mNumKeywords; ++i) reader.getFormId(formid); break; } // case ESM::fourCC("WNAM"): // ARMO FormId { reader.getFormId(mSkin); // std::cout << mEditorId << " skin " << formIdToString(mSkin) << std::endl; // FIXME break; } case ESM::fourCC("BODT"): // body template { reader.get(mBodyTemplate.bodyPart); reader.get(mBodyTemplate.flags); reader.get(mBodyTemplate.unknown1); // probably padding reader.get(mBodyTemplate.unknown2); // probably padding reader.get(mBodyTemplate.unknown3); // probably padding reader.get(mBodyTemplate.type); break; } case ESM::fourCC("BOD2"): { if (subHdr.dataSize == 8 || subHdr.dataSize == 4) // TES5, FO4 { reader.get(mBodyTemplate.bodyPart); mBodyTemplate.flags = 0; mBodyTemplate.unknown1 = 0; // probably padding mBodyTemplate.unknown2 = 0; // probably padding mBodyTemplate.unknown3 = 0; // probably padding mBodyTemplate.type = 0; if (subHdr.dataSize == 8) reader.get(mBodyTemplate.type); } else { reader.skipSubRecordData(); } break; } case ESM::fourCC("HEAD"): // TES5 { ESM::FormId formId; reader.getFormId(formId); if (currentIndex != 0xffffffff) { // FIXME: no order? head, mouth, eyes, brow, hair if (isMale) { if (currentIndex >= mHeadPartIdsMale.size()) mHeadPartIdsMale.resize(currentIndex + 1); mHeadPartIdsMale[currentIndex] = formId; } else { if (currentIndex >= mHeadPartIdsFemale.size()) mHeadPartIdsFemale.resize(currentIndex + 1); mHeadPartIdsFemale[currentIndex] = formId; } } // std::cout << mEditorId << (isMale ? " male head " : " female head ") // << formIdToString(formId) << " " << currentIndex << std::endl; // FIXME break; } case ESM::fourCC("NAM3"): // start of hkx model { curr_part = 3; // for TES5 NAM3 indicates the start of hkx model break; } // Not sure for what this is used - maybe to indicate which slots are in use? e.g.: // // ManakinRace HEAD // ManakinRace Hair // ManakinRace BODY // ManakinRace Hands // ManakinRace Forearms // ManakinRace Amulet // ManakinRace Ring // ManakinRace Feet // ManakinRace Calves // ManakinRace SHIELD // ManakinRace // ManakinRace LongHair // ManakinRace Circlet // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace DecapitateHead // ManakinRace Decapitate // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace // ManakinRace FX0 case ESM::fourCC("NAME"): // TES5 biped object names (x32) { std::string name; reader.getZString(name); // std::cout << mEditorId << " " << name << std::endl; break; } case ESM::fourCC("MTNM"): // movement type case ESM::fourCC("ATKD"): // attack data case ESM::fourCC("ATKE"): // attach event case ESM::fourCC("GNAM"): // body part data case ESM::fourCC("NAM4"): // material type case ESM::fourCC("NAM5"): // unarmed impact? case ESM::fourCC("LNAM"): // close loot sound case ESM::fourCC("QNAM"): // equipment slot formid case ESM::fourCC("HCLF"): // default hair colour case ESM::fourCC("UNES"): // unarmed equipment slot formid case ESM::fourCC("TINC"): case ESM::fourCC("TIND"): case ESM::fourCC("TINI"): case ESM::fourCC("TINL"): case ESM::fourCC("TINP"): case ESM::fourCC("TINT"): case ESM::fourCC("TINV"): case ESM::fourCC("TIRS"): case ESM::fourCC("PHWT"): case ESM::fourCC("AHCF"): case ESM::fourCC("AHCM"): case ESM::fourCC("MPAI"): case ESM::fourCC("MPAV"): case ESM::fourCC("DFTF"): case ESM::fourCC("DFTM"): case ESM::fourCC("FLMV"): case ESM::fourCC("FTSF"): case ESM::fourCC("FTSM"): case ESM::fourCC("MTYP"): case ESM::fourCC("NAM7"): case ESM::fourCC("NAM8"): case ESM::fourCC("PHTN"): case ESM::fourCC("RNAM"): case ESM::fourCC("RNMV"): case ESM::fourCC("RPRF"): case ESM::fourCC("RPRM"): case ESM::fourCC("SNMV"): case ESM::fourCC("SPCT"): case ESM::fourCC("SPED"): case ESM::fourCC("SWMV"): case ESM::fourCC("WKMV"): case ESM::fourCC("SPMV"): case ESM::fourCC("ATKR"): case ESM::fourCC("CTDA"): case ESM::fourCC("CIS1"): case ESM::fourCC("CIS2"): case ESM::fourCC("MODT"): // Model data case ESM::fourCC("MODC"): case ESM::fourCC("MODS"): case ESM::fourCC("MODF"): // Model data end // case ESM::fourCC("YNAM"): // FO3 case ESM::fourCC("NAM2"): // FO3 case ESM::fourCC("VTCK"): // FO3 case ESM::fourCC("MODD"): // FO3 case ESM::fourCC("ONAM"): // FO3 case ESM::fourCC("APPR"): // FO4 case ESM::fourCC("ATKS"): // FO4 case ESM::fourCC("ATKT"): // FO4 case ESM::fourCC("ATKW"): // FO4 case ESM::fourCC("BMMP"): // FO4 case ESM::fourCC("BSMB"): // FO4 case ESM::fourCC("BSMP"): // FO4 case ESM::fourCC("BSMS"): // FO4 case ESM::fourCC("FMRI"): // FO4 case ESM::fourCC("FMRN"): // FO4 case ESM::fourCC("HLTX"): // FO4 case ESM::fourCC("MLSI"): // FO4 case ESM::fourCC("MPGN"): // FO4 case ESM::fourCC("MPGS"): // FO4 case ESM::fourCC("MPPC"): // FO4 case ESM::fourCC("MPPF"): // FO4 case ESM::fourCC("MPPI"): // FO4 case ESM::fourCC("MPPK"): // FO4 case ESM::fourCC("MPPM"): // FO4 case ESM::fourCC("MPPN"): // FO4 case ESM::fourCC("MPPT"): // FO4 case ESM::fourCC("MSID"): // FO4 case ESM::fourCC("MSM0"): // FO4 case ESM::fourCC("MSM1"): // FO4 case ESM::fourCC("NNAM"): // FO4 case ESM::fourCC("NTOP"): // FO4 case ESM::fourCC("PRPS"): // FO4 case ESM::fourCC("PTOP"): // FO4 case ESM::fourCC("QSTI"): // FO4 case ESM::fourCC("RBPC"): // FO4 case ESM::fourCC("SADD"): // FO4 case ESM::fourCC("SAKD"): // FO4 case ESM::fourCC("SAPT"): // FO4 case ESM::fourCC("SGNM"): // FO4 case ESM::fourCC("SRAC"): // FO4 case ESM::fourCC("SRAF"): // FO4 case ESM::fourCC("STCP"): // FO4 case ESM::fourCC("STKD"): // FO4 case ESM::fourCC("TETI"): // FO4 case ESM::fourCC("TTEB"): // FO4 case ESM::fourCC("TTEC"): // FO4 case ESM::fourCC("TTED"): // FO4 case ESM::fourCC("TTEF"): // FO4 case ESM::fourCC("TTET"): // FO4 case ESM::fourCC("TTGE"): // FO4 case ESM::fourCC("TTGP"): // FO4 case ESM::fourCC("UNWP"): // FO4 case ESM::fourCC("WMAP"): // FO4 case ESM::fourCC("ZNAM"): // FO4 reader.skipSubRecordData(); break; default: throw std::runtime_error("ESM4::RACE::load - Unknown subrecord " + ESM::printName(subHdr.typeId)); } } } // void ESM4::Race::save(ESM4::Writer& writer) const //{ // } // void ESM4::Race::blank() //{ // }