You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
486 lines
11 KiB
D
486 lines
11 KiB
D
/*
|
|
OpenMW - The completely unofficial reimplementation of Morrowind
|
|
Copyright (C) 2008 Nicolay Korslund
|
|
Email: < korslund@gmail.com >
|
|
WWW: http://openmw.snaptoad.com/
|
|
|
|
This file (niffile.d) 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/ .
|
|
|
|
*/
|
|
|
|
module nif.niffile;
|
|
|
|
public import std.stream;
|
|
public import util.regions;
|
|
|
|
import std.string;
|
|
import nif.misc;
|
|
|
|
public import core.memory;
|
|
|
|
// Exception class thrown by most exceptions originating within the
|
|
// nif package
|
|
class NIFFileException : Exception
|
|
{
|
|
this(char[] msg) {super("NIFFile Exception: " ~ msg);}
|
|
}
|
|
|
|
/* The NIFFile struct is used for the task of reading from NIF
|
|
files. It provides specialized methods for handling common types,
|
|
records, etc., and also provides mechanisms for output and error
|
|
handling. It does not store any (non-transient) NIF data.
|
|
*/
|
|
NIFFile nifFile;
|
|
|
|
struct NIFFile
|
|
{
|
|
private:
|
|
Stream f; // Input stream
|
|
BufferedFile bf; // Used and reused for direct file input.
|
|
TArrayStream!(ubyte[]) ss; // For stream reading
|
|
|
|
// These are used for warnings and error messages only
|
|
char[] filename;
|
|
int recNum; // Record number
|
|
char[] recName; // Record name
|
|
Object recObj; // Record object
|
|
public:
|
|
|
|
/* ------------------------------------------------
|
|
* Object initialization and file management
|
|
* ------------------------------------------------
|
|
*/
|
|
|
|
// Open a file from the file system
|
|
void open(char[] file)
|
|
{
|
|
close();
|
|
|
|
// Create a BufferedFile, and reuse it later.
|
|
if(bf is null)
|
|
bf = new BufferedFile();
|
|
|
|
bf.open(file);
|
|
|
|
f = bf;
|
|
filename = file;
|
|
}
|
|
|
|
// Open a memory slice, the name is used for error messages.
|
|
void open(void[] data, char[] name)
|
|
{
|
|
close();
|
|
|
|
// Create a TArrayStream, and reuse it
|
|
if(ss is null)
|
|
{
|
|
ss = new TArrayStream!(ubyte[])(cast(ubyte[])data);
|
|
ss.writeable = false; // Read-only
|
|
}
|
|
else
|
|
{
|
|
// TArrayStream lacks the open() method. Instead, all the
|
|
// members are public.
|
|
ss.buf = cast(ubyte[])data;
|
|
ss.len = ss.buf.length;
|
|
ss.cur = 0;
|
|
}
|
|
|
|
f = ss;
|
|
filename = name;
|
|
}
|
|
|
|
// Close the file handle
|
|
void close()
|
|
{
|
|
// Close the buffered file
|
|
if(f !is null && f is bf) f.close();
|
|
f = null;
|
|
|
|
// Zero out variables
|
|
filename = null;
|
|
recNum = 0;
|
|
recName = null;
|
|
recObj = null;
|
|
}
|
|
|
|
long size()
|
|
in
|
|
{
|
|
if(f is null) fail("Cannot get size, file is not open");
|
|
}
|
|
body
|
|
{
|
|
return f.size();
|
|
}
|
|
|
|
bool eof() { return f.eof(); }
|
|
|
|
void seekCur(ulong skip)
|
|
in
|
|
{
|
|
if(f is null) fail("Cannot seek, file is not open");
|
|
}
|
|
body
|
|
{
|
|
f.seekCur(skip);
|
|
}
|
|
|
|
ulong position()
|
|
{
|
|
return f.position();
|
|
}
|
|
|
|
/* ------------------------------------------------
|
|
* Error reporting
|
|
* ------------------------------------------------
|
|
*/
|
|
|
|
// Used to format error messages
|
|
private char[] makeError(char[] str)
|
|
{
|
|
if(recNum > 0)
|
|
{
|
|
if(recName == "" && recObj !is null) recName = recObj.toString;
|
|
str = format("%s\n Record %d: %s", str, recNum-1, recName);
|
|
}
|
|
if(filename != "") str = format("%s\n File: %s", str, filename);
|
|
if(f !is null)
|
|
if(f.eof()) str = format("%s\n At end of file", str);
|
|
else str = format("%s\n Offset 0x%x", str, f.position);
|
|
return str ~ "\n";
|
|
}
|
|
|
|
void fail(char[] msg)
|
|
{
|
|
throw new NIFFileException(makeError(msg));
|
|
}
|
|
|
|
debug(warnstd) import std.stdio;
|
|
|
|
void warn(char[] msg)
|
|
{
|
|
debug(strict) fail("(fail on warning) " ~ msg);
|
|
else
|
|
{
|
|
debug(warnstd) writefln("WARNING: ", makeError(msg));
|
|
debug(warnlog) log.writefln("WARNING: ", makeError(msg));
|
|
}
|
|
}
|
|
|
|
void assertWarn(bool condition, char[] str)
|
|
{ if(!condition) warn(str); }
|
|
|
|
/*
|
|
void assertFail(bool condition, char[] str)
|
|
{ if(!condition) fail(str); }
|
|
*/
|
|
|
|
// Used for error message handling, hackish.
|
|
void setRec(int i, char[] n)
|
|
{
|
|
recNum = i+1;
|
|
recName = n;
|
|
recObj = null;
|
|
}
|
|
|
|
// This variant takes an object instead of a name. That way we only
|
|
// need to call toString when the name is needed (which it mostly
|
|
// isn't.)
|
|
void setRec(int i, Object o)
|
|
{
|
|
recNum = i+1;
|
|
recObj = o;
|
|
recName = null;
|
|
}
|
|
|
|
/* ------------------------------------------------
|
|
* Reader method templates (skip to next section
|
|
* for usable methods)
|
|
* ------------------------------------------------
|
|
*/
|
|
|
|
// Read a variable of a given type T
|
|
T getType(T)()
|
|
{
|
|
T t;
|
|
f.read(t);
|
|
return t;
|
|
}
|
|
|
|
// Read a variable and compare it to what we want it to be
|
|
template getTypeIs(T)
|
|
{
|
|
T getTypeIs(T[] pt ...)
|
|
{
|
|
T t = getType!(T)();
|
|
debug(check)
|
|
{
|
|
bool match;
|
|
foreach(T a; pt)
|
|
if(a == t) {match = true; break;}
|
|
|
|
if(!match)
|
|
{
|
|
char[] errMsg = format(typeid(T),
|
|
" mismatch: got %s, expected ", t);
|
|
if(pt.length > 1) errMsg ~= "one of: ";
|
|
foreach(T a; pt)
|
|
errMsg ~= format("%s ", a);
|
|
warn(errMsg);
|
|
}
|
|
}
|
|
return t;
|
|
}
|
|
}
|
|
|
|
debug(verbose)
|
|
{
|
|
// Read a variable of a given type T and print it to screen
|
|
template wgetType(T)
|
|
{
|
|
T wgetType()
|
|
{
|
|
T t;
|
|
f.read(t);
|
|
writefln(typeid(T), ": ", t);
|
|
return t;
|
|
}
|
|
}
|
|
|
|
// Read a variable and compare it to what we want it to be
|
|
template wgetTypeIs(T)
|
|
{
|
|
T wgetTypeIs(T pt[] ...)
|
|
{
|
|
T t = getType!(T)();
|
|
|
|
char[] wanted;
|
|
if(pt.length > 1) wanted = "one of: ";
|
|
foreach(T a; pt) wanted ~= format("%s ", a);
|
|
|
|
writefln(typeid(T), ": ", t, " (wanted %s)", wanted);
|
|
|
|
debug(check)
|
|
{
|
|
bool match;
|
|
foreach(T a; pt)
|
|
if(a == t) {match = true; break;}
|
|
|
|
if(!match)
|
|
{
|
|
warn(format(typeid(T),
|
|
" mismatch: got %s, expected %s", t, wanted));
|
|
}
|
|
}
|
|
return t;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fill the provided array of Ts
|
|
template getArrayLen(T)
|
|
{
|
|
T[] getArrayLen(T[] arr)
|
|
{
|
|
f.readExact(arr.ptr, arr.length*T.sizeof);
|
|
return arr;
|
|
}
|
|
}
|
|
|
|
// Set the size of the provided array to 'count', and fill it.
|
|
template getArraySize(T)
|
|
{
|
|
T[] getArraySize(int count)
|
|
{
|
|
T[] arr;
|
|
fitArray(count, T.sizeof);
|
|
arr = cast(T[])nifRegion.allocate(count * T.sizeof);
|
|
getArrayLen!(T)(arr);
|
|
return arr;
|
|
}
|
|
}
|
|
|
|
// Get an array of Ts preceded by an array length of type Index
|
|
// (assumed to be an integer type)
|
|
template getArray(T,Index)
|
|
{
|
|
T[] getArray()
|
|
{
|
|
// Read array length
|
|
Index s = getType!(Index)();
|
|
|
|
// Is array larger than file?
|
|
fitArray(s,T.sizeof);
|
|
|
|
// Allocate the buffer
|
|
T[] result = cast(T[])nifRegion.allocate(s * T.sizeof);
|
|
|
|
// Read the buffer
|
|
return getArrayLen!(T)(result);
|
|
}
|
|
}
|
|
|
|
/* ------------------------------------------------
|
|
* Reader methods
|
|
* ------------------------------------------------
|
|
*/
|
|
|
|
// Strings
|
|
alias getArrayLen!(char) getString;
|
|
alias getArray!(char,int) getString;
|
|
|
|
// Other arrays
|
|
alias getArray!(int,int) getInts;
|
|
|
|
// Checks if an array of size elements each of size esize could
|
|
// possibly follow in the file.
|
|
void fitArray(int size, int esize)
|
|
in
|
|
{
|
|
assert(esize > 0);
|
|
}
|
|
body
|
|
{
|
|
if((size*esize+8) > (f.size() - f.position()) || size < 0)
|
|
fail(format(
|
|
"Array out of bounds: %d*%d + 8 bytes needed, but only %d bytes left in file",
|
|
size,esize,(f.size-f.position)));
|
|
}
|
|
|
|
// Base types
|
|
alias getType!(byte) getByte;
|
|
alias getType!(ubyte) getUbyte;
|
|
alias getType!(short) getShort;
|
|
alias getType!(ushort) getUshort;
|
|
alias getType!(int) getInt;
|
|
alias getType!(uint) getUint;
|
|
alias getType!(float) getFloat;
|
|
|
|
// Base types with error checking
|
|
alias getTypeIs!(short) getShortIs;
|
|
alias getTypeIs!(ushort) getUshortIs;
|
|
alias getTypeIs!(byte) getByteIs;
|
|
alias getTypeIs!(ubyte) getUbyteIs;
|
|
alias getTypeIs!(int) getIntIs;
|
|
alias getTypeIs!(uint) getUintIs;
|
|
alias getTypeIs!(float) getFloatIs;
|
|
|
|
debug(verbose)
|
|
{
|
|
// Base types
|
|
alias wgetType!(byte) wgetByte;
|
|
alias wgetType!(ubyte) wgetUbyte;
|
|
alias wgetType!(short) wgetShort;
|
|
alias wgetType!(ushort) wgetUshort;
|
|
alias wgetType!(int) wgetInt;
|
|
alias wgetType!(uint) wgetUint;
|
|
alias wgetType!(float) wgetFloat;
|
|
|
|
// Base types with error checking
|
|
alias wgetTypeIs!(short) wgetShortIs;
|
|
alias wgetTypeIs!(ushort) wgetUshortIs;
|
|
alias wgetTypeIs!(byte) wgetByteIs;
|
|
alias wgetTypeIs!(ubyte) wgetUbyteIs;
|
|
alias wgetTypeIs!(int) wgetIntIs;
|
|
alias wgetTypeIs!(uint) wgetUintIs;
|
|
alias wgetTypeIs!(float) wgetFloatIs;
|
|
}
|
|
else
|
|
{
|
|
// Base types
|
|
alias getByte wgetByte;
|
|
alias getUbyte wgetUbyte;
|
|
alias getShort wgetShort;
|
|
alias getUshort wgetUshort;
|
|
alias getInt wgetInt;
|
|
alias getUint wgetUint;
|
|
alias getFloat wgetFloat;
|
|
|
|
// Base types with error checking
|
|
alias getByteIs wgetByteIs;
|
|
alias getUbyteIs wgetUbyteIs;
|
|
alias getShortIs wgetShortIs;
|
|
alias getUshortIs wgetUshortIs;
|
|
alias getIntIs wgetIntIs;
|
|
alias getUintIs wgetUintIs;
|
|
alias getFloatIs wgetFloatIs;
|
|
}
|
|
|
|
// Vectors
|
|
Vector getVector()
|
|
{
|
|
Vector v;
|
|
getArrayLen!(float)(v.array);
|
|
return v;
|
|
}
|
|
|
|
Vector4 getVector4()
|
|
{
|
|
Vector4 v;
|
|
getArrayLen!(float)(v.array);
|
|
return v;
|
|
}
|
|
|
|
Vector getVectorIs(float x, float y, float z)
|
|
{
|
|
Vector v = getVector();
|
|
debug(check)
|
|
{
|
|
Vector u;
|
|
u.set(x,y,z);
|
|
if(v != u)
|
|
warn("Vector mismatch: expected " ~ u.toString ~ ", got " ~ v.toString);
|
|
}
|
|
return v;
|
|
}
|
|
|
|
debug(verbose)
|
|
{
|
|
Vector wgetVector()
|
|
{
|
|
Vector v = getVector();
|
|
writefln("vector: ", v.toString);
|
|
return v;
|
|
}
|
|
|
|
Vector4 wgetVector4()
|
|
{
|
|
Vector4 v = getVector4();
|
|
writefln("4-vector: ", v.toString);
|
|
return v;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alias getVector wgetVector;
|
|
alias getVector4 wgetVector4;
|
|
}
|
|
|
|
void getMatrix(ref Matrix m)
|
|
{
|
|
getArrayLen!(float)(m.array);
|
|
}
|
|
|
|
void getTransformation(ref Transformation t)
|
|
{
|
|
t.pos = getVector();
|
|
getMatrix(t.rotation);
|
|
t.scale = getFloat/*Is(1)*/;
|
|
t.velocity = getVectorIs(0,0,0);
|
|
}
|
|
}
|