2012-09-23 22:11:08 +04:00
|
|
|
#ifndef OPENMW_ESM_READER_H
|
|
|
|
#define OPENMW_ESM_READER_H
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2019-02-22 23:16:47 +04:00
|
|
|
#include <cstdint>
|
2012-07-17 09:27:12 +02:00
|
|
|
#include <cassert>
|
2010-02-18 13:15:43 +01:00
|
|
|
#include <vector>
|
2010-02-19 09:12:49 +01:00
|
|
|
#include <sstream>
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2015-02-22 14:12:05 +01:00
|
|
|
#include <components/files/constrainedfilestream.hpp>
|
2012-07-15 08:31:03 -07:00
|
|
|
|
2010-06-28 12:44:55 -07:00
|
|
|
#include <components/misc/stringops.hpp>
|
2010-02-17 16:45:55 +01:00
|
|
|
|
2010-08-18 18:45:44 +02:00
|
|
|
#include <components/to_utf8/to_utf8.hpp>
|
2013-03-12 09:16:03 +01:00
|
|
|
|
2012-09-23 22:41:41 +04:00
|
|
|
#include "esmcommon.hpp"
|
2013-03-12 09:16:03 +01:00
|
|
|
#include "loadtes3.hpp"
|
2010-08-18 18:45:44 +02:00
|
|
|
|
2010-02-19 14:23:22 +01:00
|
|
|
namespace ESM {
|
|
|
|
|
2010-02-16 16:03:18 +01:00
|
|
|
class ESMReader
|
|
|
|
{
|
2010-02-18 13:15:43 +01:00
|
|
|
public:
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2013-03-12 09:16:03 +01:00
|
|
|
ESMReader();
|
2013-01-03 21:15:18 +01:00
|
|
|
|
2010-02-18 13:15:43 +01:00
|
|
|
/*************************************************************************
|
|
|
|
*
|
|
|
|
* Information retrieval
|
|
|
|
*
|
|
|
|
*************************************************************************/
|
|
|
|
|
2013-03-12 08:15:20 +01:00
|
|
|
int getVer() const { return mHeader.mData.version; }
|
2014-04-28 11:29:57 +02:00
|
|
|
int getRecordCount() const { return mHeader.mData.records; }
|
2015-01-26 13:18:21 +13:00
|
|
|
float getFVer() const { return (mHeader.mData.version == VER_12) ? 1.2f : 1.3f; }
|
2018-09-25 17:16:40 +04:00
|
|
|
const std::string getAuthor() const { return mHeader.mData.author; }
|
|
|
|
const std::string getDesc() const { return mHeader.mData.desc; }
|
2013-09-21 23:06:29 -05:00
|
|
|
const std::vector<Header::MasterData> &getGameFiles() const { return mHeader.mMaster; }
|
2015-01-17 00:11:36 +01:00
|
|
|
const Header& getHeader() const { return mHeader; }
|
2021-06-21 14:22:26 +02:00
|
|
|
int getFormat() const { return mHeader.mFormat; };
|
2013-02-28 02:43:03 +04:00
|
|
|
const NAME &retSubName() const { return mCtx.subName; }
|
2012-06-06 20:29:30 +02:00
|
|
|
uint32_t getSubSize() const { return mCtx.leftSub; }
|
2021-06-21 14:22:26 +02:00
|
|
|
std::string getName() const {return mCtx.filename; };
|
2010-02-18 13:15:43 +01:00
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
*
|
|
|
|
* Opening and closing
|
|
|
|
*
|
|
|
|
*************************************************************************/
|
|
|
|
|
2010-02-25 14:03:03 +01:00
|
|
|
/** Save the current file position and information in a ESM_Context
|
|
|
|
struct
|
|
|
|
*/
|
2011-04-06 20:11:08 +04:00
|
|
|
ESM_Context getContext();
|
2010-02-25 14:03:03 +01:00
|
|
|
|
|
|
|
/** Restore a previously saved context */
|
2011-04-06 20:11:08 +04:00
|
|
|
void restoreContext(const ESM_Context &rc);
|
2010-02-25 14:03:03 +01:00
|
|
|
|
|
|
|
/** Close the file, resets all information. After calling close()
|
|
|
|
the structure may be reused to load a new file.
|
|
|
|
*/
|
2011-04-06 20:11:08 +04:00
|
|
|
void close();
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2010-02-19 09:12:49 +01:00
|
|
|
/// Raw opening. Opens the file and sets everything up but doesn't
|
|
|
|
/// parse the header.
|
2015-02-22 14:12:05 +01:00
|
|
|
void openRaw(Files::IStreamPtr _esm, const std::string &name);
|
2010-02-19 09:12:49 +01:00
|
|
|
|
2010-05-17 17:35:42 +02:00
|
|
|
/// Load ES file from a new stream, parses the header. Closes the
|
|
|
|
/// currently open file first, if any.
|
2015-02-22 14:12:05 +01:00
|
|
|
void open(Files::IStreamPtr _esm, const std::string &name);
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2011-04-06 20:11:08 +04:00
|
|
|
void open(const std::string &file);
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2015-02-22 14:12:05 +01:00
|
|
|
void openRaw(const std::string &filename);
|
2010-02-19 09:12:49 +01:00
|
|
|
|
2013-08-27 15:48:13 +02:00
|
|
|
/// Get the current position in the file. Make sure that the file has been opened!
|
2021-06-21 14:22:26 +02:00
|
|
|
size_t getFileOffset() const { return mEsm->tellg(); };
|
2013-08-27 15:48:13 +02:00
|
|
|
|
2012-10-07 20:00:55 +02:00
|
|
|
// This is a quick hack for multiple esm/esp files. Each plugin introduces its own
|
|
|
|
// terrain palette, but ESMReader does not pass a reference to the correct plugin
|
|
|
|
// to the individual load() methods. This hack allows to pass this reference
|
|
|
|
// indirectly to the load() method.
|
2020-04-19 16:31:54 +02:00
|
|
|
void setIndex(const int index) { mCtx.index = index;}
|
|
|
|
int getIndex() {return mCtx.index;}
|
2013-02-28 02:43:03 +04:00
|
|
|
|
2012-11-17 00:21:51 +01:00
|
|
|
void setGlobalReaderList(std::vector<ESMReader> *list) {mGlobalReaderList = list;}
|
2012-11-25 17:19:29 +01:00
|
|
|
std::vector<ESMReader> *getGlobalReaderList() {return mGlobalReaderList;}
|
2012-10-07 20:00:55 +02:00
|
|
|
|
2020-04-19 16:31:54 +02:00
|
|
|
void addParentFileIndex(int index) { mCtx.parentFileIndices.push_back(index); }
|
|
|
|
const std::vector<int>& getParentFileIndices() const { return mCtx.parentFileIndices; }
|
|
|
|
|
2010-02-16 20:23:54 +01:00
|
|
|
/*************************************************************************
|
|
|
|
*
|
|
|
|
* Medium-level reading shortcuts
|
|
|
|
*
|
|
|
|
*************************************************************************/
|
|
|
|
|
|
|
|
// Read data of a given type, stored in a subrecord of a given name
|
|
|
|
template <typename X>
|
|
|
|
void getHNT(X &x, const char* name)
|
|
|
|
{
|
|
|
|
getSubNameIs(name);
|
|
|
|
getHT(x);
|
|
|
|
}
|
|
|
|
|
2010-02-22 14:09:43 +01:00
|
|
|
// Optional version of getHNT
|
|
|
|
template <typename X>
|
|
|
|
void getHNOT(X &x, const char* name)
|
|
|
|
{
|
2011-04-06 20:11:08 +04:00
|
|
|
if(isNextSub(name))
|
|
|
|
getHT(x);
|
2010-02-22 14:09:43 +01:00
|
|
|
}
|
|
|
|
|
2010-02-18 14:58:50 +01:00
|
|
|
// Version with extra size checking, to make sure the compiler
|
|
|
|
// doesn't mess up our struct padding.
|
|
|
|
template <typename X>
|
|
|
|
void getHNT(X &x, const char* name, int size)
|
|
|
|
{
|
2011-04-06 20:11:08 +04:00
|
|
|
assert(sizeof(X) == size);
|
|
|
|
getSubNameIs(name);
|
|
|
|
getHT(x);
|
2010-02-18 14:58:50 +01:00
|
|
|
}
|
|
|
|
|
2013-01-03 18:51:04 +01:00
|
|
|
template <typename X>
|
|
|
|
void getHNOT(X &x, const char* name, int size)
|
|
|
|
{
|
|
|
|
assert(sizeof(X) == size);
|
|
|
|
if(isNextSub(name))
|
|
|
|
getHT(x);
|
|
|
|
}
|
|
|
|
|
2010-02-16 20:23:54 +01:00
|
|
|
// Get data of a given type/size, including subrecord header
|
|
|
|
template <typename X>
|
|
|
|
void getHT(X &x)
|
|
|
|
{
|
2011-04-06 20:11:08 +04:00
|
|
|
getSubHeader();
|
|
|
|
if (mCtx.leftSub != sizeof(X))
|
2021-06-30 18:29:00 +02:00
|
|
|
reportSubSizeMismatch(sizeof(X), mCtx.leftSub);
|
2011-04-06 20:11:08 +04:00
|
|
|
getT(x);
|
2010-02-16 20:23:54 +01:00
|
|
|
}
|
|
|
|
|
2010-02-20 21:55:51 +01:00
|
|
|
// Version with extra size checking, to make sure the compiler
|
|
|
|
// doesn't mess up our struct padding.
|
|
|
|
template <typename X>
|
|
|
|
void getHT(X &x, int size)
|
|
|
|
{
|
2011-04-06 20:11:08 +04:00
|
|
|
assert(sizeof(X) == size);
|
|
|
|
getHT(x);
|
2010-02-20 21:55:51 +01:00
|
|
|
}
|
|
|
|
|
2010-02-18 14:58:50 +01:00
|
|
|
// Read a string by the given name if it is the next record.
|
2011-04-06 20:11:08 +04:00
|
|
|
std::string getHNOString(const char* name);
|
2010-02-18 14:58:50 +01:00
|
|
|
|
|
|
|
// Read a string with the given sub-record name
|
2011-04-06 20:11:08 +04:00
|
|
|
std::string getHNString(const char* name);
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2010-02-18 14:58:50 +01:00
|
|
|
// Read a string, including the sub-record header (but not the name)
|
2011-04-06 20:11:08 +04:00
|
|
|
std::string getHString();
|
2010-02-17 14:54:22 +01:00
|
|
|
|
2010-02-22 14:09:43 +01:00
|
|
|
// Read the given number of bytes from a subrecord
|
2011-04-06 20:11:08 +04:00
|
|
|
void getHExact(void*p, int size);
|
2010-02-22 14:09:43 +01:00
|
|
|
|
|
|
|
// Read the given number of bytes from a named subrecord
|
2011-04-06 20:11:08 +04:00
|
|
|
void getHNExact(void*p, int size, const char* name);
|
2010-02-22 14:09:43 +01:00
|
|
|
|
2010-02-16 16:03:18 +01:00
|
|
|
/*************************************************************************
|
|
|
|
*
|
2010-02-17 14:54:22 +01:00
|
|
|
* Low level sub-record methods
|
2010-02-16 16:03:18 +01:00
|
|
|
*
|
|
|
|
*************************************************************************/
|
|
|
|
|
2010-02-16 20:23:54 +01:00
|
|
|
// Get the next subrecord name and check if it matches the parameter
|
2011-04-06 20:11:08 +04:00
|
|
|
void getSubNameIs(const char* name);
|
2010-02-16 20:23:54 +01:00
|
|
|
|
2010-02-17 14:54:22 +01:00
|
|
|
/** Checks if the next sub record name matches the parameter. If it
|
|
|
|
does, it is read into 'subName' just as if getSubName() was
|
|
|
|
called. If not, the read name will still be available for future
|
|
|
|
calls to getSubName(), isNextSub() and getSubNameIs().
|
|
|
|
*/
|
2011-04-06 20:11:08 +04:00
|
|
|
bool isNextSub(const char* name);
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2015-06-21 18:18:24 +02:00
|
|
|
bool peekNextSub(const char* name);
|
|
|
|
|
2015-07-16 19:52:31 +03:00
|
|
|
// Store the current subrecord name for the next call of getSubName()
|
2021-06-21 14:22:26 +02:00
|
|
|
void cacheSubName() {mCtx.subCached = true; };
|
2015-07-16 19:52:31 +03:00
|
|
|
|
2010-02-17 14:54:22 +01:00
|
|
|
// Read subrecord name. This gets called a LOT, so I've optimized it
|
|
|
|
// slightly.
|
2011-04-06 20:11:08 +04:00
|
|
|
void getSubName();
|
2010-02-16 20:23:54 +01:00
|
|
|
|
2010-02-17 14:54:22 +01:00
|
|
|
// Skip current sub record, including header (but not including
|
|
|
|
// name.)
|
2011-04-06 20:11:08 +04:00
|
|
|
void skipHSub();
|
2010-02-17 14:54:22 +01:00
|
|
|
|
|
|
|
// Skip sub record and check its size
|
2011-04-06 20:11:08 +04:00
|
|
|
void skipHSubSize(int size);
|
2010-02-17 14:54:22 +01:00
|
|
|
|
2015-01-19 23:29:06 +01:00
|
|
|
// Skip all subrecords until the given subrecord or no more subrecords remaining
|
|
|
|
void skipHSubUntil(const char* name);
|
|
|
|
|
2010-02-21 17:55:12 +01:00
|
|
|
/* Sub-record header. This updates leftRec beyond the current
|
2010-02-16 20:23:54 +01:00
|
|
|
sub-record as well. leftSub contains size of current sub-record.
|
|
|
|
*/
|
2011-04-06 20:11:08 +04:00
|
|
|
void getSubHeader();
|
2010-02-16 20:23:54 +01:00
|
|
|
|
2010-02-17 14:54:22 +01:00
|
|
|
/*************************************************************************
|
|
|
|
*
|
|
|
|
* Low level record methods
|
|
|
|
*
|
|
|
|
*************************************************************************/
|
|
|
|
|
|
|
|
// Get the next record name
|
2011-04-06 20:11:08 +04:00
|
|
|
NAME getRecName();
|
2010-02-17 14:54:22 +01:00
|
|
|
|
2010-02-18 13:15:43 +01:00
|
|
|
// Skip the rest of this record. Assumes the name and header have
|
|
|
|
// already been read
|
2011-04-06 20:11:08 +04:00
|
|
|
void skipRecord();
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2010-02-16 16:03:18 +01:00
|
|
|
/* Read record header. This updatesleftFile BEYOND the data that
|
|
|
|
follows the header, ie beyond the entire record. You should use
|
|
|
|
leftRec to orient yourself inside the record itself.
|
|
|
|
*/
|
2013-05-16 18:50:26 +02:00
|
|
|
void getRecHeader() { getRecHeader(mRecordFlags); }
|
2011-04-06 20:11:08 +04:00
|
|
|
void getRecHeader(uint32_t &flags);
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2012-06-06 20:29:30 +02:00
|
|
|
bool hasMoreRecs() const { return mCtx.leftFile > 0; }
|
|
|
|
bool hasMoreSubs() const { return mCtx.leftRec > 0; }
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2010-02-17 14:54:22 +01:00
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
*
|
|
|
|
* Lowest level data reading and misc methods
|
|
|
|
*
|
|
|
|
*************************************************************************/
|
|
|
|
|
2010-02-16 20:23:54 +01:00
|
|
|
template <typename X>
|
2010-02-18 14:58:50 +01:00
|
|
|
void getT(X &x) { getExact(&x, sizeof(X)); }
|
|
|
|
|
2021-06-21 20:06:45 +00:00
|
|
|
void getExact(void* x, int size) { mEsm->read((char*)x, size); }
|
2010-02-16 20:23:54 +01:00
|
|
|
void getName(NAME &name) { getT(name); }
|
|
|
|
void getUint(uint32_t &u) { getT(u); }
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2010-08-18 18:45:44 +02:00
|
|
|
// Read the next 'size' bytes and return them as a string. Converts
|
|
|
|
// them from native encoding to UTF8 in the process.
|
2011-04-06 20:11:08 +04:00
|
|
|
std::string getString(int size);
|
2010-07-31 19:02:29 +02:00
|
|
|
|
2021-06-21 14:22:26 +02:00
|
|
|
void skip(int bytes) { mEsm->seekg(getFileOffset()+bytes); };
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2010-02-16 16:03:18 +01:00
|
|
|
/// Used for error handling
|
2021-04-29 21:30:25 +02:00
|
|
|
[[noreturn]] void fail(const std::string &msg);
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2013-01-06 01:37:58 +01:00
|
|
|
/// Sets font encoder for ESM strings
|
2021-06-21 14:22:26 +02:00
|
|
|
void setEncoder(ToUTF8::Utf8Encoder* encoder) { mEncoder = encoder; };
|
Added new command line option: "encoding"
Added new command line option: "encoding" which allow to
change font encoding used in game messages.
Currently there are three evailable encodings:
win1250 - Central and Eastern European (languages
that use Latin script, such as Polish,
Czech, Slovak, Hungarian, Slovene, Bosnian,
Croatian, Serbian (Latin script),
Romanian and Albanian)
win1251 - languages that use the Cyrillic alphabet
such as Russian, Bulgarian, Serbian Cyrillic
and others
win1252 - Western European (Latin) - default
Signed-off-by: Lukasz Gromanowski <lgromanowski@gmail.com>
2011-07-17 22:16:50 +02:00
|
|
|
|
2013-05-16 18:50:26 +02:00
|
|
|
/// Get record flags of last record
|
|
|
|
unsigned int getRecordFlags() { return mRecordFlags; }
|
|
|
|
|
2015-04-01 17:02:15 +02:00
|
|
|
size_t getFileSize() const { return mFileSize; }
|
|
|
|
|
2010-02-18 13:15:43 +01:00
|
|
|
private:
|
2021-06-30 18:29:00 +02:00
|
|
|
[[noreturn]] void reportSubSizeMismatch(size_t want, size_t got) {
|
2021-07-21 15:38:41 +02:00
|
|
|
fail("record size mismatch, requested " +
|
2021-06-30 18:29:00 +02:00
|
|
|
std::to_string(want) +
|
|
|
|
", got" +
|
|
|
|
std::to_string(got));
|
|
|
|
}
|
|
|
|
|
2019-02-23 13:35:25 +00:00
|
|
|
void clearCtx();
|
|
|
|
|
2015-02-22 14:12:05 +01:00
|
|
|
Files::IStreamPtr mEsm;
|
2010-02-16 16:03:18 +01:00
|
|
|
|
2011-01-10 16:13:32 -07:00
|
|
|
ESM_Context mCtx;
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2013-05-16 18:50:26 +02:00
|
|
|
unsigned int mRecordFlags;
|
|
|
|
|
2010-02-25 14:03:03 +01:00
|
|
|
// Special file signifier (see SpecialFile enum above)
|
2010-02-18 13:15:43 +01:00
|
|
|
|
2013-01-03 21:15:18 +01:00
|
|
|
// Buffer for ESM strings
|
|
|
|
std::vector<char> mBuffer;
|
|
|
|
|
2013-03-12 08:15:20 +01:00
|
|
|
Header mHeader;
|
|
|
|
|
2012-11-17 00:21:51 +01:00
|
|
|
std::vector<ESMReader> *mGlobalReaderList;
|
2013-01-06 01:37:58 +01:00
|
|
|
ToUTF8::Utf8Encoder* mEncoder;
|
2015-04-01 17:02:15 +02:00
|
|
|
|
|
|
|
size_t mFileSize;
|
|
|
|
|
2010-02-18 13:15:43 +01:00
|
|
|
};
|
2010-02-19 14:23:22 +01:00
|
|
|
}
|
2010-02-18 13:15:43 +01:00
|
|
|
#endif
|