mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-15 17:19:56 +00:00
368 lines
9.6 KiB
D
368 lines
9.6 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 (listkeeper.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 esm.listkeeper;
|
||
|
|
||
|
import monster.util.aa;
|
||
|
|
||
|
import core.memory;
|
||
|
|
||
|
import esm.filereader;
|
||
|
import esm.defs;
|
||
|
|
||
|
import std.stdio;
|
||
|
|
||
|
// Item types, used in the lookup table for inventory items, creature
|
||
|
// lists and leveled lists. We also use it for all types of references
|
||
|
// that can exist in cells.
|
||
|
enum ItemType
|
||
|
{
|
||
|
// Items
|
||
|
None = 0, Potion, Apparatus, Armor, Weapon, Book, Clothing,
|
||
|
Light, Ingredient, Pick, Probe, Repair, Misc, ItemList,
|
||
|
|
||
|
// Used for creature lists
|
||
|
Creature, CreatureList, NPC,
|
||
|
|
||
|
// Other cell references
|
||
|
Door, Activator, Static, Container//, SoundGen
|
||
|
}
|
||
|
|
||
|
abstract class ListKeeper
|
||
|
{
|
||
|
int listIndex;
|
||
|
|
||
|
new(uint size)
|
||
|
{
|
||
|
return esmRegion.allocate(size).ptr;
|
||
|
}
|
||
|
|
||
|
delete(void *p) { assert(0); }
|
||
|
|
||
|
this()
|
||
|
{
|
||
|
// Store our index for later use
|
||
|
listIndex = recordLists.length;
|
||
|
|
||
|
// Add the class to the global list
|
||
|
recordLists ~= this;
|
||
|
}
|
||
|
|
||
|
// Load a record from a master or plugin file
|
||
|
void load();
|
||
|
|
||
|
// Looks up a reference. If it does not exist it is assumed to be a
|
||
|
// forward reference within a file, and is inserted.
|
||
|
void* lookup(char[] s);
|
||
|
|
||
|
// Tell the loader that current file has ended, so it can do things
|
||
|
// like check that all referenced objects have been loaded.
|
||
|
void endFile();
|
||
|
|
||
|
// Number of inserted elements
|
||
|
uint length();
|
||
|
|
||
|
void addToList(ref ItemBaseList l, ItemType t) { assert(0); }
|
||
|
}
|
||
|
|
||
|
ListKeeper recordLists[];
|
||
|
|
||
|
// Keep the list of Type structures for records where the first
|
||
|
// subrecord is an id string called NAME. This id is used for
|
||
|
// lookup. Although almost all lookups match in case, there are a few
|
||
|
// sounds that don't, so we treat these id lookups as generally case
|
||
|
// insensitive. This hasn't posed any problems so far.
|
||
|
class ListID(Type) : ListKeeper
|
||
|
{
|
||
|
HashTable!(char[], Type, ESMRegionAlloc, CITextHash) names;
|
||
|
|
||
|
this(uint size)
|
||
|
{
|
||
|
names = names.init;
|
||
|
if(size) names.rehash(size);
|
||
|
}
|
||
|
|
||
|
// Reads the id for this header. Override if the id is not simply
|
||
|
// getHNString("NAME")
|
||
|
char[] getID()
|
||
|
{
|
||
|
return esFile.getHNString("NAME");
|
||
|
}
|
||
|
|
||
|
// Load a record from a master of plugin file
|
||
|
void load()
|
||
|
{
|
||
|
assert(esFile.getFileType == FileType.Esm ||
|
||
|
esFile.getFileType == FileType.Esp);
|
||
|
|
||
|
// Get the identifier of this record
|
||
|
char[] id = getID();
|
||
|
|
||
|
// Get pointer to a new or existing object.
|
||
|
Type *p;
|
||
|
if(names.insertEdit(id, p))
|
||
|
// A new item was inserted
|
||
|
{
|
||
|
p.state = LoadState.Unloaded;
|
||
|
p.id = id;
|
||
|
p.load();
|
||
|
p.state = LoadState.Loaded;
|
||
|
}
|
||
|
else
|
||
|
// Item already existed, either from a previous file or as a
|
||
|
// forward reference from this file. Load on top of it. The
|
||
|
// LoadState tells the struct whether it contains loaded data.
|
||
|
{
|
||
|
if(p.state == LoadState.Loaded)
|
||
|
// Make a special case for this, perhaps, or just ignore it.
|
||
|
writefln("WARNING: Duplicate record in file %s: '%s'",
|
||
|
esFile.getFilename(), id);
|
||
|
//esFile.fail("Duplicate record in file: '" ~ id ~ "'");
|
||
|
|
||
|
assert(icmp(p.id, id) == 0);
|
||
|
p.load();
|
||
|
p.state = LoadState.Loaded;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Looks up a reference. If it does not exist it is assumed to be a
|
||
|
// forward reference within a file, and is inserted.
|
||
|
void* lookup(char[] id)
|
||
|
{
|
||
|
if(!id.length) return null; // Empty reference
|
||
|
|
||
|
Type *p = names.lookup(id);
|
||
|
// Is the value in the list?
|
||
|
if(!p)
|
||
|
// No, assume it is a forward reference.
|
||
|
{
|
||
|
// Since the lookup name is stored in an internal buffer in
|
||
|
// esFile, we have to copy it.
|
||
|
id = esmRegion.copy(id);
|
||
|
|
||
|
// To avoid copying the string on every lookup, we have to
|
||
|
// insert in a separate step. But a double lookup isn't
|
||
|
// really THAT expensive. Besides, my tests show that this
|
||
|
// is used in less than 10% of the cases.
|
||
|
names.insertEdit(id, p);
|
||
|
p.id = id;
|
||
|
p.state = LoadState.Unloaded;
|
||
|
}
|
||
|
return cast(void*)p;
|
||
|
}
|
||
|
|
||
|
// Check that all referenced objects are actually loaded.
|
||
|
void endFile()
|
||
|
in
|
||
|
{
|
||
|
// We can skip this in release builds
|
||
|
names.validate();
|
||
|
}
|
||
|
body
|
||
|
{
|
||
|
foreach(char[] id, ref Type t; names)
|
||
|
// Current file is now counted as done
|
||
|
if(t.state == LoadState.Loaded) t.state = LoadState.Previous;
|
||
|
else if(t.state == LoadState.Unloaded)
|
||
|
//writefln("WARNING: Unloaded reference " ~ id);
|
||
|
esFile.fail("Unloaded reference " ~ id);
|
||
|
}
|
||
|
|
||
|
// Number of inserted elements
|
||
|
uint length() {return names.length;}
|
||
|
|
||
|
// Add the names in this list to an ItemList
|
||
|
void addToList(ref ItemBaseList l, ItemType t)
|
||
|
{
|
||
|
foreach(char[] id, ref Type s; names)
|
||
|
l.insert(id, &s, t);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
class Dummy : ListKeeper
|
||
|
{
|
||
|
this() {}
|
||
|
void load() {}
|
||
|
void* lookup(char[] s) { return null; }
|
||
|
void endFile() {}
|
||
|
uint length() { return 0; }
|
||
|
}
|
||
|
*/
|
||
|
|
||
|
// A pointer to an item
|
||
|
struct ItemBase
|
||
|
{
|
||
|
ItemType type;
|
||
|
void *p;
|
||
|
}
|
||
|
|
||
|
struct ItemBaseList
|
||
|
{
|
||
|
HashTable!(char[],ItemBase,ESMRegionAlloc) list;
|
||
|
|
||
|
void addList(ItemBaseList l)
|
||
|
{
|
||
|
foreach(char[] id, ItemBase b; l.list)
|
||
|
insert(id, b.p, b.type);
|
||
|
}
|
||
|
|
||
|
void addList(ListKeeper source, ItemType type)
|
||
|
{
|
||
|
source.addToList(*this, type);
|
||
|
}
|
||
|
|
||
|
void insert(char[] id, void* p, ItemType type)
|
||
|
{
|
||
|
ItemBase *b;
|
||
|
if(!list.insertEdit(id, b))
|
||
|
{
|
||
|
//writefln("Replacing item ", id);
|
||
|
if(b.type != ItemType.None)
|
||
|
esFile.fail("Replaced valid item: " ~ id);
|
||
|
}
|
||
|
//else writefln("Inserting new item ", id);
|
||
|
|
||
|
b.type = type;
|
||
|
b.p = p;
|
||
|
}
|
||
|
|
||
|
// Called at the end to check that all referenced items have been resolved
|
||
|
void endMerge()
|
||
|
{
|
||
|
foreach(char[] id, ref ItemBase t; list)
|
||
|
// Current file is now counted as done
|
||
|
if(t.type == ItemType.None)
|
||
|
// TODO: Don't use esFile.fail for this
|
||
|
esFile.fail("ItemBaseList: Unresolved forward reference: " ~ id);
|
||
|
}
|
||
|
|
||
|
// Look up an item, return a pointer to the ItemBase representing
|
||
|
// it. If it does not exist, it is inserted.
|
||
|
ItemBase *lookup(char[] id)
|
||
|
{
|
||
|
if(!id.length) return null; // Empty reference
|
||
|
ItemBase *b = list.lookup(id);
|
||
|
// Is the value in the list?
|
||
|
if(!b)
|
||
|
// No, assume it is a forward reference.
|
||
|
{
|
||
|
// Since the lookup name is stored in an internal buffer in
|
||
|
// esFile, we have to copy it.
|
||
|
id = esmRegion.copy(id);
|
||
|
|
||
|
// To avoid copying the string on every lookup, we have to
|
||
|
// insert in a separate step. But a double lookup isn't
|
||
|
// really THAT expensive.
|
||
|
list.insertEdit(id, b);
|
||
|
|
||
|
b.p = null;
|
||
|
b.type = ItemType.None;
|
||
|
}
|
||
|
return b;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Um... I think this code is double up because I had to split it into
|
||
|
// separate files to avoid template stuff. I don't think we need to do
|
||
|
// that anymore.
|
||
|
|
||
|
// An item. Contains a reference to an ItemBase, which again is a
|
||
|
// reference to an item. The ItemBase might change after we have
|
||
|
// looked it up (for forward references), so we have to use a pointer.
|
||
|
struct Item
|
||
|
{
|
||
|
ItemBase *i;
|
||
|
|
||
|
void* getPtr(ItemType type)
|
||
|
{
|
||
|
if(i != null && i.type == type) return i.p;
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
T* getType(T, ItemType Type)()
|
||
|
{
|
||
|
return cast(T*)getPtr(Type);
|
||
|
}
|
||
|
|
||
|
/* Avoid the templates for now
|
||
|
alias getType!(Potion, ItemType.Potion) getPotion;
|
||
|
alias getType!(Apparatus, ItemType.Apparatus) getApparatus;
|
||
|
alias getType!(Armor, ItemType.Armor) getArmor;
|
||
|
alias getType!(Weapon, ItemType.Weapon) getWeapon;
|
||
|
alias getType!(Book, ItemType.Book) getBook;
|
||
|
alias getType!(Clothing, ItemType.Clothing) getClothing;
|
||
|
alias getType!(Light, ItemType.Light) getLight;
|
||
|
alias getType!(Ingredient, ItemType.Ingredient) getIngredient;
|
||
|
alias getType!(Tool, ItemType.Pick) getPick;
|
||
|
alias getType!(Tool, ItemType.Probe) getProbe;
|
||
|
alias getType!(Tool, ItemType.Repair) getRepair;
|
||
|
alias getType!(Misc, ItemType.Misc) getMisc;
|
||
|
alias getType!(LeveledItems, ItemType.ItemList) getItemList;
|
||
|
alias getType!(Creature, ItemType.Creature) getCreature;
|
||
|
alias getType!(LeveledCreatures, ItemType.CreatureList) getCreatureList;
|
||
|
alias getType!(NPC, ItemType.NPC) getNPC;
|
||
|
alias getType!(Door, ItemType.Door) getDoor;
|
||
|
alias getType!(Activator, ItemType.Activator) getActivator;
|
||
|
alias getType!(Static, ItemType.Static) getStatic;
|
||
|
alias getType!(Container, ItemType.Container) getContainer;
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
struct ItemList
|
||
|
{
|
||
|
private:
|
||
|
ItemBaseList list;
|
||
|
|
||
|
public:
|
||
|
void addList(ItemList l)
|
||
|
{ list.addList(l.list); }
|
||
|
|
||
|
void addList(ListKeeper source, ItemType type)
|
||
|
{ list.addList(source, type); }
|
||
|
|
||
|
Item lookup(char[] id)
|
||
|
{
|
||
|
Item i;
|
||
|
i.i = list.lookup(id);
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
void endMerge()
|
||
|
{ list.endMerge(); }
|
||
|
|
||
|
void endFile()
|
||
|
in { list.list.validate(); }
|
||
|
body {}
|
||
|
|
||
|
void rehash(uint size)
|
||
|
{ list.list.rehash(size); }
|
||
|
|
||
|
uint length() { return list.list.length(); }
|
||
|
}
|
||
|
|
||
|
// Aggregate lists, made by concatinating several other lists.
|
||
|
ItemList items; // All inventory items, including leveled item lists
|
||
|
ItemList actors; // All actors, ie. NPCs, creatures and leveled lists
|
||
|
ItemList cellRefs; // All things that are referenced from cells
|