mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-09 21:42:13 +00:00
2b407a9995
- Remove broken cache locking mechanism This was supposed to unload NIFFiles after a cell transition completes, but it was never working due to a mistake on the line if (--sLockLevel), should have been if (--sLockLevel == 0). Repairing this would increase load times (NIF files would have to be reloaded more frequently), so just removed it for now. - Decouple cache from NIFFile (now a new nifcache component) - Add API for future background loading - Provide a reliable way (SharedPtr) to hold on to loaded NIFFiles. This will be useful to avoid deep copies of keyframe and text key data, which is currently a performance bottleneck.
248 lines
7.1 KiB
C++
248 lines
7.1 KiB
C++
/*
|
|
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 OPENMW_COMPONENTS_NIF_NIFFILE_HPP
|
|
#define OPENMW_COMPONENTS_NIF_NIFFILE_HPP
|
|
|
|
#include <OgreResourceGroupManager.h>
|
|
#include <OgreDataStream.h>
|
|
#include <OgreVector2.h>
|
|
#include <OgreVector3.h>
|
|
#include <OgreVector4.h>
|
|
#include <OgreMatrix3.h>
|
|
#include <OgreQuaternion.h>
|
|
#include <OgreStringConverter.h>
|
|
|
|
#include <stdexcept>
|
|
#include <vector>
|
|
#include <cassert>
|
|
#include <typeinfo>
|
|
|
|
#include <boost/weak_ptr.hpp>
|
|
#include <boost/shared_ptr.hpp>
|
|
#include <boost/make_shared.hpp>
|
|
#include <boost/detail/endian.hpp>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "record.hpp"
|
|
#include "niftypes.hpp"
|
|
#include "nifstream.hpp"
|
|
|
|
namespace Nif
|
|
{
|
|
|
|
class NIFFile
|
|
{
|
|
enum NIFVersion {
|
|
VER_MW = 0x04000002 // Morrowind NIFs
|
|
};
|
|
|
|
/// Nif file version
|
|
int ver;
|
|
|
|
/// File name, used for error messages
|
|
std::string filename;
|
|
|
|
/// Record list
|
|
std::vector<Record*> records;
|
|
|
|
/// Root list
|
|
std::vector<Record*> roots;
|
|
|
|
/// Parse the file
|
|
void parse();
|
|
|
|
NIFFile (NIFFile const &);
|
|
void operator = (NIFFile const &);
|
|
|
|
public:
|
|
/// Used for error handling
|
|
void fail(const std::string &msg)
|
|
{
|
|
std::string err = "NIFFile Error: " + msg;
|
|
err += "\nFile: " + filename;
|
|
throw std::runtime_error(err);
|
|
}
|
|
|
|
void warn(const std::string &msg)
|
|
{
|
|
std::cerr << "NIFFile Warning: " << msg <<std::endl
|
|
<< "File: " << filename <<std::endl;
|
|
}
|
|
|
|
/// Open a NIF stream. The name is used for error messages.
|
|
NIFFile(const std::string &name);
|
|
~NIFFile();
|
|
|
|
/// Get a given record
|
|
Record *getRecord(size_t index) const
|
|
{
|
|
Record *res = records.at(index);
|
|
assert(res != NULL);
|
|
return res;
|
|
}
|
|
/// Number of records
|
|
size_t numRecords() const { return records.size(); }
|
|
|
|
/// Get a given root
|
|
Record *getRoot(size_t index=0) const
|
|
{
|
|
Record *res = roots.at(index);
|
|
assert(res != NULL);
|
|
return res;
|
|
}
|
|
/// Number of roots
|
|
size_t numRoots() const { return roots.size(); }
|
|
};
|
|
|
|
|
|
template<typename T>
|
|
struct KeyT {
|
|
float mTime;
|
|
T mValue;
|
|
T mForwardValue; // Only for Quadratic interpolation, and never for QuaternionKeyList
|
|
T mBackwardValue; // Only for Quadratic interpolation, and never for QuaternionKeyList
|
|
float mTension; // Only for TBC interpolation
|
|
float mBias; // Only for TBC interpolation
|
|
float mContinuity; // Only for TBC interpolation
|
|
};
|
|
typedef KeyT<float> FloatKey;
|
|
typedef KeyT<Ogre::Vector3> Vector3Key;
|
|
typedef KeyT<Ogre::Vector4> Vector4Key;
|
|
typedef KeyT<Ogre::Quaternion> QuaternionKey;
|
|
|
|
template<typename T, T (NIFStream::*getValue)()>
|
|
struct KeyListT {
|
|
typedef std::vector< KeyT<T> > VecType;
|
|
|
|
static const unsigned int sLinearInterpolation = 1;
|
|
static const unsigned int sQuadraticInterpolation = 2;
|
|
static const unsigned int sTBCInterpolation = 3;
|
|
static const unsigned int sXYZInterpolation = 4;
|
|
|
|
unsigned int mInterpolationType;
|
|
VecType mKeys;
|
|
|
|
KeyListT() : mInterpolationType(sLinearInterpolation) {}
|
|
|
|
//Read in a KeyGroup (see http://niftools.sourceforge.net/doc/nif/NiKeyframeData.html)
|
|
void read(NIFStream *nif, bool force=false)
|
|
{
|
|
assert(nif);
|
|
|
|
mInterpolationType = 0;
|
|
|
|
size_t count = nif->getUInt();
|
|
if(count == 0 && !force)
|
|
return;
|
|
|
|
//If we aren't forcing things, make sure that read clears any previous keys
|
|
if(!force)
|
|
mKeys.clear();
|
|
|
|
mInterpolationType = nif->getUInt();
|
|
|
|
KeyT<T> key;
|
|
NIFStream &nifReference = *nif;
|
|
|
|
if(mInterpolationType == sLinearInterpolation)
|
|
{
|
|
for(size_t i = 0;i < count;i++)
|
|
{
|
|
readTimeAndValue(nifReference, key);
|
|
mKeys.push_back(key);
|
|
}
|
|
}
|
|
else if(mInterpolationType == sQuadraticInterpolation)
|
|
{
|
|
for(size_t i = 0;i < count;i++)
|
|
{
|
|
readQuadratic(nifReference, key);
|
|
mKeys.push_back(key);
|
|
}
|
|
}
|
|
else if(mInterpolationType == sTBCInterpolation)
|
|
{
|
|
for(size_t i = 0;i < count;i++)
|
|
{
|
|
readTBC(nifReference, key);
|
|
mKeys.push_back(key);
|
|
}
|
|
}
|
|
//XYZ keys aren't actually read here.
|
|
//data.hpp sees that the last type read was sXYZInterpolation and:
|
|
// Eats a floating point number, then
|
|
// Re-runs the read function 3 more times, with force enabled so that the previous values aren't cleared.
|
|
// When it does that it's reading in a bunch of sLinearInterpolation keys, not sXYZInterpolation.
|
|
else if(mInterpolationType == sXYZInterpolation)
|
|
{
|
|
//Don't try to read XYZ keys into the wrong part
|
|
if ( count != 1 )
|
|
nif->file->fail("XYZ_ROTATION_KEY count should always be '1' . Retrieved Value: "+Ogre::StringConverter::toString(count));
|
|
}
|
|
else if (0 == mInterpolationType)
|
|
{
|
|
if (count != 0)
|
|
nif->file->fail("Interpolation type 0 doesn't work with keys");
|
|
}
|
|
else
|
|
nif->file->fail("Unhandled interpolation type: "+Ogre::StringConverter::toString(mInterpolationType));
|
|
}
|
|
|
|
private:
|
|
static void readTimeAndValue(NIFStream &nif, KeyT<T> &key)
|
|
{
|
|
key.mTime = nif.getFloat();
|
|
key.mValue = (nif.*getValue)();
|
|
}
|
|
|
|
static void readQuadratic(NIFStream &nif, KeyT<Ogre::Quaternion> &key)
|
|
{
|
|
readTimeAndValue(nif, key);
|
|
}
|
|
|
|
template <typename U>
|
|
static void readQuadratic(NIFStream &nif, KeyT<U> &key)
|
|
{
|
|
readTimeAndValue(nif, key);
|
|
key.mForwardValue = (nif.*getValue)();
|
|
key.mBackwardValue = (nif.*getValue)();
|
|
}
|
|
|
|
static void readTBC(NIFStream &nif, KeyT<T> &key)
|
|
{
|
|
readTimeAndValue(nif, key);
|
|
key.mTension = nif.getFloat();
|
|
key.mBias = nif.getFloat();
|
|
key.mContinuity = nif.getFloat();
|
|
}
|
|
};
|
|
typedef KeyListT<float,&NIFStream::getFloat> FloatKeyList;
|
|
typedef KeyListT<Ogre::Vector3,&NIFStream::getVector3> Vector3KeyList;
|
|
typedef KeyListT<Ogre::Vector4,&NIFStream::getVector4> Vector4KeyList;
|
|
typedef KeyListT<Ogre::Quaternion,&NIFStream::getQuaternion> QuaternionKeyList;
|
|
|
|
} // Namespace
|
|
#endif
|