/* OpenMW - The completely unofficial reimplementation of Morrowind Copyright (C) 2008-2010 Nicolay Korslund Email: < korslund@gmail.com > WWW: http://openmw.sourceforge.net/ This file (nif_file.h) is part of the OpenMW package. OpenMW is distributed as free software: you can redistribute it and/or modify it under the terms of the GNU General Public License version 3, as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License version 3 along with this program. If not, see http://www.gnu.org/licenses/ . */ #ifndef _NIF_FILE_H_ #define _NIF_FILE_H_ #include #include #include #include "../tools/slice_array.hpp" #include #include #include #include "record.hpp" #include "nif_types.hpp" using namespace Mangle::Stream; namespace Nif { class NIFFile { enum NIFVersion { VER_MW = 0x04000002 // Morrowind NIFs }; /// Nif file version int ver; /// Input stream StreamPtr inp; /// File name, used for error messages std::string filename; /// Record list std::vector records; /// Parse the file void parse(); public: /// Used for error handling void fail(const std::string &msg) { std::string err = "NIFFile Error: " + msg; err += "\nFile: " + filename; throw str_exception(err); } /// Open a NIF stream. The name is used for error messages. NIFFile(StreamPtr nif, const std::string &name) : filename(name) { /* Load the entire file into memory. This allows us to use direct pointers to the data arrays in the NIF, instead of individually allocating and copying each one. The NIF data is only stored temporarily in memory, since once the mesh data is loaded it is siphoned into OGRE and deleted. For that reason, we might improve this further and use a shared region/pool based allocation scheme in the future, especially since only one NIFFile will ever be loaded at any given time. */ inp = StreamPtr(new BufferStream(nif)); parse(); } ~NIFFile() { for(int i=0; i= 0 && index < records.size()); Record *res = records[index]; assert(res != NULL); return res; } /// Number of records int numRecords() { return records.size(); } /* ************************************************ Parser functions ****************************************************/ void skip(size_t size) { inp->getPtr(size); } template const X* getPtr() { return (const X*)inp->getPtr(sizeof(X)); } template X getType() { return *getPtr(); } unsigned short getShort() { return getType(); } int getInt() { return getType(); } float getFloat() { return getType(); } char getByte() { return getType(); } template SliceArray getArrayLen(int num) { return SliceArray((const X*)inp->getPtr(num*sizeof(X)),num); } template SliceArray getArray() { int len = getInt(); return getArrayLen(len); } SString getString() { return getArray(); } const Vector *getVector() { return getPtr(); } const Matrix *getMatrix() { return getPtr(); } const Transformation *getTrafo() { return getPtr(); } const Vector4 *getVector4() { return getPtr(); } FloatArray getFloatLen(int num) { return getArrayLen(num); } // For fixed-size strings where you already know the size const char *getString(int size) { return (const char*)inp->getPtr(size); } }; } // Namespace #endif