mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-31 06:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			207 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			207 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "loadscpt.hpp"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <sstream>
 | |
| 
 | |
| #include <components/debug/debuglog.hpp>
 | |
| #include <components/misc/concepts.hpp>
 | |
| 
 | |
| #include "esmreader.hpp"
 | |
| #include "esmwriter.hpp"
 | |
| 
 | |
| namespace ESM
 | |
| {
 | |
|     template <Misc::SameAsWithoutCvref<Script::SCHDstruct> T>
 | |
|     void decompose(T&& v, const auto& f)
 | |
|     {
 | |
|         f(v.mNumShorts, v.mNumLongs, v.mNumFloats, v.mScriptDataSize, v.mStringTableSize);
 | |
|     }
 | |
| 
 | |
|     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)
 | |
|                 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.emplace_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();
 | |
|             str = tmp.data();
 | |
|         }
 | |
| 
 | |
|         const auto tmpEnd = tmp.data() + tmp.size();
 | |
|         for (size_t i = 0; i < mVarNames.size(); i++)
 | |
|         {
 | |
|             mVarNames[i] = std::string(str);
 | |
|             str += mVarNames[i].size() + 1;
 | |
|             if (str >= tmpEnd)
 | |
|             {
 | |
|                 if (str > tmpEnd)
 | |
|                 {
 | |
|                     // SCVR subrecord is unused and variable names are determined
 | |
|                     // from the script source, so an overflow is not fatal.
 | |
|                     std::stringstream ss;
 | |
|                     ss << "String table overflow";
 | |
|                     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();
 | |
|                 }
 | |
|                 // Get rid of empty strings in the list.
 | |
|                 mVarNames.resize(i + 1);
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Script::load(ESMReader& esm, bool& isDeleted)
 | |
|     {
 | |
|         isDeleted = false;
 | |
|         mRecordFlags = esm.getRecordFlags();
 | |
| 
 | |
|         mVarNames.clear();
 | |
| 
 | |
|         bool hasHeader = false;
 | |
|         while (esm.hasMoreSubs())
 | |
|         {
 | |
|             esm.getSubName();
 | |
|             switch (esm.retSubName().toInt())
 | |
|             {
 | |
|                 case fourCC("SCHD"):
 | |
|                 {
 | |
|                     esm.getSubHeader();
 | |
|                     mId = esm.getMaybeFixedRefIdSize(32);
 | |
|                     esm.getComposite(mData);
 | |
| 
 | |
|                     hasHeader = true;
 | |
|                     break;
 | |
|                 }
 | |
|                 case fourCC("SCVR"):
 | |
|                     // list of local variables
 | |
|                     loadSCVR(esm);
 | |
|                     break;
 | |
|                 case fourCC("SCDT"):
 | |
|                 {
 | |
|                     // compiled script
 | |
|                     esm.getSubHeader();
 | |
|                     uint32_t subSize = esm.getSubSize();
 | |
| 
 | |
|                     if (subSize != mData.mScriptDataSize)
 | |
|                     {
 | |
|                         std::stringstream ss;
 | |
|                         ss << "Script data size defined in SCHD subrecord does not match size of SCDT subrecord";
 | |
|                         ss << "\n  File: " << esm.getName();
 | |
|                         ss << "\n  Offset: 0x" << std::hex << esm.getFileOffset();
 | |
|                         Log(Debug::Verbose) << ss.str();
 | |
|                     }
 | |
| 
 | |
|                     mScriptData.resize(subSize);
 | |
|                     esm.getExact(mScriptData.data(), mScriptData.size());
 | |
|                     break;
 | |
|                 }
 | |
|                 case fourCC("SCTX"):
 | |
|                     mScriptText = esm.getHString();
 | |
|                     break;
 | |
|                 case SREC_DELE:
 | |
|                     esm.skipHSub();
 | |
|                     isDeleted = true;
 | |
|                     break;
 | |
|                 default:
 | |
|                     esm.fail("Unknown subrecord");
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!hasHeader)
 | |
|             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
 | |
|     {
 | |
|         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.writeMaybeFixedSizeRefId(mId, 32);
 | |
|         esm.writeComposite(mData);
 | |
|         esm.endRecord("SCHD");
 | |
| 
 | |
|         if (isDeleted)
 | |
|         {
 | |
|             esm.writeHNString("DELE", "", 3);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if (!mVarNames.empty())
 | |
|         {
 | |
|             esm.startSubRecord("SCVR");
 | |
|             for (std::vector<std::string>::const_iterator it = mVarNames.begin(); it != mVarNames.end(); ++it)
 | |
|             {
 | |
|                 esm.writeHCString(*it);
 | |
|             }
 | |
|             esm.endRecord("SCVR");
 | |
|         }
 | |
| 
 | |
|         esm.startSubRecord("SCDT");
 | |
|         esm.write(reinterpret_cast<const char*>(mScriptData.data()), mData.mScriptDataSize);
 | |
|         esm.endRecord("SCDT");
 | |
| 
 | |
|         esm.writeHNOString("SCTX", mScriptText);
 | |
|     }
 | |
| 
 | |
|     void Script::blank()
 | |
|     {
 | |
|         mRecordFlags = 0;
 | |
|         mData.mNumShorts = mData.mNumLongs = mData.mNumFloats = 0;
 | |
|         mData.mScriptDataSize = 0;
 | |
|         mData.mStringTableSize = 0;
 | |
| 
 | |
|         mVarNames.clear();
 | |
|         mScriptData.clear();
 | |
|         const std::string& stringId = mId.getRefIdString();
 | |
|         if (stringId.find("::") != std::string::npos)
 | |
|             mScriptText = "Begin \"" + stringId + "\"\n\nEnd " + stringId + "\n";
 | |
|         else
 | |
|             mScriptText = "Begin " + stringId + "\n\nEnd " + stringId + "\n";
 | |
|     }
 | |
| 
 | |
| }
 |