2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2007-07-17 17:33:48 +02:00
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
2007-09-02 20:11:17 +02:00
|
|
|
|
* Enno Rehling (enno@eressea.de)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
|
*
|
|
|
|
|
* based on:
|
|
|
|
|
*
|
|
|
|
|
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
|
|
|
|
|
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
|
|
|
|
|
*
|
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
|
* This program may not be sold or used commercially without prior written
|
|
|
|
|
* permission from the authors.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
2008-04-20 16:48:15 +02:00
|
|
|
|
#include <kernel/eressea.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "save.h"
|
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "alchemy.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "alliance.h"
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "border.h"
|
|
|
|
|
#include "building.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "faction.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "group.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "item.h"
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "karma.h"
|
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "message.h"
|
2007-08-05 14:26:46 +02:00
|
|
|
|
#include "move.h"
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "objtypes.h"
|
|
|
|
|
#include "order.h"
|
|
|
|
|
#include "pathfinder.h"
|
|
|
|
|
#include "plane.h"
|
|
|
|
|
#include "race.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "region.h"
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#include "resources.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "ship.h"
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "spell.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "terrain.h"
|
|
|
|
|
#include "terrainid.h" /* only for conversion code */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "unit.h"
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include "version.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
#include "textstore.h"
|
|
|
|
|
#include "binarystore.h"
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* attributes includes */
|
|
|
|
|
#include <attributes/key.h>
|
|
|
|
|
|
|
|
|
|
/* util includes */
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
#include <util/base36.h>
|
2007-02-25 19:49:28 +01:00
|
|
|
|
#include <util/bsdstring.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/event.h>
|
2007-06-26 11:32:28 +02:00
|
|
|
|
#include <util/filereader.h>
|
2004-08-22 11:33:15 +02:00
|
|
|
|
#include <util/goodies.h>
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include <util/lists.h>
|
2007-08-05 14:19:56 +02:00
|
|
|
|
#include <util/log.h>
|
2007-06-22 00:31:28 +02:00
|
|
|
|
#include <util/parser.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/rand.h>
|
2008-04-26 16:55:22 +02:00
|
|
|
|
#include <util/resolve.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/rng.h>
|
2008-04-26 16:55:22 +02:00
|
|
|
|
#include <util/sql.h>
|
|
|
|
|
#include <util/storage.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/umlaut.h>
|
2007-09-16 17:34:49 +02:00
|
|
|
|
#include <util/unicode.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
#include <libxml/encoding.h>
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <ctype.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
2001-02-13 01:41:17 +01:00
|
|
|
|
#define xisdigit(c) (((c) >= '0' && (c) <= '9') || (c) == '-')
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
#define ESCAPE_FIX
|
2005-07-27 13:48:16 +02:00
|
|
|
|
#define MAXORDERS 256
|
2005-07-28 15:20:36 +02:00
|
|
|
|
#define MAXPERSISTENT 128
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-04-12 01:56:47 +02:00
|
|
|
|
/* exported symbols symbols */
|
|
|
|
|
const char * xmlfile = "eressea.xml";
|
2001-01-25 10:37:55 +01:00
|
|
|
|
const char * g_datadir;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
int firstx = 0, firsty = 0;
|
2007-08-12 11:51:26 +02:00
|
|
|
|
int enc_gamedata = 0;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* local symbols */
|
2001-12-10 01:13:39 +01:00
|
|
|
|
static region * current_region;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
rns(FILE * f, char *c, size_t size)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
char * s = c;
|
|
|
|
|
do {
|
|
|
|
|
*s = (char) getc(f);
|
|
|
|
|
} while (*s!='"');
|
|
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
|
*s = (char) getc(f);
|
|
|
|
|
if (*s=='"') break;
|
|
|
|
|
if (s<c+size) ++s;
|
|
|
|
|
}
|
|
|
|
|
*s = 0;
|
|
|
|
|
return c;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
extern unsigned int __at_hashkey(const char* s);
|
|
|
|
|
|
|
|
|
|
FILE *
|
|
|
|
|
cfopen(const char *filename, const char *mode)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
FILE * F = fopen(filename, mode);
|
|
|
|
|
|
|
|
|
|
if (F == 0) {
|
|
|
|
|
perror(filename);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
setvbuf(F, 0, _IOFBF, 32 * 1024); /* 32 kb buffer size */
|
|
|
|
|
return F;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2007-08-12 11:51:26 +02:00
|
|
|
|
int
|
|
|
|
|
freadstr(FILE * F, int encoding, char * start, size_t size)
|
|
|
|
|
{
|
|
|
|
|
char * str = start;
|
|
|
|
|
boolean quote = false;
|
|
|
|
|
for (;;) {
|
|
|
|
|
int c = fgetc(F);
|
|
|
|
|
|
|
|
|
|
if (isspace(c)) {
|
|
|
|
|
if (str==start) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (!quote) {
|
|
|
|
|
*str = 0;
|
|
|
|
|
return (int)(str-start);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
switch (c) {
|
|
|
|
|
case EOF:
|
|
|
|
|
return EOF;
|
|
|
|
|
case '"':
|
|
|
|
|
if (!quote && str!=start) {
|
|
|
|
|
log_error(("datafile contains a \" that isn't at the start of a string.\n"));
|
|
|
|
|
assert(!"datafile contains a \" that isn't at the start of a string.\n");
|
|
|
|
|
}
|
|
|
|
|
if (quote) {
|
|
|
|
|
*str = 0;
|
|
|
|
|
return (int)(str-start);
|
|
|
|
|
}
|
|
|
|
|
quote = true;
|
|
|
|
|
break;
|
|
|
|
|
case '\\':
|
|
|
|
|
c = fgetc(F);
|
|
|
|
|
switch (c) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
case EOF:
|
|
|
|
|
return EOF;
|
|
|
|
|
case 'n':
|
|
|
|
|
if ((size_t)(str-start+1)<size) {
|
|
|
|
|
*str++ = '\n';
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
if ((size_t)(str-start+1)<size) {
|
|
|
|
|
if (encoding == XML_CHAR_ENCODING_8859_1 && c&0x80) {
|
|
|
|
|
char inbuf = (char)c;
|
|
|
|
|
size_t inbytes = 1;
|
|
|
|
|
size_t outbytes = size-(str-start);
|
|
|
|
|
int ret = unicode_latin1_to_utf8(str, &outbytes, &inbuf, &inbytes);
|
|
|
|
|
if (ret>0) str+=ret;
|
|
|
|
|
else {
|
|
|
|
|
log_error(("input data was not iso-8859-1! assuming utf-8\n"));
|
|
|
|
|
encoding = XML_CHAR_ENCODING_ERROR;
|
|
|
|
|
*str++ = (char)c;
|
2007-08-12 11:51:26 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
} else {
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-08-12 11:51:26 +02:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
if ((size_t)(str-start+1)<size) {
|
|
|
|
|
if (encoding == XML_CHAR_ENCODING_8859_1 && c&0x80) {
|
|
|
|
|
char inbuf = (char)c;
|
2007-09-16 17:34:49 +02:00
|
|
|
|
size_t inbytes = 1;
|
|
|
|
|
size_t outbytes = size-(str-start);
|
2008-04-25 16:31:38 +02:00
|
|
|
|
int ret = unicode_latin1_to_utf8(str, &outbytes, &inbuf, &inbytes);
|
2007-08-12 11:51:26 +02:00
|
|
|
|
if (ret>0) str+=ret;
|
2007-09-16 17:34:49 +02:00
|
|
|
|
else {
|
|
|
|
|
log_error(("input data was not iso-8859-1! assuming utf-8\n"));
|
|
|
|
|
encoding = XML_CHAR_ENCODING_ERROR;
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
2007-08-12 11:51:26 +02:00
|
|
|
|
} else {
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
|
2007-08-12 11:51:26 +02:00
|
|
|
|
/** writes a quoted string to the file
|
|
|
|
|
* no trailing space, since this is used to make the creport.
|
|
|
|
|
*/
|
|
|
|
|
int
|
|
|
|
|
fwritestr(FILE * F, const char * str)
|
|
|
|
|
{
|
|
|
|
|
int nwrite = 0;
|
|
|
|
|
fputc('\"', F);
|
2007-12-18 00:56:53 +01:00
|
|
|
|
if (str) while (*str) {
|
2007-08-12 11:51:26 +02:00
|
|
|
|
int c = (int)(unsigned char)*str++;
|
|
|
|
|
switch (c) {
|
|
|
|
|
case '"':
|
|
|
|
|
case '\\':
|
|
|
|
|
fputc('\\', F);
|
|
|
|
|
fputc(c, F);
|
|
|
|
|
nwrite+=2;
|
|
|
|
|
break;
|
|
|
|
|
case '\n':
|
|
|
|
|
fputc('\\', F);
|
|
|
|
|
fputc('n', F);
|
|
|
|
|
nwrite+=2;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
fputc(c, F);
|
|
|
|
|
++nwrite;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
fputc('\"', F);
|
|
|
|
|
return nwrite + 2;
|
|
|
|
|
}
|
|
|
|
|
|
2005-11-20 13:58:59 +01:00
|
|
|
|
#ifdef ENEMIES
|
2002-12-18 01:34:19 +01:00
|
|
|
|
static void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
read_enemies(struct storage * store, faction * f)
|
2002-12-18 01:34:19 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<ENEMIES_VERSION) return;
|
2002-12-18 01:34:19 +01:00
|
|
|
|
f->enemies = NULL;
|
|
|
|
|
for (;;) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int fno = store->r_id(store);
|
2008-05-17 00:37:13 +02:00
|
|
|
|
if (fno<=0) break;
|
2002-12-18 01:34:19 +01:00
|
|
|
|
else {
|
2008-05-17 17:21:17 +02:00
|
|
|
|
variant id;
|
2002-12-18 01:34:19 +01:00
|
|
|
|
faction_list * flist = malloc(sizeof(faction_list));
|
|
|
|
|
flist->next = f->enemies;
|
|
|
|
|
f->enemies = flist;
|
2008-05-17 17:21:17 +02:00
|
|
|
|
id.i = fno;
|
|
|
|
|
ur_add(id, &flist->data, resolve_faction);
|
2002-12-18 01:34:19 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_enemies(struct storage * store, const faction_list * flist)
|
2002-12-18 01:34:19 +01:00
|
|
|
|
{
|
|
|
|
|
while (flist) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_faction_reference(flist->data, store);
|
2002-12-18 01:34:19 +01:00
|
|
|
|
}
|
2008-05-17 00:37:13 +02:00
|
|
|
|
store->w_id(store, 0);
|
2002-12-18 01:34:19 +01:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
static unit *
|
2007-06-26 11:32:28 +02:00
|
|
|
|
unitorders(FILE * F, int enc, struct faction * f)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
int i;
|
|
|
|
|
unit *u;
|
|
|
|
|
|
|
|
|
|
if (!f) return NULL;
|
|
|
|
|
|
|
|
|
|
i = getid();
|
|
|
|
|
u = findunitg(i, NULL);
|
|
|
|
|
|
2005-05-27 23:04:13 +02:00
|
|
|
|
if (u && u->race == new_race[RC_SPELL]) return NULL;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if (u && u->faction == f) {
|
|
|
|
|
order ** ordp;
|
|
|
|
|
|
2005-05-01 13:32:24 +02:00
|
|
|
|
if (!fval(u, UFL_ORDERS)) {
|
|
|
|
|
/* alle wiederholbaren, langen befehle werden gesichert: */
|
|
|
|
|
fset(u, UFL_ORDERS);
|
|
|
|
|
u->old_orders = u->orders;
|
|
|
|
|
ordp = &u->old_orders;
|
|
|
|
|
while (*ordp) {
|
|
|
|
|
order * ord = *ordp;
|
|
|
|
|
if (!is_repeated(ord)) {
|
2005-05-01 13:38:06 +02:00
|
|
|
|
*ordp = ord->next;
|
2005-05-01 15:33:34 +02:00
|
|
|
|
ord->next = NULL;
|
2005-05-01 13:32:24 +02:00
|
|
|
|
free_order(ord);
|
|
|
|
|
} else {
|
|
|
|
|
ordp = &ord->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
free_orders(&u->orders);
|
|
|
|
|
}
|
2004-06-21 18:45:27 +02:00
|
|
|
|
u->orders = 0;
|
|
|
|
|
|
|
|
|
|
ordp = &u->orders;
|
|
|
|
|
|
|
|
|
|
for (;;) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * s;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
/* Erst wenn wir sicher sind, dass kein Befehl
|
|
|
|
|
* eingegeben wurde, checken wir, ob nun eine neue
|
|
|
|
|
* Einheit oder ein neuer Spieler drankommt */
|
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
s = getbuf(F, enc);
|
|
|
|
|
if (s==NULL) break;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
if (s[0]) {
|
2007-08-12 15:02:24 +02:00
|
|
|
|
const char * stok = s;
|
|
|
|
|
stok = parse_token(&stok);
|
|
|
|
|
|
|
|
|
|
if (stok) {
|
|
|
|
|
boolean quit = false;
|
2008-01-08 21:00:19 +01:00
|
|
|
|
param_t param = findparam(stok, u->faction->locale);
|
|
|
|
|
switch (param) {
|
2007-08-12 15:02:24 +02:00
|
|
|
|
case P_UNIT:
|
|
|
|
|
case P_REGION:
|
2008-01-08 21:00:19 +01:00
|
|
|
|
quit = true;
|
|
|
|
|
break;
|
2007-08-12 15:02:24 +02:00
|
|
|
|
case P_FACTION:
|
|
|
|
|
case P_NEXT:
|
|
|
|
|
case P_GAMENAME:
|
2008-01-08 21:00:19 +01:00
|
|
|
|
/* these terminate the orders, so we apply extra checking */
|
|
|
|
|
if (strlen(stok)>=3) {
|
|
|
|
|
quit = true;
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
quit = false;
|
|
|
|
|
}
|
2007-08-12 15:02:24 +02:00
|
|
|
|
}
|
|
|
|
|
if (quit) break;
|
|
|
|
|
}
|
2004-06-21 18:45:27 +02:00
|
|
|
|
/* Nun wird der Befehl erzeut und eingeh<65>ngt */
|
2007-06-26 11:32:28 +02:00
|
|
|
|
*ordp = parse_order(s, u->faction->locale);
|
2005-04-23 11:47:03 +02:00
|
|
|
|
if (*ordp) ordp = &(*ordp)->next;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
} else {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
/* cmistake(?, buf, 160, MSG_EVENT); */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
return u;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static faction *
|
2002-03-03 16:16:28 +01:00
|
|
|
|
factionorders(void)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-02-25 19:49:28 +01:00
|
|
|
|
faction * f = NULL;
|
2007-06-27 02:08:46 +02:00
|
|
|
|
int fid = getid();
|
2006-12-03 14:24:11 +01:00
|
|
|
|
|
2007-06-27 02:08:46 +02:00
|
|
|
|
f = findfaction(fid);
|
2006-03-26 23:14:35 +02:00
|
|
|
|
|
2008-05-04 07:57:34 +02:00
|
|
|
|
if (f!=NULL && !is_monsters(f)) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * pass = getstrtoken();
|
2008-04-26 16:55:22 +02:00
|
|
|
|
|
2007-08-05 14:19:56 +02:00
|
|
|
|
if (!checkpasswd(f, (const char *)pass, true)) {
|
2007-06-27 02:08:46 +02:00
|
|
|
|
log_warning(("Invalid password for faction %s\n", itoa36(fid)));
|
2007-03-25 22:00:14 +02:00
|
|
|
|
ADDMSG(&f->msgs, msg_message("wrongpasswd", "faction password",
|
|
|
|
|
f->no, pass));
|
2006-03-26 23:14:35 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
/* Die Partei hat sich zumindest gemeldet, so da<64> sie noch
|
|
|
|
|
* nicht als unt<EFBFBD>tig gilt */
|
|
|
|
|
|
2006-08-20 15:32:03 +02:00
|
|
|
|
/* TODO: +1 ist ein Workaround, weil turn erst in process_orders
|
|
|
|
|
* incrementiert wird. */
|
|
|
|
|
f->lastorders = global.data_turn+1;
|
|
|
|
|
|
|
|
|
|
} else {
|
2007-06-27 02:08:46 +02:00
|
|
|
|
log_warning(("orders for invalid faction %s\n", itoa36(fid)));
|
2006-08-20 15:32:03 +02:00
|
|
|
|
}
|
|
|
|
|
return f;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double
|
|
|
|
|
version(void)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
return RELEASE_VERSION * 0.1;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
static param_t
|
2007-08-10 09:03:23 +02:00
|
|
|
|
igetparam (const char *s, const struct locale *lang)
|
2004-06-21 18:45:27 +02:00
|
|
|
|
{
|
|
|
|
|
return findparam (igetstrtoken (s), lang);
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
int
|
2007-08-16 08:53:00 +02:00
|
|
|
|
readorders(const char *filename)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
FILE * F = NULL;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *b;
|
2007-07-17 17:33:48 +02:00
|
|
|
|
int nfactions=0;
|
|
|
|
|
struct faction *f = NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2008-01-07 20:26:36 +01:00
|
|
|
|
if (filename) F = cfopen(filename, "rb");
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (F==NULL) return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
puts(" - lese Befehlsdatei...\n");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-08-21 09:04:44 +02:00
|
|
|
|
/* TODO: recognize UTF8 BOM */
|
2007-08-16 08:53:00 +02:00
|
|
|
|
b = getbuf(F, enc_gamedata);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
/* Auffinden der ersten Partei, und danach abarbeiten bis zur letzten
|
|
|
|
|
* Partei */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
while (b) {
|
|
|
|
|
const struct locale * lang = f?f->locale:default_locale;
|
|
|
|
|
int p;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * s;
|
2001-04-16 16:34:19 +02:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
switch (igetparam(b, lang)) {
|
|
|
|
|
case P_LOCALE:
|
|
|
|
|
s = getstrtoken();
|
2004-05-09 12:18:09 +02:00
|
|
|
|
#undef LOCALE_CHANGE
|
|
|
|
|
#ifdef LOCALE_CHANGE
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (f && find_locale(s)) {
|
|
|
|
|
f->locale = find_locale(s);
|
|
|
|
|
}
|
2001-04-16 16:34:19 +02:00
|
|
|
|
#endif
|
2001-02-17 16:02:50 +01:00
|
|
|
|
|
2007-08-16 08:53:00 +02:00
|
|
|
|
b = getbuf(F, enc_gamedata);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
break;
|
|
|
|
|
case P_GAMENAME:
|
|
|
|
|
case P_FACTION:
|
|
|
|
|
f = factionorders();
|
|
|
|
|
if (f) {
|
|
|
|
|
++nfactions;
|
|
|
|
|
}
|
|
|
|
|
|
2007-08-16 08:53:00 +02:00
|
|
|
|
b = getbuf(F, enc_gamedata);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
/* in factionorders wird nur eine zeile gelesen:
|
|
|
|
|
* diejenige mit dem passwort. Die befehle der units
|
|
|
|
|
* werden geloescht, und die Partei wird als aktiv
|
|
|
|
|
* vermerkt. */
|
|
|
|
|
|
|
|
|
|
case P_UNIT:
|
2007-08-16 08:53:00 +02:00
|
|
|
|
if (!f || !unitorders(F, enc_gamedata, f)) do {
|
|
|
|
|
b = getbuf(F, enc_gamedata);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (!b) break;
|
|
|
|
|
p = igetparam(b, lang);
|
|
|
|
|
} while ((p != P_UNIT || !f) && p != P_FACTION && p != P_NEXT && p != P_GAMENAME);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
/* Falls in unitorders() abgebrochen wird, steht dort entweder eine neue
|
|
|
|
|
* Partei, eine neue Einheit oder das File-Ende. Das switch() wird erneut
|
|
|
|
|
* durchlaufen, und die entsprechende Funktion aufgerufen. Man darf buf
|
|
|
|
|
* auf alle F<EFBFBD>lle nicht <EFBFBD>berschreiben! Bei allen anderen Eintr<EFBFBD>gen hier
|
|
|
|
|
* mu<EFBFBD> buf erneut gef<EFBFBD>llt werden, da die betreffende Information in nur
|
|
|
|
|
* einer Zeile steht, und nun die n<EFBFBD>chste gelesen werden mu<EFBFBD>. */
|
|
|
|
|
|
|
|
|
|
case P_NEXT:
|
|
|
|
|
f = NULL;
|
2007-08-16 08:53:00 +02:00
|
|
|
|
b = getbuf(F, enc_gamedata);
|
2005-04-23 11:47:03 +02:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
2007-08-16 08:53:00 +02:00
|
|
|
|
b = getbuf(F, enc_gamedata);
|
2005-04-23 11:47:03 +02:00
|
|
|
|
break;
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
fclose(F);
|
|
|
|
|
puts("\n");
|
|
|
|
|
log_printf(" %d Befehlsdateien gelesen\n", nfactions);
|
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
/* #define INNER_WORLD */
|
|
|
|
|
/* f<>rs debuggen nur den inneren Teil der Welt laden */
|
|
|
|
|
/* -9;-27;-1;-19;Sumpfloch */
|
|
|
|
|
int
|
|
|
|
|
inner_world(region * r)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
static int xy[2] =
|
|
|
|
|
{18, -45};
|
|
|
|
|
static int size[2] =
|
|
|
|
|
{27, 27};
|
|
|
|
|
|
|
|
|
|
if (r->x >= xy[0] && r->x < xy[0] + size[0] && r->y >= xy[1] && r->y < xy[1] + size[1])
|
|
|
|
|
return 2;
|
|
|
|
|
if (r->x >= xy[0] - 9 && r->x < xy[0] + size[0] + 9 && r->y >= xy[1] - 9 && r->y < xy[1] + size[1] + 9)
|
|
|
|
|
return 1;
|
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int maxregions = -1;
|
2002-05-05 11:34:36 +02:00
|
|
|
|
int loadplane = 0;
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
enum {
|
2007-07-17 17:33:48 +02:00
|
|
|
|
U_MAN,
|
|
|
|
|
U_UNDEAD,
|
|
|
|
|
U_ILLUSION,
|
|
|
|
|
U_FIREDRAGON,
|
|
|
|
|
U_DRAGON,
|
|
|
|
|
U_WYRM,
|
|
|
|
|
U_SPELL,
|
|
|
|
|
U_TAVERNE,
|
|
|
|
|
U_MONSTER,
|
|
|
|
|
U_BIRTHDAYDRAGON,
|
|
|
|
|
U_TREEMAN,
|
|
|
|
|
MAXTYPES
|
2001-01-25 10:37:55 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
race_t
|
|
|
|
|
typus2race(unsigned char typus)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (typus>0 && typus <=11) return (race_t)(typus-1);
|
|
|
|
|
return NORACE;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
create_backup(char *file)
|
|
|
|
|
{
|
2008-04-20 16:48:15 +02:00
|
|
|
|
#ifdef HAVE_LINK
|
2005-12-18 22:53:12 +01:00
|
|
|
|
char bfile[MAX_PATH];
|
|
|
|
|
int c = 1;
|
2005-12-31 14:45:10 +01:00
|
|
|
|
|
2006-03-24 08:35:51 +01:00
|
|
|
|
if (access(file, R_OK) == 0) return;
|
2005-12-18 22:53:12 +01:00
|
|
|
|
do {
|
|
|
|
|
sprintf(bfile, "%s.backup%d", file, c);
|
|
|
|
|
c++;
|
|
|
|
|
} while(access(bfile, R_OK) == 0);
|
|
|
|
|
link(file, bfile);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2001-09-05 21:40:40 +02:00
|
|
|
|
const char *
|
2001-01-25 10:37:55 +01:00
|
|
|
|
datapath(void)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
static char zText[MAX_PATH];
|
|
|
|
|
if (g_datadir) return g_datadir;
|
|
|
|
|
return strcat(strcpy(zText, basepath()), "/data");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2001-09-05 21:40:40 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
read_items(struct storage * store, item **ilist)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
for (;;) {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char ibuf[32];
|
2003-12-14 11:10:30 +01:00
|
|
|
|
const item_type * itype;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, ibuf, sizeof(ibuf));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (!strcmp("end", ibuf)) break;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
itype = it_find(ibuf);
|
2005-10-03 23:46:57 +02:00
|
|
|
|
assert(itype!=NULL);
|
|
|
|
|
if (itype!=NULL) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int i = store->r_int(store);
|
2007-08-12 15:02:24 +02:00
|
|
|
|
if (i<=0) {
|
|
|
|
|
log_error(("data contains an entry with %d %s\n", i, itype->rtype->_name[1]));
|
|
|
|
|
} else {
|
|
|
|
|
i_change(ilist, itype, i);
|
|
|
|
|
}
|
2005-10-03 23:46:57 +02:00
|
|
|
|
}
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2004-06-27 18:56:01 +02:00
|
|
|
|
static void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
read_alliances(struct storage * store)
|
2002-08-26 16:16:16 +02:00
|
|
|
|
{
|
2004-06-27 18:56:01 +02:00
|
|
|
|
char pbuf[32];
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<SAVEALLIANCE_VERSION) {
|
2004-06-27 18:56:01 +02:00
|
|
|
|
if (!AllianceRestricted() && !AllianceAuto()) return;
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, pbuf, sizeof(pbuf));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
while (strcmp(pbuf, "end")!=0) {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char aname[128];
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, aname, sizeof(aname));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
makealliance(atoi36(pbuf), aname);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, pbuf, sizeof(pbuf));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2002-08-26 16:16:16 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_alliances(struct storage * store)
|
2002-08-26 16:16:16 +02:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
alliance * al = alliances;
|
|
|
|
|
while (al) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_id(store, al->id);
|
|
|
|
|
store->w_tok(store, al->name);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
al = al->next;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
store->w_brk(store);
|
2002-08-26 16:16:16 +02:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_items(struct storage * store, item *ilist)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
item * itm;
|
2007-08-12 15:02:24 +02:00
|
|
|
|
for (itm=ilist;itm;itm=itm->next) {
|
|
|
|
|
assert(itm->number>=0);
|
|
|
|
|
if (itm->number) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, resourcename(itm->type->rtype, 0));
|
|
|
|
|
store->w_int(store, itm->number);
|
2007-08-12 15:02:24 +02:00
|
|
|
|
}
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "end");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
lastturn(void)
|
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
int turn = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#ifdef HAVE_READDIR
|
2007-07-17 17:33:48 +02:00
|
|
|
|
DIR *data_dir = NULL;
|
|
|
|
|
struct dirent *entry = NULL;
|
|
|
|
|
const char * dir = datapath();
|
|
|
|
|
data_dir = opendir(dir);
|
|
|
|
|
if (data_dir != NULL) {
|
|
|
|
|
entry = readdir(data_dir);
|
|
|
|
|
}
|
|
|
|
|
if (data_dir != NULL && entry != NULL) {
|
|
|
|
|
turn = 0;
|
|
|
|
|
do {
|
|
|
|
|
int i = atoi(entry->d_name);
|
|
|
|
|
if (i > turn)
|
|
|
|
|
turn = i;
|
|
|
|
|
entry = readdir(data_dir);
|
|
|
|
|
} while (entry != NULL);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#ifdef HAVE_CLOSEDIR
|
2007-07-17 17:33:48 +02:00
|
|
|
|
closedir(data_dir);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#endif
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#else
|
|
|
|
|
# error "requires dirent.h or an equivalent to compile!"
|
|
|
|
|
#endif
|
2007-07-17 17:33:48 +02:00
|
|
|
|
return turn;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
static void
|
|
|
|
|
writeorder(struct storage * store, const struct order * ord, const struct locale * lang)
|
|
|
|
|
{
|
|
|
|
|
char obuf[1024];
|
|
|
|
|
write_order(ord, lang, obuf, sizeof(obuf));
|
|
|
|
|
if (obuf[0]) store->w_str(store, obuf);
|
2004-06-21 18:45:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
2002-03-03 16:16:28 +01:00
|
|
|
|
unit *
|
2008-04-26 16:55:22 +02:00
|
|
|
|
readunit(struct storage * store)
|
2002-03-03 16:16:28 +01:00
|
|
|
|
{
|
2007-02-25 14:35:52 +01:00
|
|
|
|
skill_t sk;
|
|
|
|
|
unit * u;
|
|
|
|
|
int number, n, p;
|
2005-07-27 14:09:53 +02:00
|
|
|
|
order ** orderp;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char obuf[1024];
|
2008-05-17 00:37:13 +02:00
|
|
|
|
faction * f;
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
n = store->r_id(store);
|
2007-02-25 14:35:52 +01:00
|
|
|
|
u = findunit(n);
|
|
|
|
|
if (u==NULL) {
|
|
|
|
|
u = calloc(sizeof(unit), 1);
|
|
|
|
|
u->no = n;
|
|
|
|
|
uhash(u);
|
|
|
|
|
} else {
|
|
|
|
|
while (u->attribs) a_remove(&u->attribs, u->attribs);
|
|
|
|
|
while (u->items) i_free(i_remove(&u->items, u->items));
|
|
|
|
|
free(u->skills);
|
|
|
|
|
u->skills = 0;
|
|
|
|
|
u->skill_size = 0;
|
|
|
|
|
u_setfaction(u, NULL);
|
|
|
|
|
}
|
2008-05-17 00:37:13 +02:00
|
|
|
|
|
|
|
|
|
n = store->r_id(store);
|
|
|
|
|
f = findfaction(n);
|
|
|
|
|
if (f!=u->faction) u_setfaction(u, f);
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
u->name = store->r_str(store);
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
|
|
|
|
} else {
|
|
|
|
|
u->display = store->r_str(store);
|
2007-06-25 03:50:34 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
number = store->r_int(store);
|
|
|
|
|
u->age = (short)store->r_int(store);
|
|
|
|
|
if (store->version<NEWRACE_VERSION) {
|
|
|
|
|
u->race = new_race[(race_t)store->r_int(store)];
|
|
|
|
|
u->irace = new_race[(race_t)store->r_int(store)];
|
2005-08-21 11:24:14 +02:00
|
|
|
|
} else {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char rname[32];
|
2005-08-21 11:24:14 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<STORAGE_VERSION) {
|
|
|
|
|
char * space;
|
|
|
|
|
store->r_str_buf(store, rname, sizeof(rname));
|
|
|
|
|
space = strchr(rname, ' ');
|
|
|
|
|
if (space!=NULL) {
|
|
|
|
|
char * inc = space+1;
|
|
|
|
|
char * outc = space;
|
|
|
|
|
do {
|
|
|
|
|
while (*inc==' ') ++inc;
|
|
|
|
|
while (*inc) {
|
|
|
|
|
*outc++ = *inc++;
|
|
|
|
|
if (*inc==' ') break;
|
|
|
|
|
}
|
|
|
|
|
} while (*inc);
|
|
|
|
|
*outc = 0;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
store->r_tok_buf(store, rname, sizeof(rname));
|
2005-08-21 11:24:14 +02:00
|
|
|
|
}
|
2007-06-26 11:32:28 +02:00
|
|
|
|
u->race = rc_find(rname);
|
2005-08-21 11:24:14 +02:00
|
|
|
|
assert(u->race);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<STORAGE_VERSION) {
|
|
|
|
|
store->r_str_buf(store, rname, sizeof(rname));
|
|
|
|
|
} else {
|
|
|
|
|
store->r_tok_buf(store, rname, sizeof(rname));
|
|
|
|
|
}
|
|
|
|
|
if (rname[0]) u->irace = rc_find(rname);
|
2005-08-21 11:24:14 +02:00
|
|
|
|
else u->irace = u->race;
|
|
|
|
|
}
|
2007-06-25 03:50:34 +02:00
|
|
|
|
if (u->race->describe) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * rcdisp = u->race->describe(u, u->faction->locale);
|
2007-06-25 03:50:34 +02:00
|
|
|
|
if (u->display && rcdisp) {
|
|
|
|
|
/* see if the data file contains old descriptions */
|
2007-08-10 09:03:23 +02:00
|
|
|
|
if (strcmp(rcdisp, u->display)==0) {
|
2007-07-17 17:33:48 +02:00
|
|
|
|
free(u->display);
|
|
|
|
|
u->display = NULL;
|
2007-06-25 03:50:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-02-25 14:35:52 +01:00
|
|
|
|
if (u->faction == NULL) {
|
|
|
|
|
log_error(("unit %s has faction == NULL\n", unitname(u)));
|
2008-05-12 00:51:34 +02:00
|
|
|
|
u_setfaction(u, get_monsters());
|
2007-02-25 14:35:52 +01:00
|
|
|
|
set_number(u, 0);
|
|
|
|
|
}
|
2006-07-29 16:04:49 +02:00
|
|
|
|
|
|
|
|
|
if (count_unit(u)) u->faction->no_units++;
|
|
|
|
|
|
|
|
|
|
set_number(u, number);
|
2007-02-12 08:27:04 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
n = store->r_id(store);
|
2008-05-17 00:37:13 +02:00
|
|
|
|
if (n>0) u->building = findbuilding(n);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
|
|
|
|
|
n = store->r_id(store);
|
2008-05-17 00:37:13 +02:00
|
|
|
|
if (n>0) u->ship = findship(n);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
|
|
|
|
|
if (store->version <= 73) {
|
|
|
|
|
if (store->r_int(store)) {
|
2007-02-12 08:27:04 +01:00
|
|
|
|
fset(u, UFL_OWNER);
|
|
|
|
|
} else {
|
|
|
|
|
freset(u, UFL_OWNER);
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
setstatus(u, store->r_int(store));
|
|
|
|
|
if (store->version <= 73) {
|
|
|
|
|
if (store->r_int(store)) {
|
2007-02-12 08:27:04 +01:00
|
|
|
|
guard(u, GUARD_ALL);
|
|
|
|
|
} else {
|
|
|
|
|
guard(u, GUARD_NONE);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
u->flags = store->r_int(store) & ~UFL_DEBUG;
|
2005-05-28 10:59:14 +02:00
|
|
|
|
u->flags &= UFL_SAVEMASK;
|
|
|
|
|
}
|
2007-02-25 14:35:52 +01:00
|
|
|
|
/* Persistente Befehle einlesen */
|
|
|
|
|
free_orders(&u->orders);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, obuf, sizeof(obuf));
|
2005-07-27 13:48:16 +02:00
|
|
|
|
p = n = 0;
|
2005-07-27 14:09:53 +02:00
|
|
|
|
orderp = &u->orders;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
while (obuf[0]) {
|
2007-09-10 08:51:11 +02:00
|
|
|
|
if (!lomem) {
|
|
|
|
|
order * ord = parse_order(obuf, u->faction->locale);
|
|
|
|
|
if (ord!=NULL) {
|
|
|
|
|
if (++n<MAXORDERS) {
|
|
|
|
|
if (!is_persistent(ord) || ++p<MAXPERSISTENT) {
|
|
|
|
|
*orderp = ord;
|
|
|
|
|
orderp = &ord->next;
|
|
|
|
|
ord = NULL;
|
|
|
|
|
} else if (p==MAXPERSISTENT) {
|
|
|
|
|
log_warning(("%s had %d or more persistent orders\n", unitname(u), MAXPERSISTENT));
|
|
|
|
|
}
|
|
|
|
|
} else if (n==MAXORDERS) {
|
|
|
|
|
log_warning(("%s had %d or more orders\n", unitname(u), MAXORDERS));
|
2005-07-27 13:48:16 +02:00
|
|
|
|
}
|
2007-09-10 08:51:11 +02:00
|
|
|
|
if (ord!=NULL) free_order(ord);
|
2005-07-27 13:48:16 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, obuf, sizeof(obuf));
|
2007-02-25 14:35:52 +01:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<NOLASTORDER_VERSION) {
|
2005-04-30 19:07:46 +02:00
|
|
|
|
order * ord;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, obuf, sizeof(obuf));
|
2007-08-10 09:03:23 +02:00
|
|
|
|
ord = parse_order(obuf, u->faction->locale);
|
2005-04-30 19:07:46 +02:00
|
|
|
|
if (ord!=NULL) {
|
|
|
|
|
#ifdef LASTORDER
|
|
|
|
|
set_order(&u->lastorder, ord);
|
|
|
|
|
#else
|
|
|
|
|
addlist(&u->orders, ord);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
2004-06-26 22:51:19 +02:00
|
|
|
|
set_order(&u->thisorder, NULL);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2006-10-29 15:03:56 +01:00
|
|
|
|
assert(u->race);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<NEWSKILL_VERSION) {
|
2006-10-29 15:03:56 +01:00
|
|
|
|
/* convert old data */
|
2006-10-15 16:09:59 +02:00
|
|
|
|
if (u->number) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
while ((sk = (skill_t) store->r_int(store)) != NOSKILL) {
|
|
|
|
|
int days = store->r_int(store) / u->number;
|
2006-10-29 15:03:56 +01:00
|
|
|
|
int lvl = level(days);
|
|
|
|
|
int weeks = lvl + 1 - (days - level_days(lvl))/30;
|
|
|
|
|
assert(weeks>0 && weeks<=lvl+1);
|
|
|
|
|
if (lvl) {
|
|
|
|
|
skill * sv = add_skill(u, sk);
|
|
|
|
|
sv->level = sv->old = (unsigned char)lvl;
|
|
|
|
|
sv->weeks = (unsigned char)weeks;
|
|
|
|
|
}
|
2006-10-15 16:09:59 +02:00
|
|
|
|
}
|
2006-10-29 15:03:56 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
while ((sk = (skill_t) store->r_int(store)) != NOSKILL) {
|
|
|
|
|
int level = store->r_int(store);
|
|
|
|
|
int weeks = store->r_int(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (level) {
|
|
|
|
|
skill * sv = add_skill(u, sk);
|
|
|
|
|
sv->level = sv->old = (unsigned char)level;
|
|
|
|
|
sv->weeks = (unsigned char)weeks;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
read_items(store, &u->items);
|
|
|
|
|
u->hp = store->r_int(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (u->hp < u->number) {
|
|
|
|
|
log_error(("Einheit %s hat %u Personen, und %u Trefferpunkte\n", itoa36(u->no),
|
|
|
|
|
u->number, u->hp));
|
|
|
|
|
u->hp=u->number;
|
|
|
|
|
}
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_read(store, &u->attribs);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
return u;
|
2002-03-03 16:16:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writeunit(struct storage * store, const unit * u)
|
2002-03-03 16:16:28 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
order * ord;
|
|
|
|
|
int i, p = 0;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_unit_reference(u, store);
|
|
|
|
|
write_faction_reference(u->faction, store);
|
|
|
|
|
store->w_str(store, (const char *)u->name);
|
|
|
|
|
store->w_str(store, u->display?(const char *)u->display:"");
|
|
|
|
|
store->w_int(store, u->number);
|
|
|
|
|
store->w_int(store, u->age);
|
|
|
|
|
store->w_tok(store, u->race->_name[0]);
|
|
|
|
|
store->w_tok(store, u->irace!=u->race?u->irace->_name[0]:"");
|
|
|
|
|
write_building_reference(u->building, store);
|
|
|
|
|
write_ship_reference(u->ship, store);
|
|
|
|
|
store->w_int(store, u->status);
|
|
|
|
|
store->w_int(store, u->flags & UFL_SAVEMASK);
|
|
|
|
|
store->w_brk(store);
|
2005-05-01 13:32:24 +02:00
|
|
|
|
#ifndef LASTORDER
|
|
|
|
|
for (ord = u->old_orders; ord; ord=ord->next) {
|
2005-08-22 21:49:51 +02:00
|
|
|
|
if (++p<MAXPERSISTENT) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writeorder(store, ord, u->faction->locale);
|
2005-08-22 21:49:51 +02:00
|
|
|
|
} else {
|
|
|
|
|
log_error(("%s had %d or more persistent orders\n", unitname(u), MAXPERSISTENT));
|
|
|
|
|
break;
|
|
|
|
|
}
|
2005-05-01 13:32:24 +02:00
|
|
|
|
}
|
|
|
|
|
#endif
|
2005-08-07 15:48:18 +02:00
|
|
|
|
for (ord = u->orders; ord; ord=ord->next) {
|
2005-06-05 18:48:22 +02:00
|
|
|
|
if (u->old_orders && is_repeated(ord)) continue; /* has new defaults */
|
2005-08-07 15:48:18 +02:00
|
|
|
|
if (is_persistent(ord)) {
|
2005-08-22 21:49:51 +02:00
|
|
|
|
if (++p<MAXPERSISTENT) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writeorder(store, ord, u->faction->locale);
|
2005-08-22 21:49:51 +02:00
|
|
|
|
} else {
|
|
|
|
|
log_error(("%s had %d or more persistent orders\n", unitname(u), MAXPERSISTENT));
|
|
|
|
|
break;
|
|
|
|
|
}
|
2005-08-07 15:48:18 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2004-07-03 21:16:17 +02:00
|
|
|
|
/* write an empty string to terminate the list */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_str(store, "");
|
|
|
|
|
store->w_brk(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2005-08-07 15:48:18 +02:00
|
|
|
|
assert(u->race);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2006-10-29 15:03:56 +01:00
|
|
|
|
for (i=0;i!=u->skill_size;++i) {
|
|
|
|
|
skill * sv = u->skills+i;
|
|
|
|
|
assert(sv->weeks<=sv->level*2+1);
|
|
|
|
|
if (sv->level>0) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, sv->id);
|
|
|
|
|
store->w_int(store, sv->level);
|
|
|
|
|
store->w_int(store, sv->weeks);
|
2006-10-29 15:03:56 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, -1);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_items(store, u->items);
|
|
|
|
|
store->w_brk(store);
|
2006-10-29 15:03:56 +01:00
|
|
|
|
if (u->hp == 0) {
|
|
|
|
|
log_error(("Einheit %s hat 0 Trefferpunkte\n", itoa36(u->no)));
|
|
|
|
|
((unit*)u)->hp = 1;
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, u->hp);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
a_write(store, u->attribs);
|
|
|
|
|
store->w_brk(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
static region *
|
|
|
|
|
readregion(struct storage * store, short x, short y)
|
2002-03-03 16:16:28 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
region * r = findregion(x, y);
|
|
|
|
|
const terrain_type * terrain;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char token[32];
|
2008-04-22 12:06:53 +02:00
|
|
|
|
unsigned int uid = 0;
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version>=UID_VERSION) {
|
|
|
|
|
uid = store->r_int(store);
|
2008-04-22 12:06:53 +02:00
|
|
|
|
} else {
|
|
|
|
|
uid = generate_region_id();
|
|
|
|
|
}
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (r==NULL) {
|
2008-04-22 12:06:53 +02:00
|
|
|
|
r = new_region(x, y, uid);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
} else {
|
2008-04-22 12:06:53 +02:00
|
|
|
|
r->uid = uid;
|
2007-07-17 17:33:48 +02:00
|
|
|
|
current_region = r;
|
|
|
|
|
while (r->attribs) a_remove(&r->attribs, r->attribs);
|
|
|
|
|
if (r->land) {
|
|
|
|
|
free(r->land); /* mem leak */
|
|
|
|
|
r->land->demands = 0; /* mem leak */
|
|
|
|
|
}
|
|
|
|
|
while (r->resources) {
|
|
|
|
|
rawmaterial * rm = r->resources;
|
|
|
|
|
r->resources = rm->next;
|
|
|
|
|
free(rm);
|
|
|
|
|
}
|
|
|
|
|
r->land = 0;
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
|
|
|
|
} else {
|
|
|
|
|
r->display = store->r_str(store);
|
|
|
|
|
}
|
2005-10-25 14:38:01 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version < TERRAIN_VERSION) {
|
|
|
|
|
int ter = store->r_int(store);
|
2005-10-25 14:38:01 +02:00
|
|
|
|
terrain = newterrain((terrain_t)ter);
|
|
|
|
|
} else {
|
|
|
|
|
char name[64];
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, name, sizeof(name));
|
2005-10-25 14:38:01 +02:00
|
|
|
|
terrain = get_terrain(name);
|
2005-12-31 14:45:10 +01:00
|
|
|
|
if (terrain==NULL) {
|
|
|
|
|
log_error(("Unknown terrain '%s'\n", name));
|
|
|
|
|
assert(!"unknown terrain");
|
|
|
|
|
}
|
2005-10-25 14:38:01 +02:00
|
|
|
|
}
|
2005-12-31 14:45:10 +01:00
|
|
|
|
r->terrain = terrain;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
r->flags = (char) store->r_int(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
r->age = (unsigned short) store->r_int(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2005-12-31 14:45:10 +01:00
|
|
|
|
if (fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
r->land = calloc(1, sizeof(land_region));
|
2008-04-26 16:55:22 +02:00
|
|
|
|
r->land->name = store->r_str(store);
|
2005-12-31 14:45:10 +01:00
|
|
|
|
}
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (r->land) {
|
|
|
|
|
int i;
|
2007-08-21 09:04:44 +02:00
|
|
|
|
rawmaterial ** pres = &r->resources;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version < GROWTREE_VERSION) {
|
|
|
|
|
i = store->r_int(store);
|
|
|
|
|
rsettrees(r, 2, i);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
i = store->r_int(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (i<0) {
|
|
|
|
|
log_error(("number of trees in %s is %d.\n",
|
|
|
|
|
regionname(r, NULL), i));
|
|
|
|
|
i=0;
|
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 0, i);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
i = store->r_int(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (i<0) {
|
|
|
|
|
log_error(("number of young trees in %s is %d.\n",
|
|
|
|
|
regionname(r, NULL), i));
|
|
|
|
|
i=0;
|
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 1, i);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
i = store->r_int(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (i<0) {
|
|
|
|
|
log_error(("number of seeds in %s is %d.\n",
|
|
|
|
|
regionname(r, NULL), i));
|
|
|
|
|
i=0;
|
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 2, i);
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
i = store->r_int(store); rsethorses(r, i);
|
2007-08-21 09:04:44 +02:00
|
|
|
|
assert(*pres==NULL);
|
|
|
|
|
for (;;) {
|
|
|
|
|
rawmaterial * res;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-08-21 09:04:44 +02:00
|
|
|
|
if (strcmp(token, "end")==0) break;
|
|
|
|
|
res = malloc(sizeof(rawmaterial));
|
|
|
|
|
res->type = rmt_find(token);
|
|
|
|
|
if (res->type==NULL) {
|
|
|
|
|
log_error(("invalid resourcetype %s in data.\n", token));
|
|
|
|
|
}
|
|
|
|
|
assert(res->type!=NULL);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
res->level = store->r_int(store);
|
|
|
|
|
res->amount = store->r_int(store);
|
2007-08-21 09:04:44 +02:00
|
|
|
|
res->flags = 0;
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if(store->version >= RANDOMIZED_RESOURCES_VERSION) {
|
|
|
|
|
res->startlevel = store->r_int(store);
|
|
|
|
|
res->base = store->r_int(store);
|
|
|
|
|
res->divisor = store->r_int(store);
|
2007-08-21 09:04:44 +02:00
|
|
|
|
} else {
|
|
|
|
|
int i;
|
|
|
|
|
for (i=0;r->terrain->production[i].type;++i) {
|
|
|
|
|
if (res->type->rtype == r->terrain->production[i].type) break;
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-08-21 09:04:44 +02:00
|
|
|
|
res->base = dice_rand(r->terrain->production[i].base);
|
|
|
|
|
res->divisor = dice_rand(r->terrain->production[i].divisor);
|
|
|
|
|
res->startlevel = 1;
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2007-08-21 09:04:44 +02:00
|
|
|
|
|
|
|
|
|
*pres = res;
|
|
|
|
|
pres=&res->next;
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2007-08-21 09:04:44 +02:00
|
|
|
|
*pres = NULL;
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (strcmp(token, "noherb") != 0) {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
const resource_type * rtype = rt_find(token);
|
2005-10-26 22:12:28 +02:00
|
|
|
|
assert(rtype && rtype->itype && fval(rtype->itype, ITF_HERB));
|
|
|
|
|
rsetherbtype(r, rtype->itype);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
} else {
|
|
|
|
|
rsetherbtype(r, NULL);
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
rsetherbs(r, (short)store->r_int(store));
|
|
|
|
|
rsetpeasants(r, store->r_int(store));
|
|
|
|
|
rsetmoney(r, store->r_int(store));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(r->terrain!=NULL);
|
|
|
|
|
assert(rhorses(r) >= 0);
|
|
|
|
|
assert(rpeasants(r) >= 0);
|
|
|
|
|
assert(rmoney(r) >= 0);
|
|
|
|
|
|
|
|
|
|
if (r->land) {
|
|
|
|
|
for (;;) {
|
2005-11-01 21:33:21 +01:00
|
|
|
|
const struct item_type * itype;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (!strcmp(token, "end")) break;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
itype = it_find(token);
|
2005-11-01 21:33:21 +01:00
|
|
|
|
assert(itype->rtype->ltype);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
r_setdemand(r, itype->rtype->ltype, store->r_int(store));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version>=REGIONITEMS_VERSION) {
|
|
|
|
|
read_items(store, &r->land->items);
|
2005-12-11 15:31:06 +01:00
|
|
|
|
}
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_read(store, &r->attribs);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2007-07-17 17:33:48 +02:00
|
|
|
|
return r;
|
2002-03-03 16:16:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writeregion(struct storage * store, const region * r)
|
2002-03-03 16:16:28 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, r->uid);
|
|
|
|
|
store->w_str(store, r->display?(const char *)r->display:"");
|
|
|
|
|
store->w_tok(store, r->terrain->_name);
|
|
|
|
|
store->w_int(store, r->flags & RF_SAVEMASK);
|
|
|
|
|
store->w_int(store, r->age);
|
|
|
|
|
store->w_brk(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
const item_type *rht;
|
|
|
|
|
struct demand * demand;
|
2005-10-25 14:38:01 +02:00
|
|
|
|
rawmaterial * res = r->resources;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_str(store, (const char *)r->land->name);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
assert(rtrees(r,0)>=0);
|
|
|
|
|
assert(rtrees(r,1)>=0);
|
|
|
|
|
assert(rtrees(r,2)>=0);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, rtrees(r,0));
|
|
|
|
|
store->w_int(store, rtrees(r,1));
|
|
|
|
|
store->w_int(store, rtrees(r,2));
|
|
|
|
|
store->w_int(store, rhorses(r));
|
2005-10-25 14:38:01 +02:00
|
|
|
|
|
|
|
|
|
while (res) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, res->type->name);
|
|
|
|
|
store->w_int(store, res->level);
|
|
|
|
|
store->w_int(store, res->amount);
|
|
|
|
|
store->w_int(store, res->startlevel);
|
|
|
|
|
store->w_int(store, res->base);
|
|
|
|
|
store->w_int(store, res->divisor);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
res = res->next;
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "end");
|
2005-10-25 14:38:01 +02:00
|
|
|
|
|
|
|
|
|
rht = rherbtype(r);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (rht) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, resourcename(rht->rtype, 0));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "noherb");
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, rherbs(r));
|
|
|
|
|
store->w_int(store, rpeasants(r));
|
|
|
|
|
store->w_int(store, rmoney(r));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
if (r->land) for (demand=r->land->demands; demand; demand=demand->next) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, resourcename(demand->type->itype->rtype, 0));
|
|
|
|
|
store->w_int(store, demand->value);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "end");
|
2005-12-11 15:31:06 +01:00
|
|
|
|
#if RELEASE_VERSION>=REGIONITEMS_VERSION
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_items(store, r->land->items);
|
|
|
|
|
store->w_brk(store);
|
2005-12-11 15:31:06 +01:00
|
|
|
|
#endif
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_write(store, r->attribs);
|
|
|
|
|
store->w_brk(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
2004-05-04 21:24:03 +02:00
|
|
|
|
static ally **
|
2002-09-02 22:36:12 +02:00
|
|
|
|
addally(const faction * f, ally ** sfp, int aid, int state)
|
|
|
|
|
{
|
2004-05-04 21:24:03 +02:00
|
|
|
|
struct faction * af = findfaction(aid);
|
|
|
|
|
ally * sf;
|
2005-06-10 00:10:35 +02:00
|
|
|
|
|
2004-05-04 21:24:03 +02:00
|
|
|
|
state &= ~HELP_OBSERVE;
|
magicskillboost
-- implementiert. Kann nur einmal pro Partei verwendet werden, und nur von einem Magier. Müssen wir erklären.
Amulett des wahren Sehens und Ring der Unsichtbarkeit geben lediglich einen Bonus auf Wahrnehmung (+2) und Tarnung (+4).
-- das define war nicht aktiviert
Halblinge haben Armbrustschiessen +0 (statt +1).
-- spearates Rassenfile für WDW
Es gibt spezielle Regeln für Allianzen, und der HELFE und KONTAKTIERE Befehl sind eingeschränkt. Konkret kann kein HELP_FIGHT HELP_GUARD HELP_FSTEALTH HELP_MONEY gegeben werden (die anderen sind erlaubt).
-- mit #defeine ALLIES_ONLY definiert
Jede Allianz kann bis zu 15 Magier und 15 Alchemisten haben. Jede einzelne Partei der Allianz kann dabei nicht mehr als 15/Anzahl_Parteien (aufgerundet) Magier bzw. Alchemisten haben, und die Gesamtsumme darf 15 nicht übersteigen.
-- mit #define ALLIANCE_LIMITS gemacht.
Die Startgeschenke (Personen, Silber, ...) werden pro Allianz, nicht pro Spieler verteilt. Größere Allianzen bekommen also weniger pro Spieler.
-- Nochmal geändert: Die Allianzen kriegen jede 168 Personen zum Start, weil sich das gut aufteilen lässt. Das wird auf 28 Einheiten pro Partei gesplittet, jede Einheit hat eines der Talente, außer der Starteinheit, die hat den magicskillboost. Einige Skills kommen öfter vor als andere, das ist nicht einfach vermeidbar. Sollte aber auch wurscht sein, es geht primär darum, lehren zu können.
Es gibt ein Einheitenlimit von 1000 Einheiten pro Allianz.
-- die Regel sparen wir uns einfach mal.
2003-12-14 22:45:47 +01:00
|
|
|
|
#ifndef REGIONOWNERS
|
2004-05-04 21:24:03 +02:00
|
|
|
|
state &= ~HELP_TRAVEL;
|
2002-09-02 22:36:12 +02:00
|
|
|
|
#endif
|
2004-06-27 18:56:01 +02:00
|
|
|
|
|
2004-05-04 21:34:35 +02:00
|
|
|
|
if (state==0) return sfp;
|
magicskillboost
-- implementiert. Kann nur einmal pro Partei verwendet werden, und nur von einem Magier. Müssen wir erklären.
Amulett des wahren Sehens und Ring der Unsichtbarkeit geben lediglich einen Bonus auf Wahrnehmung (+2) und Tarnung (+4).
-- das define war nicht aktiviert
Halblinge haben Armbrustschiessen +0 (statt +1).
-- spearates Rassenfile für WDW
Es gibt spezielle Regeln für Allianzen, und der HELFE und KONTAKTIERE Befehl sind eingeschränkt. Konkret kann kein HELP_FIGHT HELP_GUARD HELP_FSTEALTH HELP_MONEY gegeben werden (die anderen sind erlaubt).
-- mit #defeine ALLIES_ONLY definiert
Jede Allianz kann bis zu 15 Magier und 15 Alchemisten haben. Jede einzelne Partei der Allianz kann dabei nicht mehr als 15/Anzahl_Parteien (aufgerundet) Magier bzw. Alchemisten haben, und die Gesamtsumme darf 15 nicht übersteigen.
-- mit #define ALLIANCE_LIMITS gemacht.
Die Startgeschenke (Personen, Silber, ...) werden pro Allianz, nicht pro Spieler verteilt. Größere Allianzen bekommen also weniger pro Spieler.
-- Nochmal geändert: Die Allianzen kriegen jede 168 Personen zum Start, weil sich das gut aufteilen lässt. Das wird auf 28 Einheiten pro Partei gesplittet, jede Einheit hat eines der Talente, außer der Starteinheit, die hat den magicskillboost. Einige Skills kommen öfter vor als andere, das ist nicht einfach vermeidbar. Sollte aber auch wurscht sein, es geht primär darum, lehren zu können.
Es gibt ein Einheitenlimit von 1000 Einheiten pro Allianz.
-- die Regel sparen wir uns einfach mal.
2003-12-14 22:45:47 +01:00
|
|
|
|
|
|
|
|
|
sf = calloc(1, sizeof(ally));
|
2004-05-04 21:24:03 +02:00
|
|
|
|
sf->faction = af;
|
2005-06-10 00:10:35 +02:00
|
|
|
|
if (!sf->faction) {
|
|
|
|
|
variant id;
|
|
|
|
|
id.i = aid;
|
2008-05-17 17:21:17 +02:00
|
|
|
|
ur_add(id, &sf->faction, resolve_faction);
|
2005-06-10 00:10:35 +02:00
|
|
|
|
}
|
2005-02-27 15:44:21 +01:00
|
|
|
|
sf->status = state & HELP_ALL;
|
2002-09-02 22:36:12 +02:00
|
|
|
|
|
2004-05-04 21:24:03 +02:00
|
|
|
|
while (*sfp) sfp=&(*sfp)->next;
|
|
|
|
|
*sfp = sf;
|
2004-05-04 21:34:35 +02:00
|
|
|
|
return &sf->next;
|
2002-09-02 22:36:12 +02:00
|
|
|
|
}
|
|
|
|
|
|
2002-03-03 16:16:28 +01:00
|
|
|
|
/** Reads a faction from a file.
|
|
|
|
|
* This function requires no context, can be called in any state. The
|
|
|
|
|
* faction may not already exist, however.
|
|
|
|
|
*/
|
|
|
|
|
faction *
|
2008-04-26 16:55:22 +02:00
|
|
|
|
readfaction(struct storage * store)
|
2002-03-03 16:16:28 +01:00
|
|
|
|
{
|
2004-04-10 22:25:40 +02:00
|
|
|
|
ally **sfp;
|
|
|
|
|
int planes;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int i = store->r_id(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
faction * f = findfaction(i);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
char email[128];
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char token[32];
|
2004-08-22 11:33:15 +02:00
|
|
|
|
|
2004-04-10 22:25:40 +02:00
|
|
|
|
if (f==NULL) {
|
|
|
|
|
f = (faction *) calloc(1, sizeof(faction));
|
|
|
|
|
f->no = i;
|
|
|
|
|
} else {
|
|
|
|
|
f->allies = NULL; /* mem leak */
|
|
|
|
|
while (f->attribs) a_remove(&f->attribs, f->attribs);
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
f->subscription = store->r_int(store);
|
2004-06-27 18:56:01 +02:00
|
|
|
|
|
|
|
|
|
if (alliances!=NULL) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int allianceid = store->r_id(store);
|
2008-05-17 00:37:13 +02:00
|
|
|
|
if (allianceid>0) f->alliance = findalliance(allianceid);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
if (f->alliance) {
|
|
|
|
|
faction_list * flist = malloc(sizeof(faction_list));
|
|
|
|
|
flist->data = f;
|
|
|
|
|
flist->next = f->alliance->members;
|
|
|
|
|
f->alliance->members = flist;
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
f->name = store->r_str(store);
|
|
|
|
|
f->banner = store->r_str(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
log_info((3, " - Lese Partei %s (%s)\n", f->name, factionid(f)));
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, email, sizeof(email));
|
2004-08-22 11:33:15 +02:00
|
|
|
|
if (set_email(&f->email, email)!=0) {
|
2007-02-21 20:59:27 +01:00
|
|
|
|
log_warning(("Invalid email address for faction %s: %s\n", itoa36(f->no), email));
|
2004-08-22 11:33:15 +02:00
|
|
|
|
set_email(&f->email, "");
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
|
|
|
|
|
f->passw = store->r_str(store);
|
|
|
|
|
if (store->version >= OVERRIDE_VERSION) {
|
|
|
|
|
f->override = store->r_str(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
} else {
|
2007-08-05 14:19:56 +02:00
|
|
|
|
f->override = strdup(itoa36(rng_int()));
|
2004-04-10 22:25:40 +02:00
|
|
|
|
}
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-08-21 09:04:44 +02:00
|
|
|
|
f->locale = find_locale(token);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
f->lastorders = store->r_int(store);
|
|
|
|
|
f->age = store->r_int(store);
|
|
|
|
|
if (store->version < NEWRACE_VERSION) {
|
|
|
|
|
race_t rc = (char) store->r_int(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
f->race = new_race[rc];
|
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-06-26 11:32:28 +02:00
|
|
|
|
f->race = rc_find(token);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
assert(f->race);
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
f->magiegebiet = (magic_t)store->r_int(store);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
|
2008-04-20 17:18:43 +02:00
|
|
|
|
#if KARMA_MODULE
|
2008-04-26 16:55:22 +02:00
|
|
|
|
f->karma = store->r_int(store);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#else
|
|
|
|
|
/* ignore karma */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_int(store);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#endif /* KARMA_MODULE */
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
f->flags = store->r_int(store);
|
2008-05-12 11:44:01 +02:00
|
|
|
|
if (f->no==0) {
|
|
|
|
|
f->flags |= FFL_NPC;
|
|
|
|
|
}
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_read(store, &f->attribs);
|
|
|
|
|
if (store->version>=CLAIM_VERSION) {
|
|
|
|
|
read_items(store, &f->items);
|
2004-12-22 01:53:12 +01:00
|
|
|
|
}
|
2004-04-10 22:25:40 +02:00
|
|
|
|
for (;;) {
|
|
|
|
|
int level;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_tok_buf(store, token, sizeof(token));
|
2007-06-26 11:32:28 +02:00
|
|
|
|
if (strcmp("end", token)==0) break;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
level = store->r_int(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
planes = store->r_int(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
while(--planes >= 0) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int id = store->r_int(store);
|
|
|
|
|
short ux = (short)store->r_int(store);
|
|
|
|
|
short uy = (short)store->r_int(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
set_ursprung(f, id, ux, uy);
|
|
|
|
|
}
|
|
|
|
|
f->newbies = 0;
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
i = f->options = store->r_int(store);
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-05-04 07:57:34 +02:00
|
|
|
|
if ((i & (want(O_REPORT)|want(O_COMPUTER)))==0 && !is_monsters(f)) {
|
2004-04-10 22:25:40 +02:00
|
|
|
|
/* Kein Report eingestellt, Fehler */
|
2006-07-29 16:04:49 +02:00
|
|
|
|
f->options = f->options | want(O_REPORT) | want(O_ZUGVORLAGE);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sfp = &f->allies;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<ALLIANCES_VERSION) {
|
|
|
|
|
int p = store->r_int(store);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
while (--p >= 0) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int aid = store->r_id(store);
|
|
|
|
|
int state = store->r_int(store);
|
2004-05-04 21:24:03 +02:00
|
|
|
|
sfp = addally(f, sfp, aid, state);
|
2004-04-10 22:25:40 +02:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for (;;) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int aid = 0;
|
|
|
|
|
if (store->version<STORAGE_VERSION) {
|
|
|
|
|
store->r_tok_buf(store, token, sizeof(token));
|
|
|
|
|
if (strcmp(token, "end")!=0) {
|
|
|
|
|
aid = atoi36(token);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
aid = store->r_id(store);
|
|
|
|
|
}
|
|
|
|
|
if (aid>0) {
|
|
|
|
|
int state = store->r_int(store);
|
2004-05-04 21:24:03 +02:00
|
|
|
|
sfp = addally(f, sfp, aid, state);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
} else {
|
|
|
|
|
break;
|
2004-04-10 22:25:40 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
read_groups(store, f);
|
2005-11-20 13:58:59 +01:00
|
|
|
|
#ifdef ENEMIES
|
2008-04-26 16:55:22 +02:00
|
|
|
|
read_enemies(store, f);
|
2002-12-18 01:34:19 +01:00
|
|
|
|
#endif
|
2004-04-10 22:25:40 +02:00
|
|
|
|
return f;
|
2002-03-03 16:16:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writefaction(struct storage * store, const faction * f)
|
2002-03-03 16:16:28 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
ally *sf;
|
|
|
|
|
ursprung *ur;
|
2002-03-03 16:16:28 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_faction_reference(f, store);
|
|
|
|
|
store->w_int(store, f->subscription);
|
2004-06-27 18:56:01 +02:00
|
|
|
|
if (alliances!=NULL) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (f->alliance) store->w_id(store, f->alliance->id);
|
2008-05-17 00:37:13 +02:00
|
|
|
|
else store->w_id(store, 0);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
store->w_str(store, (const char *)f->name);
|
|
|
|
|
store->w_str(store, (const char *)f->banner);
|
|
|
|
|
store->w_str(store, f->email);
|
|
|
|
|
store->w_tok(store, (const char *)f->passw);
|
|
|
|
|
store->w_tok(store, (const char *)f->override);
|
|
|
|
|
store->w_tok(store, locale_name(f->locale));
|
|
|
|
|
store->w_int(store, f->lastorders);
|
|
|
|
|
store->w_int(store, f->age);
|
|
|
|
|
store->w_tok(store, f->race->_name[0]);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, f->magiegebiet);
|
2008-04-20 17:18:43 +02:00
|
|
|
|
#if KARMA_MODULE
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, f->karma);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#else
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, 0);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#endif /* KARMA_MODULE */
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, f->flags&FFL_SAVEMASK);
|
|
|
|
|
a_write(store, f->attribs);
|
|
|
|
|
store->w_brk(store);
|
2004-12-22 19:16:59 +01:00
|
|
|
|
#if RELEASE_VERSION>=CLAIM_VERSION
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_items(store, f->items);
|
|
|
|
|
store->w_brk(store);
|
2004-12-22 19:16:59 +01:00
|
|
|
|
#endif
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, listlen(f->ursprung));
|
2007-07-17 17:33:48 +02:00
|
|
|
|
for(ur = f->ursprung;ur;ur=ur->next) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, ur->id);
|
|
|
|
|
store->w_int(store, ur->x);
|
|
|
|
|
store->w_int(store, ur->y);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, f->options & ~want(O_DEBUG));
|
|
|
|
|
store->w_brk(store);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
|
|
|
|
|
for (sf = f->allies; sf; sf = sf->next) {
|
|
|
|
|
int no = (sf->faction!=NULL)?sf->faction->no:0;
|
|
|
|
|
int status = alliedfaction(NULL, f, sf->faction, HELP_ALL);
|
|
|
|
|
if (status!=0) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_id(store, no);
|
|
|
|
|
store->w_int(store, sf->status);
|
2007-07-17 17:33:48 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-05-17 00:37:13 +02:00
|
|
|
|
store->w_id(store, 0);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_groups(store, f->groups);
|
2005-11-20 13:58:59 +01:00
|
|
|
|
#ifdef ENEMIES
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_enemies(store, f->enemies);
|
2002-12-18 01:34:19 +01:00
|
|
|
|
#endif
|
2002-03-03 16:26:17 +01:00
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
readgame(const char * filename, int mode, int backup)
|
2004-04-12 01:56:47 +02:00
|
|
|
|
{
|
|
|
|
|
int i, n, p;
|
|
|
|
|
faction *f, **fp;
|
|
|
|
|
region *r;
|
|
|
|
|
building *b, **bp;
|
2006-02-17 23:59:03 +01:00
|
|
|
|
ship **shp;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
unit *u;
|
|
|
|
|
int rmax = maxregions;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char path[MAX_PATH];
|
|
|
|
|
char token[32];
|
2007-10-27 17:09:36 +02:00
|
|
|
|
const struct building_type * bt_lighthouse = bt_find("lighthouse");
|
2008-04-26 16:55:22 +02:00
|
|
|
|
storage my_store = (mode==IO_BINARY)?binary_store:text_store;
|
|
|
|
|
storage * store = &my_store;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
sprintf(path, "%s/%s", datapath(), filename);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
log_printf("- reading game data from %s\n", filename);
|
2007-06-26 11:32:28 +02:00
|
|
|
|
if (backup) create_backup(path);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->encoding = enc_gamedata;
|
|
|
|
|
if (store->open(store, path, IO_READ)!=0) {
|
|
|
|
|
return -1;
|
2007-09-02 11:59:28 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
enc_gamedata = store->encoding;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
assert(store->version>=MIN_VERSION || !"unsupported data format");
|
|
|
|
|
assert(store->version<=RELEASE_VERSION || !"unsupported data format");
|
|
|
|
|
assert(store->version >= GROWTREE_VERSION);
|
|
|
|
|
if(store->version >= SAVEXMLNAME_VERSION) {
|
2004-04-12 01:56:47 +02:00
|
|
|
|
char basefile[1024];
|
|
|
|
|
const char *basearg = "(null)";
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, basefile, sizeof(basefile));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
assert(xmlfile != NULL);
|
|
|
|
|
basearg = strrchr(xmlfile, '/');
|
|
|
|
|
if (basearg==NULL) {
|
|
|
|
|
basearg = xmlfile;
|
|
|
|
|
} else {
|
|
|
|
|
++basearg;
|
|
|
|
|
}
|
|
|
|
|
if (strcmp(basearg, basefile)!=0) {
|
2004-05-25 23:50:23 +02:00
|
|
|
|
log_warning(("xmlfile mismatch: datafile contains %s, argument/default is %s\n", basefile, basearg));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
printf("WARNING: any key to continue, Ctrl-C to stop\n");
|
|
|
|
|
getchar();
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_read(store, &global.attribs);
|
|
|
|
|
global.data_turn = turn = store->r_int(store);
|
2008-05-22 16:40:21 +02:00
|
|
|
|
rng_init(turn);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_int(store); /* max_unique_id = */
|
|
|
|
|
nextborder = store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* Planes */
|
|
|
|
|
planes = NULL;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
n = store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
while(--n >= 0) {
|
2008-04-26 18:45:39 +02:00
|
|
|
|
int id = store->r_int(store);
|
|
|
|
|
plane *pl = getplanebyid(id);
|
|
|
|
|
if (pl==NULL) {
|
2008-04-26 18:48:50 +02:00
|
|
|
|
pl = calloc(1, sizeof(plane));
|
2008-04-26 18:45:39 +02:00
|
|
|
|
} else {
|
|
|
|
|
log_warning(("the plane with id=%d already exists.\n", id));
|
|
|
|
|
}
|
|
|
|
|
pl->id = id;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
pl->name = store->r_str(store);
|
|
|
|
|
pl->minx = (short)store->r_int(store);
|
|
|
|
|
pl->maxx = (short)store->r_int(store);
|
|
|
|
|
pl->miny = (short)store->r_int(store);
|
|
|
|
|
pl->maxy = (short)store->r_int(store);
|
|
|
|
|
pl->flags = store->r_int(store);
|
|
|
|
|
if (store->version>WATCHERS_VERSION) {
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-06-26 11:32:28 +02:00
|
|
|
|
while (strcmp(token, "end")!=0) {
|
2004-04-12 01:56:47 +02:00
|
|
|
|
watcher * w = calloc(sizeof(watcher),1);
|
2005-06-10 00:10:35 +02:00
|
|
|
|
variant fno;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
fno.i = atoi36(token);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
w->mode = (unsigned char)store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
w->next = pl->watchers;
|
|
|
|
|
pl->watchers = w;
|
2008-05-17 17:21:17 +02:00
|
|
|
|
ur_add(fno, &w->faction, resolve_faction);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_read(store, &pl->attribs);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
addlist(&planes, pl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Read factions */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version>=ALLIANCES_VERSION) {
|
|
|
|
|
read_alliances(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
n = store->r_int(store);
|
|
|
|
|
log_info((1, " - Einzulesende Parteien: %d\n", n));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
fp = &factions;
|
|
|
|
|
while (*fp) fp=&(*fp)->next;
|
|
|
|
|
|
|
|
|
|
while (--n >= 0) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
faction * f = readfaction(store);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
|
|
|
|
|
*fp = f;
|
|
|
|
|
fp = &f->next;
|
2005-01-19 21:33:13 +01:00
|
|
|
|
fhash(f);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
*fp = 0;
|
|
|
|
|
|
|
|
|
|
/* Benutzte Faction-Ids */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<STORAGE_VERSION) {
|
|
|
|
|
i = store->r_int(store);
|
|
|
|
|
while (i--) {
|
|
|
|
|
store->r_int(store); /* used faction ids. ignore. */
|
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Regionen */
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
n = store->r_int(store);
|
2008-04-22 12:06:53 +02:00
|
|
|
|
assert(n<MAXREGIONS);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
if (rmax<0) rmax = n;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
log_info((1, " - Einzulesende Regionen: %d/%d\r", rmax, n));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
while (--n >= 0) {
|
|
|
|
|
unit **up;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
short x = (short)store->r_int(store);
|
|
|
|
|
short y = (short)store->r_int(store);
|
|
|
|
|
|
|
|
|
|
if ((n & 0x3FF) == 0) { /* das spart extrem Zeit */
|
|
|
|
|
log_info((2, " - Einzulesende Regionen: %d/%d * %d,%d \r", rmax, n, x, y));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
--rmax;
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
r = readregion(store, x, y);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* Burgen */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
p = store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
bp = &r->buildings;
|
|
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
|
|
|
|
|
|
|
|
|
b = (building *) calloc(1, sizeof(building));
|
2008-04-26 16:55:22 +02:00
|
|
|
|
b->no = store->r_id(store);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
*bp = b;
|
|
|
|
|
bp = &b->next;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
bhash(b);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
b->name = store->r_str(store);
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
b->display = store->r_str(store);
|
|
|
|
|
}
|
|
|
|
|
b->size = store->r_int(store);
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-08-21 09:04:44 +02:00
|
|
|
|
b->type = bt_find(token);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
b->region = r;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_read(store, &b->attribs);
|
2007-10-27 17:09:36 +02:00
|
|
|
|
if (b->type==bt_lighthouse) {
|
|
|
|
|
r->flags |= RF_LIGHTHOUSE;
|
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
/* Schiffe */
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
p = store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
shp = &r->ships;
|
|
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2006-02-17 23:59:03 +01:00
|
|
|
|
ship * sh = (ship *) calloc(1, sizeof(ship));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
sh->region = r;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
sh->no = store->r_id(store);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
*shp = sh;
|
|
|
|
|
shp = &sh->next;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
shash(sh);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
sh->name = store->r_str(store);
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
sh->display = store->r_str(store);
|
|
|
|
|
}
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2007-06-26 11:32:28 +02:00
|
|
|
|
sh->type = st_find(token);
|
|
|
|
|
if (sh->type==NULL) {
|
|
|
|
|
/* old datafiles */
|
2007-07-17 17:33:48 +02:00
|
|
|
|
sh->type = st_find((const char *)locale_string(default_locale, token));
|
2007-06-26 11:32:28 +02:00
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
assert(sh->type || !"ship_type not registered!");
|
2008-04-26 16:55:22 +02:00
|
|
|
|
sh->size = store->r_int(store);
|
|
|
|
|
sh->damage = store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* Attribute rekursiv einlesen */
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
sh->coast = (direction_t)store->r_int(store);
|
|
|
|
|
a_read(store, &sh->attribs);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*shp = 0;
|
|
|
|
|
|
|
|
|
|
/* Einheiten */
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
p = store->r_int(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
up = &r->units;
|
|
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
unit * u = readunit(store);
|
2006-02-05 16:48:52 +01:00
|
|
|
|
sc_mage * mage;
|
2006-02-17 23:59:03 +01:00
|
|
|
|
|
2004-04-12 01:56:47 +02:00
|
|
|
|
assert(u->region==NULL);
|
|
|
|
|
u->region = r;
|
2007-02-28 22:50:19 +01:00
|
|
|
|
if (u->flags&UFL_GUARD) fset(r, RF_GUARDED);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
*up = u;
|
|
|
|
|
up = &u->next;
|
|
|
|
|
|
2005-04-28 09:04:10 +02:00
|
|
|
|
update_interval(u->faction, u->region);
|
2006-02-05 16:48:52 +01:00
|
|
|
|
mage = get_mage(u);
|
|
|
|
|
if (mage && mage->spellcount<0) {
|
|
|
|
|
mage->spellcount = 0;
|
|
|
|
|
updatespelllist(u);
|
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
log_info((1, "\n"));
|
|
|
|
|
read_borders(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->close(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* Unaufgeloeste Zeiger initialisieren */
|
2008-05-12 11:44:01 +02:00
|
|
|
|
log_info((1, "fixing unresolved references.\n"));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
resolve();
|
|
|
|
|
|
2008-05-12 11:44:01 +02:00
|
|
|
|
log_info((1, "updating area information for lighthouses.\n"));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
for (r=regions;r;r=r->next) {
|
2007-10-27 17:09:36 +02:00
|
|
|
|
if (r->flags & RF_LIGHTHOUSE) {
|
|
|
|
|
building * b;
|
|
|
|
|
for (b=r->buildings;b;b=b->next) update_lighthouse(b);
|
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
2008-05-12 11:44:01 +02:00
|
|
|
|
log_info((1, "marking factions as alive.\n"));
|
2004-07-02 07:41:47 +02:00
|
|
|
|
for (f = factions; f; f = f->next) {
|
2008-05-12 11:44:01 +02:00
|
|
|
|
if (f->flags & FFL_NPC) {
|
|
|
|
|
f->alive = 1;
|
|
|
|
|
if (f->no==0) {
|
|
|
|
|
int no=666;
|
|
|
|
|
while (findfaction(no)) ++no;
|
|
|
|
|
log_warning(("renum(monsters, %d)\n", no));
|
|
|
|
|
renumber_faction(f, no);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for (u = f->units; u; u = u->nextF) {
|
|
|
|
|
if (u->number>0) {
|
|
|
|
|
f->alive = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2004-07-02 07:41:47 +02:00
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (loadplane || maxregions>=0) {
|
|
|
|
|
remove_empty_factions(false);
|
|
|
|
|
}
|
2008-05-12 11:44:01 +02:00
|
|
|
|
log_info((1, "Done loading turn %d.\n", turn));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2008-05-12 11:44:01 +02:00
|
|
|
|
|
2004-04-12 01:56:47 +02:00
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writegame(const char *filename, int mode)
|
2004-04-12 01:56:47 +02:00
|
|
|
|
{
|
|
|
|
|
char *base;
|
2006-02-17 23:59:03 +01:00
|
|
|
|
int n;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
faction *f;
|
|
|
|
|
region *r;
|
|
|
|
|
building *b;
|
|
|
|
|
ship *sh;
|
|
|
|
|
unit *u;
|
|
|
|
|
plane *pl;
|
2007-06-26 11:32:28 +02:00
|
|
|
|
char path[MAX_PATH];
|
2008-04-26 16:55:22 +02:00
|
|
|
|
storage my_store = (mode==IO_BINARY)?binary_store:text_store;
|
|
|
|
|
storage * store = &my_store;
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
sprintf(path, "%s/%s", datapath(), filename);
|
2005-12-18 22:53:12 +01:00
|
|
|
|
#ifdef HAVE_UNISTD_H
|
2007-06-26 11:32:28 +02:00
|
|
|
|
if (access(path, R_OK) == 0) {
|
2005-12-18 22:53:12 +01:00
|
|
|
|
/* make sure we don't overwrite some hardlinkedfile */
|
2007-06-26 11:32:28 +02:00
|
|
|
|
unlink(path);
|
2005-12-18 22:53:12 +01:00
|
|
|
|
}
|
|
|
|
|
#endif
|
2008-04-26 16:55:22 +02:00
|
|
|
|
|
|
|
|
|
store->encoding = enc_gamedata;
|
|
|
|
|
if (store->open(store, path, IO_WRITE)!=0) {
|
2004-04-12 01:56:47 +02:00
|
|
|
|
return -1;
|
2007-08-21 09:04:44 +02:00
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* globale Variablen */
|
|
|
|
|
|
|
|
|
|
base = strrchr(xmlfile, '/');
|
2007-12-18 00:24:48 +01:00
|
|
|
|
if (base) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_str(store, base+1);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_str(store, xmlfile);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_write(store, global.attribs);
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, turn);
|
|
|
|
|
store->w_int(store, 0/*max_unique_id*/);
|
|
|
|
|
store->w_int(store, nextborder);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
/* Write planes */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, listlen(planes));
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
for(pl = planes; pl; pl=pl->next) {
|
|
|
|
|
watcher * w;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, pl->id);
|
|
|
|
|
store->w_str(store, pl->name);
|
|
|
|
|
store->w_int(store, pl->minx);
|
|
|
|
|
store->w_int(store, pl->maxx);
|
|
|
|
|
store->w_int(store, pl->miny);
|
|
|
|
|
store->w_int(store, pl->maxy);
|
|
|
|
|
store->w_int(store, pl->flags);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
w = pl->watchers;
|
|
|
|
|
while (w) {
|
|
|
|
|
if (w->faction) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_faction_reference(w->faction, store);
|
|
|
|
|
store->w_int(store, w->mode);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
w = w->next;
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
a_write(store, pl->attribs);
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Write factions */
|
2004-06-27 18:56:01 +02:00
|
|
|
|
#if RELEASE_VERSION>=ALLIANCES_VERSION
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_alliances(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
#endif
|
2004-06-27 18:56:01 +02:00
|
|
|
|
n = listlen(factions);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, n);
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
log_info((1, " - Schreibe %d Parteien...\n", n));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
for (f = factions; f; f = f->next) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writefaction(store, f);
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Write regions */
|
|
|
|
|
|
|
|
|
|
n=listlen(regions);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, n);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
log_info((1, " - Schreibe Regionen: %d \r", n));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next, --n) {
|
|
|
|
|
/* plus leerzeile */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if ((n%1024)==0) { /* das spart extrem Zeit */
|
|
|
|
|
log_info((2, " - Schreibe Regionen: %d \r", n));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
fflush(stdout);
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, r->x);
|
|
|
|
|
store->w_int(store, r->y);
|
|
|
|
|
writeregion(store, r);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, listlen(r->buildings));
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
for (b = r->buildings; b; b = b->next) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_building_reference(b, store);
|
|
|
|
|
store->w_str(store, b->name);
|
|
|
|
|
store->w_str(store, b->display?b->display:"");
|
|
|
|
|
store->w_int(store, b->size);
|
|
|
|
|
store->w_tok(store, b->type->_name);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
a_write(store, b->attribs);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
store->w_int(store, listlen(r->ships));
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
for (sh = r->ships; sh; sh = sh->next) {
|
|
|
|
|
assert(sh->region == r);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_ship_reference(sh, store);
|
|
|
|
|
store->w_str(store, (const char *)sh->name);
|
|
|
|
|
store->w_str(store, sh->display?(const char *)sh->display:"");
|
|
|
|
|
store->w_tok(store, sh->type->name[0]);
|
|
|
|
|
store->w_int(store, sh->size);
|
|
|
|
|
store->w_int(store, sh->damage);
|
|
|
|
|
store->w_int(store, sh->coast);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
a_write(store, sh->attribs);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
store->w_int(store, listlen(r->units));
|
|
|
|
|
store->w_brk(store);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
for (u = r->units; u; u = u->next) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
writeunit(store, u);
|
2004-04-12 01:56:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_borders(store);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
|
|
|
|
store->close(store);
|
|
|
|
|
|
|
|
|
|
log_info((1, "\nOk.\n"));
|
2004-04-12 01:56:47 +02:00
|
|
|
|
return 0;
|
2004-04-12 03:47:03 +02:00
|
|
|
|
}
|
2006-02-10 20:48:43 +01:00
|
|
|
|
|
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_readint(attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2007-07-17 17:33:48 +02:00
|
|
|
|
/* assert(sizeof(int)==sizeof(a->data)); */
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a->data.i = store->r_int(store);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_writeint(const attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, a->data.i);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_readshorts(attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<ATTRIBREAD_VERSION) {
|
|
|
|
|
return a_readint(a, store);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a->data.sa[0] = (short)store->r_int(store);
|
|
|
|
|
a->data.sa[1] = (short)store->r_int(store);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_writeshorts(const attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, a->data.sa[0]);
|
|
|
|
|
store->w_int(store, a->data.sa[1]);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_readchars(attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2006-02-12 00:44:17 +01:00
|
|
|
|
int i;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<ATTRIBREAD_VERSION) {
|
|
|
|
|
return a_readint(a, store);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
2006-02-12 00:44:17 +01:00
|
|
|
|
for (i=0;i!=4;++i) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a->data.ca[i] = (char)store->r_int(store);
|
2006-02-12 00:44:17 +01:00
|
|
|
|
}
|
2006-02-10 20:48:43 +01:00
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_writechars(const attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2006-02-12 10:22:37 +01:00
|
|
|
|
int i;
|
2006-02-12 10:42:52 +01:00
|
|
|
|
|
2006-02-12 00:44:17 +01:00
|
|
|
|
for (i=0;i!=4;++i) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, a->data.ca[i]);
|
2006-02-12 00:44:17 +01:00
|
|
|
|
}
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_readvoid(attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version<ATTRIBREAD_VERSION) {
|
|
|
|
|
return a_readint(a, store);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_writevoid(const attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_readstring(attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a->data.v = store->r_str(store);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
a_writestring(const attrib * a, struct storage * store)
|
2006-02-10 20:48:43 +01:00
|
|
|
|
{
|
|
|
|
|
assert(a->data.v);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_str(store, (const char *)a->data.v);
|
2006-02-10 20:48:43 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
a_finalizestring(attrib * a)
|
|
|
|
|
{
|
|
|
|
|
free(a->data.v);
|
|
|
|
|
}
|