server/src/common/kernel/xmlreader.c

1254 lines
40 KiB
C
Raw Normal View History

/* vi: set ts=2:
+-------------------+
| | Enno Rehling <enno@eressea.de>
| Eressea PBEM host | Christian Schlittchen <corwin@amber.kn-bremen.de>
| (c) 1998 - 2004 | Katja Zedel <katze@felidae.kn-bremen.de>
| | Henning Peters <faroul@beyond.kn-bremen.de>
+-------------------+
This program may not be used, modified or distributed
without prior permission by the authors of Eressea.
*/
#include <config.h>
#include "eressea.h"
#include "xmlreader.h"
#include <xml.h>
/* kernel includes */
#include "building.h"
#include "item.h"
#include "message.h"
#include "race.h"
#include "ship.h"
#include "skill.h"
#include "spell.h"
/* util includes */
#include <util/functions.h>
#include <util/message.h>
#include <util/nrmessage.h>
#include <util/crmessage.h>
/* libxml includes */
#include <libxml/tree.h>
#include <libxml/xpath.h>
#include <iconv.h>
/* libc includes */
#include <assert.h>
#include <ctype.h>
#include <string.h>
static boolean gamecode_enabled = false;
void
enable_xml_gamecode(void)
{
gamecode_enabled = true;
}
static void
xml_readtext(xmlNodePtr node, struct locale ** lang, xmlChar **text)
{
xmlChar * property = xmlGetProp(node, BAD_CAST "locale");
assert(property!=NULL);
*lang = find_locale((const char*)property);
if (*lang==NULL) *lang = make_locale((const char*)property);
xmlFree(property);
*text = xmlNodeListGetString(node->doc, node->children, 1);
}
static const char *
xml_to_locale(const xmlChar * xmlStr)
{
static iconv_t context = (iconv_t)-1;
static char zText[1024];
char * inbuf = (char*)xmlStr;
char * outbuf = zText;
size_t inbytes = strlen((const char*)xmlStr)+1;
size_t outbytes = sizeof(zText);
if (context==(iconv_t)-1) {
context = iconv_open("", "UTF-8");
}
assert(context!=(iconv_t)-1);
iconv(context, &inbuf, &inbytes, &outbuf, &outbytes);
if (inbytes!=0) {
log_error(("string is too long: %d chars remain in %s.\n", inbytes, (const char*)xmlStr));
}
return zText;
}
static xmlChar *
xml_cleanup_string(xmlChar * str)
{
xmlChar * read = str;
xmlChar * write = str;
while (*read) {
/* eat leading whitespace */
if (*read && isspace(*read)) {
while (*read && isspace(*read)) {
++read;
}
*write++ = ' ';
}
while (*read) {
if (*read== '\n') break;
if (*read== '\r') break;
*write++ = *read++;
}
}
*write = 0;
return str;
}
static void
xml_readrequirements(xmlNodePtr * nodeTab, int nodeNr, requirement ** reqArray)
{
int req;
requirement * radd = *reqArray;
assert (radd==NULL);
if (nodeNr==0) return;
radd = *reqArray = calloc(sizeof(requirement), nodeNr+1);
for (req=0;req!=nodeNr;++req) {
xmlNodePtr node = nodeTab[req];
xmlChar * property;
const resource_type * rtype;
resource_t type;
radd->number = xml_ivalue(node, "quantity", 0);
radd->recycle = xml_fvalue(node, "recycle", 0.0);
property = xmlGetProp(node, BAD_CAST "type");
rtype = rt_find((const char*)property);
2004-06-12 13:04:40 +02:00
assert(rtype!=NULL);
for (type=0;type!=MAX_RESOURCES;++type) {
if (oldresourcetype[type]==rtype) {
radd->type = type;
break;
}
}
xmlFree(property);
++radd;
}
}
static void
xml_readconstruction(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr, construction ** consPtr)
{
xmlNodePtr pushNode = xpath->node;
int k;
for (k=0;k!=nodeNr;++k) {
xmlNodePtr node = nodeTab[k];
xmlChar * property;
construction * con;
xmlXPathObjectPtr req;
int m;
assert(*consPtr==NULL);
con = *consPtr = calloc(sizeof(construction), 1);
property = xmlGetProp(node, BAD_CAST "skill");
assert(property!=NULL);
con->skill = sk_find((const char*)property);
2004-06-12 12:38:33 +02:00
assert(con->skill!=NOSKILL);
xmlFree(property);
con->maxsize = xml_ivalue(node, "maxsize", -1);
con->minskill = xml_ivalue(node, "minskill", -1);
con->reqsize = xml_ivalue(node, "reqsize", -1);
/* read construction/requirement */
xpath->node = node;
req = xmlXPathEvalExpression(BAD_CAST "requirement", xpath);
xml_readrequirements(req->nodesetval->nodeTab,
req->nodesetval->nodeNr, &con->materials);
xmlXPathFreeObject(req);
/* read construction/modifier */
xpath->node = node;
req = xmlXPathEvalExpression(BAD_CAST "modifier", xpath);
for (m=0;m!=req->nodesetval->nodeNr;++m) {
xmlNodePtr node = req->nodesetval->nodeTab[m];
property = xmlGetProp(node, BAD_CAST "function");
if (property!=NULL) {
pf_generic foo = get_function((const char*)property);
a_add(&con->attribs, make_skillmod(NOSKILL, SMF_PRODUCTION, (skillmod_fun)foo, 1.0, 0));
xmlFree(property);
}
}
xmlXPathFreeObject(req);
}
xpath->node = pushNode;
}
static int
parse_buildings(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr buildings;
xmlNodeSetPtr nodes;
int i;
/* reading eressea/buildings/building */
buildings = xmlXPathEvalExpression(BAD_CAST "/eressea/buildings/building", xpath);
nodes = buildings->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * property;
building_type * bt = calloc(sizeof(building_type), 1);
xmlXPathObjectPtr result;
int k;
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
bt->_name = strdup((const char *)property);
xmlFree(property);
bt->capacity = xml_ivalue(node, "capacity", -1);
bt->maxcapacity = xml_ivalue(node, "maxcapacity", -1);
bt->maxsize = xml_ivalue(node, "maxsize", -1);
bt->magres = xml_ivalue(node, "magres", 0);
bt->magresbonus = xml_ivalue(node, "magresbonus", 0);
bt->fumblebonus = xml_ivalue(node, "fumblebonus", 0);
bt->auraregen = xml_fvalue(node, "auraregen", 1.0);
if (xml_bvalue(node, "nodestroy", false)) bt->flags |= BTF_INDESTRUCTIBLE;
if (xml_bvalue(node, "nobuild", false)) bt->flags |= BTF_NOBUILD;
if (xml_bvalue(node, "unique", false)) bt->flags |= BTF_UNIQUE;
if (xml_bvalue(node, "decay", false)) bt->flags |= BTF_DECAY;
if (xml_bvalue(node, "magic", false)) bt->flags |= BTF_MAGIC;
if (xml_bvalue(node, "protection", false)) bt->flags |= BTF_PROTECTION;
/* reading eressea/buildings/building/construction */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
xml_readconstruction(xpath, result->nodesetval->nodeTab, result->nodesetval->nodeNr, &bt->construction);
xmlXPathFreeObject(result);
/* reading eressea/buildings/building/function */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
pf_generic fun;
property = xmlGetProp(node, BAD_CAST "value");
assert(property!=NULL);
fun = get_function((const char*)property);
if (fun==NULL) {
log_error(("unknown function name '%s' for building %s\n",
(const char*)property, bt->_name));
xmlFree(property);
continue;
}
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
if (strcmp((const char*)property, "name")==0) {
bt->name = (const char * (*)(int size))fun;
} else if (strcmp((const char*)property, "init")==0) {
bt->init = (void (*)(struct building_type*))fun;
} else {
log_error(("unknown function type '%s' for building %s\n", (const char*)property, bt->_name));
}
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/buildings/building/maintenance */
result = xmlXPathEvalExpression(BAD_CAST "maintenance", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
maintenance * mt;
if (bt->maintenance==NULL) {
bt->maintenance = calloc(sizeof(struct maintenance), result->nodesetval->nodeNr+1);
}
mt = bt->maintenance + k;
mt->number = xml_ivalue(node, "amount", 0);
property = xmlGetProp(node, BAD_CAST "type");
assert(property!=NULL);
mt->rtype = rt_find((const char*)property);
assert(mt->rtype!=NULL);
xmlFree(property);
if (xml_bvalue(node, "variable", false)) mt->flags |= MTF_VARIABLE;
if (xml_bvalue(node, "vital", false)) mt->flags |= MTF_VITAL;
}
xmlXPathFreeObject(result);
/* finally, register the new building type */
bt_register(bt);
}
xmlXPathFreeObject(buildings);
xmlXPathFreeContext(xpath);
return 0;
}
static int
parse_ships(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr ships;
xmlNodeSetPtr nodes;
int i;
/* reading eressea/ships/ship */
ships = xmlXPathEvalExpression(BAD_CAST "/eressea/ships/ship", xpath);
nodes = ships->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * property;
ship_type * st = calloc(sizeof(ship_type), 1);
xmlXPathObjectPtr result;
int k;
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
st->name[0] = strdup((const char *)property);
st->name[1] = strcat(strcpy(malloc(strlen(st->name[0])+3), st->name[0]),"_a");
xmlFree(property);
st->cabins = xml_ivalue(node, "cabins", 0);
st->cargo = xml_ivalue(node, "cargo", 0);
st->combat = xml_ivalue(node, "combat", 0);
st->cptskill = xml_ivalue(node, "cptskill", 0);
st->damage = xml_fvalue(node, "damage", 0.0);
if (xml_bvalue(node, "fly", false)) st->flags |= SFL_FLY;
if (xml_bvalue(node, "opensea", false)) st->flags |= SFL_OPENSEA;
st->minskill = xml_ivalue(node, "minskill", 0);
st->range = xml_ivalue(node, "range", 0);
st->storm = xml_fvalue(node, "storm", 1.0);
st->sumskill = xml_ivalue(node, "sumskill", 0);
/* reading eressea/ships/ship/construction */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
xml_readconstruction(xpath, result->nodesetval->nodeTab, result->nodesetval->nodeNr, &st->construction);
xmlXPathFreeObject(result);
/* reading eressea/ships/ship/coast */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "coast", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
terrain_t t;
if (k==0) {
assert(st->coast==NULL);
st->coast = malloc(sizeof(terrain_t) * (result->nodesetval->nodeNr+1));
st->coast[result->nodesetval->nodeNr] = NOTERRAIN;
}
property = xmlGetProp(node, BAD_CAST "terrain");
assert(property!=NULL);
st->coast[k] = NOTERRAIN;
for (t=0;t!=MAXTERRAINS;++t) {
if (strcmp((const char*)property, terrain[t].name)==0) {
st->coast[k] = t;
break;
}
}
assert(st->coast[k]!=NOTERRAIN);
xmlFree(property);
}
xmlXPathFreeObject(result);
/* finally, register the new building type */
st_register(st);
}
xmlXPathFreeObject(ships);
xmlXPathFreeContext(xpath);
return 0;
}
static void
race_compat(void)
{
/* required for old_race, do not change order! */
const char * oldracenames[MAXRACES] = {
"dwarf", "elf", "orc", "goblin", "human", "troll", "demon", "insect",
"halfling", "cat", "aquarian", "uruk", "snotling", "undead", "illusion",
"young dragon", "dragon", "wyrm", "ent", "catdragon", "dracoid",
"special", "spell", "irongolem", "stone golem", "shadowdemon",
"shadowmaster", "mountainguard", "alp", "toad", "braineater", "peasant",
"wolf", "lynx", "tunnelworm", "eagle", "rat", "songdragon", "nymph",
"unicorn", "direwolf", "ghost", "imp", "dreamcat", "fairy", "owl",
"hellcat", "tiger", "dolphin", "giantturtle", "kraken", "sea serpent",
"shadow knight", "centaur", "skeleton", "skeleton lord", "zombie",
"juju-zombie", "ghoul", "ghast", "museumghost", "gnome", "template",
"clone", "shadowdragon", "shadowbat", "nightmare", "vampunicorn",
"phoenix", "sphinx", "littlescarab", "greenscarab", "bluescarab",
"redscarab", "undeadpharaoh", "mummy"
};
int i;
for (i=0;i!=MAXRACES;++i) {
race * rc = rc_find(oldracenames[i]);
if (rc) {
new_race[i] = rc;
if (rc == new_race[RC_TROLL]) {
a_add(&rc->attribs, make_skillmod(NOSKILL, SMF_RIDING, NULL, 0.0, -1));
}
}
}
}
static weapon_type *
xml_readweapon(xmlXPathContextPtr xpath, item_type * itype)
{
xmlNodePtr node = xpath->node;
weapon_type * wtype = NULL;
unsigned int flags = WTF_NONE;
xmlXPathObjectPtr result;
xmlChar * property;
int k;
skill_t sk;
int minskill = xml_ivalue(node, "minskill", 0);
int offmod = xml_ivalue(node, "offmod", 0);
int defmod = xml_ivalue(node, "defmod", 0);
int reload = xml_ivalue(node, "reload", 0);
double magres = xml_fvalue(node, "magres", 0.0);
if (xml_bvalue(node, "armorpiercing", false)) flags |= WTF_ARMORPIERCING;
if (xml_bvalue(node, "magical", false)) flags |= WTF_MAGICAL;
if (xml_bvalue(node, "missile", false)) flags |= WTF_MISSILE;
if (xml_bvalue(node, "pierce", false)) flags |= WTF_PIERCE;
if (xml_bvalue(node, "cut", false)) flags |= WTF_CUT;
if (xml_bvalue(node, "blunt", false)) flags |= WTF_BLUNT;
property = xmlGetProp(node, BAD_CAST "skill");
assert(property!=NULL);
sk = sk_find((const char*)property);
assert(sk!=NOSKILL);
xmlFree(property);
wtype = new_weapontype(itype, flags, magres, NULL, offmod, defmod, reload, sk, minskill);
/* reading weapon/damage */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "damage", xpath);
assert(result->nodesetval->nodeNr<=2);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
int pos = 0;
property = xmlGetProp(node, BAD_CAST "type");
if (strcmp((const char *)property, "footman")!=0) {
pos = 1;
}
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "value");
wtype->damage[pos] = gc_add(strdup((const char*)property));
if (k==0) wtype->damage[1-pos] = wtype->damage[pos];
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading weapon/modifier */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "modifier", xpath);
assert(wtype->modifiers==NULL);
wtype->modifiers = calloc(sizeof(weapon_mod), result->nodesetval->nodeNr+1);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
xmlXPathObjectPtr races;
int r, flags = 0;
if (xml_bvalue(node, "walking", false)) flags|=WMF_WALKING;
if (xml_bvalue(node, "riding", false)) flags|=WMF_RIDING;
if (xml_bvalue(node, "against_walking", false)) flags|=WMF_AGAINST_WALKING;
if (xml_bvalue(node, "against_riding", false)) flags|=WMF_AGAINST_RIDING;
if (xml_bvalue(node, "offensive", false)) flags|=WMF_OFFENSIVE;
if (xml_bvalue(node, "defensive", false)) flags|=WMF_DEFENSIVE;
property = xmlGetProp(node, BAD_CAST "type");
if (strcmp((const char*)property, "damage")==0) flags|=WMF_DAMAGE;
else if (strcmp((const char*)property, "skill")==0) flags|=WMF_SKILL;
else if (strcmp((const char*)property, "missile_target")==0) flags|=WMF_MISSILE_TARGET;
xmlFree(property);
wtype->modifiers[k].flags = flags;
wtype->modifiers[k].value = xml_ivalue(node, "value", 0);
xpath->node = node;
races = xmlXPathEvalExpression(BAD_CAST "race", xpath);
for (r=0;r!=races->nodesetval->nodeNr;++r) {
xmlNodePtr node = races->nodesetval->nodeTab[r];
property = xmlGetProp(node, BAD_CAST "name");
if (property!=NULL) {
const race * rc = rc_find((const char*)property);
assert(rc!=NULL);
racelist_insert(&wtype->modifiers[k].races, rc);
xmlFree(property);
}
}
xmlXPathFreeObject(races);
}
xmlXPathFreeObject(result);
xpath->node = node;
return wtype;
}
static item_type *
xml_readitem(xmlXPathContextPtr xpath, resource_type * rtype)
{
xmlNodePtr node = xpath->node;
item_type * itype = NULL;
unsigned int flags = ITF_NONE;
xmlXPathObjectPtr result;
int k;
int weight = xml_ivalue(node, "weight", 0);
int capacity = xml_ivalue(node, "capacity", 0);
if (xml_bvalue(node, "cursed", false)) flags |= ITF_CURSED;
if (xml_bvalue(node, "notlost", false)) flags |= ITF_NOTLOST;
if (xml_bvalue(node, "big", false)) flags |= ITF_BIG;
if (xml_bvalue(node, "animal", false)) flags |= ITF_ANIMAL;
itype = new_itemtype(rtype, flags, weight, capacity);
/* reading item/construction */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
xml_readconstruction(xpath, result->nodesetval->nodeTab, result->nodesetval->nodeNr, &itype->construction);
xmlXPathFreeObject(result);
/* reading item/weapon */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "weapon", xpath);
assert(result->nodesetval->nodeNr<=1);
if (result->nodesetval->nodeNr!=0) {
itype->flags |= ITF_WEAPON;
xpath->node = result->nodesetval->nodeTab[0];
rtype->wtype = xml_readweapon(xpath, itype);
}
xmlXPathFreeObject(result);
if (gamecode_enabled) {
/* reading item/function */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
xmlChar * property;
pf_generic fun;
property = xmlGetProp(node, BAD_CAST "value");
assert(property!=NULL);
fun = get_function((const char*)property);
if (fun==NULL) {
log_error(("unknown function name '%s' for item '%s'\n",
(const char*)property, rtype->_name[0]));
xmlFree(property);
continue;
}
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
if (strcmp((const char*)property, "give")==0) {
itype->give = (boolean (*)(const struct unit*, const struct unit*, const struct item_type *, int, struct order *))fun;
}
else if (strcmp((const char*)property, "use")==0) {
itype->use = (int (*)(struct unit *, const struct item_type *, int, struct order *))fun;
} else {
log_error(("unknown function type '%s' for item '%s'\n",
(const char*)property, rtype->_name[0]));
}
xmlFree(property);
}
xmlXPathFreeObject(result);
}
return itype;
}
static int
parse_resources(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr resources;
xmlNodeSetPtr nodes;
int i;
2004-06-12 13:04:40 +02:00
init_resources();
/* reading eressea/resources/resource */
resources = xmlXPathEvalExpression(BAD_CAST "/eressea/resources/resource", xpath);
nodes = resources->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * property;
char *names[2], *appearance[2];
resource_type * rtype;
unsigned int flags = RTF_NONE;
xmlXPathObjectPtr result;
2004-04-12 02:57:09 +02:00
int k;
if (xml_bvalue(node, "pooled", true)) flags |= RTF_POOLED;
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
names[0] = strdup((const char*)property);
names[1] = strcat(strcpy((char*)malloc(strlen(names[0])+3), names[0]), "_p");
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "appearance");
if (property!=NULL) {
assert(property!=NULL);
appearance[0] = strdup((const char*)property);
appearance[1] = strcat(strcpy((char*)malloc(strlen(appearance[0])+3), appearance[0]), "_p");
rtype = new_resourcetype((const char**)names, (const char**)appearance, flags);
xmlFree(property);
free(appearance[0]);
free(appearance[1]);
} else {
rtype = new_resourcetype((const char**)names, NULL, flags);
}
free(names[0]);
free(names[1]);
2004-04-12 02:57:09 +02:00
/* reading eressea/resources/resource/function */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
if (result->nodesetval!=NULL) for (k=0;k!=result->nodesetval->nodeNr;++k) {
2004-04-12 02:57:09 +02:00
xmlNodePtr node = result->nodesetval->nodeTab[k];
pf_generic fun;
property = xmlGetProp(node, BAD_CAST "value");
assert(property!=NULL);
fun = get_function((const char*)property);
if (fun==NULL) {
log_error(("unknown function name '%s' for resource %s\n",
(const char*)property, rtype->_name[0]));
xmlFree(property);
continue;
}
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
if (strcmp((const char*)property, "change")==0) {
rtype->uchange = (rtype_uchange)fun;
} else if (strcmp((const char*)property, "get")==0) {
rtype->uget = (rtype_uget)fun;
} else if (strcmp((const char*)property, "name")==0) {
rtype->name = (rtype_name)fun;
} else {
log_error(("unknown function type '%s' for resource %s\n",
(const char*)property, rtype->_name[0]));
}
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/resources/resource/resourcelimit/function */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "resourcelimit/function", xpath);
if (result->nodesetval!=NULL) for (k=0;k!=result->nodesetval->nodeNr;++k) {
2004-04-12 02:57:09 +02:00
attrib * a = a_find(rtype->attribs, &at_resourcelimit);
xmlNodePtr node = result->nodesetval->nodeTab[k];
pf_generic fun;
property = xmlGetProp(node, BAD_CAST "value");
assert(property!=NULL);
fun = get_function((const char*)property);
if (fun==NULL) {
log_error(("unknown limit '%s' for resource %s\n",
(const char*)property, rtype->_name[0]));
xmlFree(property);
continue;
}
xmlFree(property);
if (a==NULL) a = a_add(&rtype->attribs, a_new(&at_resourcelimit));
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
if (strcmp((const char*)property, "use")==0) {
resource_limit * rdata = (resource_limit*)a->data.v;
rdata->use = (rlimit_use)fun;
} else if (strcmp((const char*)property, "limit")==0) {
resource_limit * rdata = (resource_limit*)a->data.v;
rdata->limit = (rlimit_limit)fun;
} else {
log_error(("unknown limit '%s' for resource %s\n",
(const char*)property, rtype->_name[0]));
}
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/resources/resource/item */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "item", xpath);
assert(result->nodesetval->nodeNr<=1);
if (result->nodesetval->nodeNr!=0) {
rtype->flags |= RTF_ITEM;
xpath->node = result->nodesetval->nodeTab[0];
rtype->itype = xml_readitem(xpath, rtype);
}
xmlXPathFreeObject(result);
}
xmlXPathFreeObject(resources);
xmlXPathFreeContext(xpath);
return 0;
}
static int
parse_equipment(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr items;
xmlNodeSetPtr nodes;
int i;
/* reading eressea/races/race */
items = xmlXPathEvalExpression(BAD_CAST "/eressea/equipment/item", xpath);
nodes = items->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * property;
const struct item_type * itype;
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
itype = it_find((const char*)property);
xmlFree(property);
if (itype!=NULL) {
int num = 0;
property = xmlGetProp(node, BAD_CAST "amount");
if (property!=NULL) {
num = atoi((const char*)property);
xmlFree(property);
}
add_equipment(itype, num);
}
}
xmlXPathFreeObject(items);
xmlXPathFreeContext(xpath);
return 0;
}
static int
parse_races(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr races;
xmlNodeSetPtr nodes;
int i;
/* reading eressea/races/race */
races = xmlXPathEvalExpression(BAD_CAST "/eressea/races/race", xpath);
nodes = races->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * property;
race * rc;
xmlXPathObjectPtr result;
int k;
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
rc = rc_find((const char*)property);
if (rc==NULL) rc = rc_add(rc_new((const char*)property));
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "damage");
assert(property!=NULL);
rc->def_damage = strdup((const char*)property);
xmlFree(property);
rc->magres = (float)xml_fvalue(node, "magres", 0.0);
rc->maxaura = (float)xml_fvalue(node, "maxaura", 0.0);
rc->regaura = (float)xml_fvalue(node, "regaura", 1.0);
rc->recruitcost = xml_ivalue(node, "recruitcost", 0);
rc->maintenance = xml_ivalue(node, "maintenance", 0);
rc->weight = xml_ivalue(node, "weight", 0);
#if RACE_CAPACITY
rc->capacity = xml_ivalue(node, "capacity", 0);
#endif
rc->speed = (float)xml_fvalue(node, "speed", 1.0F);
rc->hitpoints = xml_ivalue(node, "hp", 0);
rc->armor = (char)xml_ivalue(node, "ac", 0);
rc->at_default = (char)xml_ivalue(node, "unarmedattack", -2);
rc->df_default = (char)xml_ivalue(node, "unarmeddefense", -2);
rc->at_bonus = (char)xml_ivalue(node, "attackmodifier", 0);
rc->df_bonus = (char)xml_ivalue(node, "defensemodifier", 0);
if (xml_bvalue(node, "playerrace", false)) rc->flags |= RCF_PLAYERRACE;
if (xml_bvalue(node, "scarepeasants", false)) rc->flags |= RCF_SCAREPEASANTS;
if (xml_bvalue(node, "cannotmove", false)) rc->flags |= RCF_CANNOTMOVE;
if (xml_bvalue(node, "fly", false)) rc->flags |= RCF_FLY;
if (xml_bvalue(node, "coastal", false)) rc->flags |= RCF_COASTAL;
if (xml_bvalue(node, "swim", false)) rc->flags |= RCF_SWIM;
if (xml_bvalue(node, "walk", false)) rc->flags |= RCF_WALK;
if (xml_bvalue(node, "nolearn", false)) rc->flags |= RCF_NOLEARN;
if (xml_bvalue(node, "noteach", false)) rc->flags |= RCF_NOTEACH;
if (xml_bvalue(node, "horse", false)) rc->flags |= RCF_HORSE;
if (xml_bvalue(node, "desert", false)) rc->flags |= RCF_DESERT;
if (xml_bvalue(node, "absorbpeasants", false)) rc->flags |= RCF_ABSORBPEASANTS;
if (xml_bvalue(node, "noheal", false)) rc->flags |= RCF_NOHEAL;
if (xml_bvalue(node, "noweapons", false)) rc->flags |= RCF_NOWEAPONS;
if (xml_bvalue(node, "shapeshift", false)) rc->flags |= RCF_SHAPESHIFT;
if (xml_bvalue(node, "shapeshiftany", false)) rc->flags |= RCF_SHAPESHIFTANY;
if (xml_bvalue(node, "illusionary", false)) rc->flags |= RCF_ILLUSIONARY;
if (xml_bvalue(node, "undead", false)) rc->flags |= RCF_UNDEAD;
if (xml_bvalue(node, "dragon", false)) rc->flags |= RCF_DRAGON;
if (xml_bvalue(node, "nogive", false)) rc->ec_flags |= NOGIVE;
if (xml_bvalue(node, "giveitem", false)) rc->ec_flags |= GIVEITEM;
if (xml_bvalue(node, "giveperson", false)) rc->ec_flags |= GIVEPERSON;
if (xml_bvalue(node, "giveunit", false)) rc->ec_flags |= GIVEUNIT;
if (xml_bvalue(node, "getitem", false)) rc->ec_flags |= GETITEM;
if (xml_bvalue(node, "canguard", false)) rc->ec_flags |= CANGUARD;
if (xml_bvalue(node, "recruithorses", false)) rc->ec_flags |= ECF_REC_HORSES;
if (xml_bvalue(node, "recruitethereal", false)) rc->ec_flags |= ECF_REC_ETHEREAL;
if (xml_bvalue(node, "recruitunlimited", false)) rc->ec_flags |= ECF_REC_UNLIMITED;
if (xml_bvalue(node, "equipment", false)) rc->battle_flags |= BF_EQUIPMENT;
if (xml_bvalue(node, "noblock", false)) rc->battle_flags |= BF_NOBLOCK;
if (xml_bvalue(node, "invinciblenonmagic", false)) rc->battle_flags |= BF_INV_NONMAGIC;
if (xml_bvalue(node, "resistbash", false)) rc->battle_flags |= BF_RES_BASH;
if (xml_bvalue(node, "resistcut", false)) rc->battle_flags |= BF_RES_CUT;
if (xml_bvalue(node, "resistpierce", false)) rc->battle_flags |= BF_RES_PIERCE;
/* reading eressea/races/race/ai */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "ai", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
rc->splitsize = xml_ivalue(node, "splitsize", 0);
rc->aggression = (float)xml_fvalue(node, "aggression", 0.04);
if (xml_bvalue(node, "killpeasants", false)) rc->flags |= RCF_KILLPEASANTS;
if (xml_bvalue(node, "moverandom", false)) rc->flags |= RCF_MOVERANDOM;
if (xml_bvalue(node, "learn", false)) rc->flags |= RCF_LEARN;
}
xmlXPathFreeObject(result);
/* reading eressea/races/race/skill */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "skill", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
int mod = xml_ivalue(node, "modifier", 0);
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
if (mod!=0) {
skill_t sk = sk_find((const char*)property);
if (sk!=NOSKILL) {
rc->bonus[sk] = (char)mod;
} else {
log_error(("unknown skill '%s' in race '%s'\n",
(const char*)property, rc->_name[0]));
}
}
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/races/race/function */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
pf_generic fun;
property = xmlGetProp(node, BAD_CAST "value");
assert(property!=NULL);
fun = get_function((const char*)property);
if (fun==NULL) {
log_error(("unknown function name '%s' for race %s\n",
(const char*)property, rc->_name[0]));
xmlFree(property);
continue;
}
xmlFree(property);
property = xmlGetProp(node, BAD_CAST "name");
assert(property!=NULL);
if (strcmp((const char*)property, "name")==0) {
rc->generate_name = (const char* (*)(const struct unit*))fun;
} else if (strcmp((const char*)property, "age")==0) {
rc->age = (void(*)(struct unit*))fun;
} else if (strcmp((const char*)property, "move")==0) {
rc->move_allowed = (boolean(*)(const struct region *, const struct region *))fun;
} else if (strcmp((const char*)property, "itemdrop")==0) {
rc->itemdrop = (struct item *(*)(const struct race *, int))fun;
} else if (strcmp((const char*)property, "initfamiliar")==0) {
rc->init_familiar = (void(*)(struct unit *))fun;
} else {
log_error(("unknown function type '%s' for race %s\n",
(const char*)property, rc->_name[0]));
}
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/races/race/familiar */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "familiar", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
race * frc;
property = xmlGetProp(node, BAD_CAST "race");
assert(property!=NULL);
frc = rc_find((const char *)property);
if (frc == NULL) {
log_error(("%s not registered, is familiar for %s\n",
(const char*)property, rc->_name[0]));
assert(frc!=NULL);
frc = rc_add(rc_new((const char*)property));
}
if (xml_bvalue(node, "default", false)) {
rc->familiars[k] = rc->familiars[0];
rc->familiars[0] = frc;
} else {
rc->familiars[k] = frc;
}
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/races/race/precombatspell */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "precombatspell", xpath);
assert(rc->precombatspell==SPL_NOSPELL || !"precombatspell is already initialized");
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
rc->precombatspell = (spellid_t)xml_ivalue(node, "spell", SPL_NOSPELL);
}
xmlXPathFreeObject(result);
/* reading eressea/races/race/attack */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "attack", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
struct att * a = &rc->attack[k];
property = xmlGetProp(node, BAD_CAST "damage");
if (property!=NULL) {
a->data.dice = strdup((const char*)property);
xmlFree(property);
} else {
a->data.iparam = xml_ivalue(node, "spell", 0);
}
a->type = xml_ivalue(node, "type", 0);
a->flags = xml_ivalue(node, "flags", 0);
}
xmlXPathFreeObject(result);
}
xmlXPathFreeObject(races);
xmlXPathFreeContext(xpath);
race_compat();
return 0;
}
static int
parse_messages(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr messages;
xmlNodeSetPtr nodes;
int i;
/* reading eressea/strings/string */
messages = xmlXPathEvalExpression(BAD_CAST "/eressea/messages/message", xpath);
nodes = messages->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
const char * default_section = "events";
xmlChar * section;
xmlChar * property;
xmlXPathObjectPtr result;
int k;
char ** argv = NULL;
const message_type * mtype;
/* arguments */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "type/arg", xpath);
if (result->nodesetval->nodeNr>0) {
argv = malloc(sizeof(char*)*(result->nodesetval->nodeNr+1));
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
char zBuffer[128];
xmlChar * name, * type;
name = xmlGetProp(node, BAD_CAST "name");
type = xmlGetProp(node, BAD_CAST "type");
sprintf(zBuffer, "%s:%s", (const char*)name, (const char*)type);
xmlFree(name);
xmlFree(type);
argv[k] = strdup(zBuffer);
}
argv[result->nodesetval->nodeNr] = NULL;
}
xmlXPathFreeObject(result);
/* add the messagetype */
property = xmlGetProp(node, BAD_CAST "name");
mtype = mt_find((const char *)property);
if (mtype==NULL) {
mtype = mt_register(mt_new((const char *)property, (const char**)argv));
} else {
assert(argv!=NULL || !"cannot redefine arguments of message now");
}
xmlFree(property);
/* register the type for the CR */
crt_register(mtype);
/* let's clean up the mess */
if (argv!=NULL) {
for (k=0;argv[k]!=NULL;++k) free(argv[k]);
free(argv);
}
section = xmlGetProp(node, BAD_CAST "section");
if (section==NULL) section = BAD_CAST default_section;
mc_add((const char*)section);
/* strings */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "text", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr node = result->nodesetval->nodeTab[k];
struct locale * lang;
xmlChar * text;
xml_readtext(node, &lang, &text);
xml_cleanup_string(text);
nrt_register(mtype, lang, xml_to_locale(text), 0, (const char*)section);
xmlFree(text);
}
xmlXPathFreeObject(result);
if (section != BAD_CAST default_section) xmlFree(section);
}
xmlXPathFreeObject(messages);
xmlXPathFreeContext(xpath);
return 0;
}
static void
xml_readstrings(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr, boolean names)
{
int i;
for (i=0;i!=nodeNr;++i) {
xmlNodePtr stringNode = nodeTab[i];
xmlChar * name = xmlGetProp(stringNode, BAD_CAST "name");
xmlChar * nspc = NULL;
xmlXPathObjectPtr result;
int k;
char zName[128];
assert(name!=NULL);
if (names) nspc = xmlGetProp(stringNode->parent, BAD_CAST "name");
mkname_buf((const char*)nspc, (const char*)name, zName);
if (nspc!=NULL) xmlFree(nspc);
xmlFree(name);
/* strings */
xpath->node = stringNode;
result = xmlXPathEvalExpression(BAD_CAST "text", xpath);
for (k=0;k!=result->nodesetval->nodeNr;++k) {
xmlNodePtr textNode = result->nodesetval->nodeTab[k];
struct locale * lang;
xmlChar * text;
xml_readtext(textNode, &lang, &text);
if (text!=NULL) {
assert(strcmp(zName, (const char*)xml_cleanup_string(BAD_CAST zName))==0);
xml_cleanup_string(text);
locale_setstring(lang, zName, xml_to_locale(text));
xmlFree(text);
} else {
log_warning(("string %s has no text in locale %s\n",
zName, locale_name(lang)));
}
}
xmlXPathFreeObject(result);
}
}
static int
parse_strings(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr strings;
/* reading eressea/strings/string */
strings = xmlXPathEvalExpression(BAD_CAST "/eressea/strings/string", xpath);
xml_readstrings(xpath, strings->nodesetval->nodeTab, strings->nodesetval->nodeNr, false);
xmlXPathFreeObject(strings);
strings = xmlXPathEvalExpression(BAD_CAST "/eressea/strings/namespace/string", xpath);
xml_readstrings(xpath, strings->nodesetval->nodeTab, strings->nodesetval->nodeNr, true);
xmlXPathFreeObject(strings);
xmlXPathFreeContext(xpath);
return 0;
}
static void
xml_readprefixes(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr, boolean names)
{
int i;
for (i=0;i!=nodeNr;++i) {
xmlNodePtr node = nodeTab[i];
xmlChar * text = xmlNodeListGetString(node->doc, node->children, 1);
if (text!=NULL) {
add_raceprefix((const char*)text);
xmlFree(text);
}
}
}
static int
parse_prefixes(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr strings;
/* reading eressea/strings/string */
strings = xmlXPathEvalExpression(BAD_CAST "/eressea/prefixes/prefix", xpath);
xml_readprefixes(xpath, strings->nodesetval->nodeTab, strings->nodesetval->nodeNr, false);
xmlXPathFreeObject(strings);
xmlXPathFreeContext(xpath);
return 0;
}
static int
parse_main(xmlDocPtr doc)
{
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
xmlXPathObjectPtr result = xmlXPathEvalExpression(BAD_CAST "/eressea/game", xpath);
xmlNodeSetPtr nodes = result->nodesetval;
int i;
xmlChar * property;
xmlNodePtr node = nodes->nodeTab[0];
property = xmlGetProp(node, BAD_CAST "welcome");
if (property!=NULL) {
global.welcomepath = strdup((const char*)property);
xmlFree(property);
}
property = xmlGetProp(node, BAD_CAST "units");
if (property!=NULL) {
global.maxunits = atoi((const char*)property);
xmlFree(property);
}
property = xmlGetProp(node, BAD_CAST "name");
if (property!=NULL) {
global.gamename = strdup((const char*)property);
xmlFree(property);
}
xmlXPathFreeObject(result);
/* reading eressea/game/param */
xpath->node = node;
result = xmlXPathEvalExpression(BAD_CAST "param", xpath);
nodes = result->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * name = xmlGetProp(node, BAD_CAST "name");
xmlChar * value = xmlGetProp(node, BAD_CAST "value");
set_param(&global.parameters, (const char*)name, (const char*)value);
xmlFree(name);
xmlFree(value);
}
xmlXPathFreeObject(result);
/* reading eressea/game/order */
result = xmlXPathEvalExpression(BAD_CAST "order", xpath);
nodes = result->nodesetval;
for (i=0;i!=nodes->nodeNr;++i) {
xmlNodePtr node = nodes->nodeTab[i];
xmlChar * name = xmlGetProp(node, BAD_CAST "name");
boolean disable = xml_bvalue(node, "disable", false);
if (disable) {
int k;
for (k=0;k!=MAXKEYWORDS;++k) {
if (strcmp(keywords[k], (const char*)name)==0) {
global.disabled[k]=1;
break;
}
}
if (k==MAXKEYWORDS) {
log_error(("trying to disable unknown comand %s\n", (const char*)name));
}
}
xmlFree(name);
}
xmlXPathFreeObject(result);
xmlXPathFreeContext(xpath);
return 0;
}
void
register_xmlreader(void)
{
xml_register_callback(parse_main);
xml_register_callback(parse_strings);
xml_register_callback(parse_prefixes);
xml_register_callback(parse_messages);
xml_register_callback(parse_races);
xml_register_callback(parse_resources);
xml_register_callback(parse_buildings);
xml_register_callback(parse_ships);
xml_register_callback(parse_equipment);
}