2010-01-04 13:48:18 +00:00
|
|
|
/*
|
|
|
|
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_
|
|
|
|
|
2010-06-25 20:28:59 +00:00
|
|
|
#include <libs/mangle/stream/stream.hpp>
|
|
|
|
#include <libs/mangle/stream/filters/buffer_stream.hpp>
|
2010-06-28 19:44:55 +00:00
|
|
|
#include <components/misc/slice_array.hpp>
|
2010-01-04 17:52:20 +00:00
|
|
|
|
2010-09-02 20:30:39 +00:00
|
|
|
#include <stdexcept>
|
2010-01-04 13:48:18 +00:00
|
|
|
#include <vector>
|
2010-01-04 17:52:20 +00:00
|
|
|
#include <string>
|
2010-01-06 11:28:37 +00:00
|
|
|
#include <assert.h>
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2010-06-03 18:44:55 +00:00
|
|
|
#include "record.hpp"
|
|
|
|
#include "nif_types.hpp"
|
2010-01-04 18:35:11 +00:00
|
|
|
|
2010-01-04 13:48:18 +00:00
|
|
|
using namespace Mangle::Stream;
|
|
|
|
|
2010-01-06 11:28:37 +00:00
|
|
|
namespace Nif
|
|
|
|
{
|
|
|
|
|
2010-01-04 13:48:18 +00:00
|
|
|
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
|
2010-01-06 11:28:37 +00:00
|
|
|
std::vector<Record*> records;
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2010-01-07 18:11:03 +00:00
|
|
|
/// Parse the file
|
|
|
|
void parse();
|
|
|
|
|
|
|
|
public:
|
2010-01-04 13:48:18 +00:00
|
|
|
/// Used for error handling
|
|
|
|
void fail(const std::string &msg)
|
|
|
|
{
|
|
|
|
std::string err = "NIFFile Error: " + msg;
|
|
|
|
err += "\nFile: " + filename;
|
2010-09-02 20:30:39 +00:00
|
|
|
throw std::runtime_error(err);
|
2010-01-04 13:48:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// 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();
|
|
|
|
}
|
|
|
|
|
2010-01-17 12:37:20 +00:00
|
|
|
~NIFFile()
|
|
|
|
{
|
2010-06-25 12:41:49 +00:00
|
|
|
for(std::size_t i=0; i<records.size(); i++)
|
2010-01-17 12:37:20 +00:00
|
|
|
{
|
|
|
|
delete records[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-12 13:46:44 +00:00
|
|
|
/// Get a given record
|
2010-01-06 11:28:37 +00:00
|
|
|
Record *getRecord(int index)
|
|
|
|
{
|
2010-06-25 12:41:49 +00:00
|
|
|
assert(index >= 0 && index < static_cast<int> (records.size()));
|
2010-01-06 11:28:37 +00:00
|
|
|
Record *res = records[index];
|
|
|
|
assert(res != NULL);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2010-01-12 13:46:44 +00:00
|
|
|
/// Number of records
|
|
|
|
int numRecords() { return records.size(); }
|
2010-01-04 13:48:18 +00:00
|
|
|
|
|
|
|
/* ************************************************
|
|
|
|
|
|
|
|
Parser functions
|
|
|
|
|
|
|
|
****************************************************/
|
|
|
|
|
2010-01-06 15:22:34 +00:00
|
|
|
void skip(size_t size) { inp->getPtr(size); }
|
|
|
|
|
2010-01-06 14:00:08 +00:00
|
|
|
template<class X> const X* getPtr() { return (const X*)inp->getPtr(sizeof(X)); }
|
|
|
|
template<class X> X getType() { return *getPtr<X>(); }
|
2010-01-07 18:11:03 +00:00
|
|
|
unsigned short getShort() { return getType<unsigned short>(); }
|
2010-01-04 13:48:18 +00:00
|
|
|
int getInt() { return getType<int>(); }
|
2010-01-06 15:22:34 +00:00
|
|
|
float getFloat() { return getType<float>(); }
|
|
|
|
char getByte() { return getType<char>(); }
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2010-01-06 19:42:32 +00:00
|
|
|
template<class X>
|
2011-06-19 16:14:03 +00:00
|
|
|
Misc::SliceArray<X> getArrayLen(int num)
|
|
|
|
{ return Misc::SliceArray<X>((const X*)inp->getPtr(num*sizeof(X)),num); }
|
2010-01-06 19:42:32 +00:00
|
|
|
|
2010-01-04 13:48:18 +00:00
|
|
|
template<class X>
|
2011-06-19 16:14:03 +00:00
|
|
|
Misc::SliceArray<X> getArray()
|
2010-01-04 13:48:18 +00:00
|
|
|
{
|
|
|
|
int len = getInt();
|
2010-01-06 19:42:32 +00:00
|
|
|
return getArrayLen<X>(len);
|
2010-01-04 13:48:18 +00:00
|
|
|
}
|
|
|
|
|
2011-06-19 16:14:03 +00:00
|
|
|
Misc::SString getString() { return getArray<char>(); }
|
2010-01-04 13:48:18 +00:00
|
|
|
|
2010-01-06 14:00:08 +00:00
|
|
|
const Vector *getVector() { return getPtr<Vector>(); }
|
|
|
|
const Matrix *getMatrix() { return getPtr<Matrix>(); }
|
|
|
|
const Transformation *getTrafo() { return getPtr<Transformation>(); }
|
2010-01-06 15:22:34 +00:00
|
|
|
const Vector4 *getVector4() { return getPtr<Vector4>(); }
|
2010-01-06 14:00:08 +00:00
|
|
|
|
2011-06-19 16:14:03 +00:00
|
|
|
Misc::FloatArray getFloatLen(int num)
|
2010-01-06 19:42:32 +00:00
|
|
|
{ return getArrayLen<float>(num); }
|
|
|
|
|
2010-01-06 14:00:08 +00:00
|
|
|
// For fixed-size strings where you already know the size
|
2010-01-04 13:48:18 +00:00
|
|
|
const char *getString(int size)
|
|
|
|
{ return (const char*)inp->getPtr(size); }
|
|
|
|
};
|
2010-01-06 11:28:37 +00:00
|
|
|
|
|
|
|
} // Namespace
|
2010-01-04 13:48:18 +00:00
|
|
|
#endif
|