2010-08-08 10:06:34 +02:00
|
|
|
|
/*
|
|
|
|
|
Copyright (c) 1998-2010, Enno Rehling <enno@eressea.de>
|
|
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
|
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
|
#include <kernel/config.h>
|
|
|
|
|
#include "save.h"
|
|
|
|
|
|
|
|
|
|
#include "alchemy.h"
|
|
|
|
|
#include "alliance.h"
|
|
|
|
|
#include "connection.h"
|
|
|
|
|
#include "building.h"
|
|
|
|
|
#include "faction.h"
|
|
|
|
|
#include "group.h"
|
|
|
|
|
#include "item.h"
|
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "message.h"
|
|
|
|
|
#include "move.h"
|
|
|
|
|
#include "objtypes.h"
|
|
|
|
|
#include "order.h"
|
|
|
|
|
#include "pathfinder.h"
|
|
|
|
|
#include "plane.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "resources.h"
|
|
|
|
|
#include "ship.h"
|
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "spell.h"
|
2012-05-24 09:56:54 +02:00
|
|
|
|
#include "spellbook.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include "terrain.h"
|
2011-03-07 08:02:35 +01:00
|
|
|
|
#include "terrainid.h" /* only for conversion code */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include "unit.h"
|
|
|
|
|
#include "version.h"
|
|
|
|
|
|
|
|
|
|
#include "textstore.h"
|
|
|
|
|
#include "binarystore.h"
|
|
|
|
|
|
|
|
|
|
/* attributes includes */
|
|
|
|
|
#include <attributes/key.h>
|
2012-04-08 03:11:58 +02:00
|
|
|
|
#include <triggers/timeout.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
#include <util/base36.h>
|
|
|
|
|
#include <util/bsdstring.h>
|
|
|
|
|
#include <util/event.h>
|
|
|
|
|
#include <util/filereader.h>
|
|
|
|
|
#include <util/goodies.h>
|
|
|
|
|
#include <util/language.h>
|
|
|
|
|
#include <util/lists.h>
|
|
|
|
|
#include <util/log.h>
|
2011-02-27 07:55:27 +01:00
|
|
|
|
#include <util/os.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include <util/parser.h>
|
2012-05-31 04:55:17 +02:00
|
|
|
|
#include <quicklist.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include <util/rand.h>
|
|
|
|
|
#include <util/resolve.h>
|
|
|
|
|
#include <util/rng.h>
|
|
|
|
|
#include <util/sql.h>
|
|
|
|
|
#include <util/storage.h>
|
|
|
|
|
#include <util/umlaut.h>
|
|
|
|
|
#include <util/unicode.h>
|
|
|
|
|
|
|
|
|
|
#include <libxml/encoding.h>
|
|
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <ctype.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
|
|
#define xisdigit(c) (((c) >= '0' && (c) <= '9') || (c) == '-')
|
|
|
|
|
|
|
|
|
|
#define ESCAPE_FIX
|
|
|
|
|
#define MAXORDERS 256
|
|
|
|
|
#define MAXPERSISTENT 128
|
|
|
|
|
|
|
|
|
|
/* exported symbols symbols */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const char *game_name = "eressea";
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int firstx = 0, firsty = 0;
|
|
|
|
|
int enc_gamedata = 0;
|
|
|
|
|
|
|
|
|
|
/* local symbols */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static region *current_region;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
char *rns(FILE * f, char *c, size_t size)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
char *s = c;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
do {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
*s = (char)getc(f);
|
|
|
|
|
} while (*s != '"');
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
for (;;) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
*s = (char)getc(f);
|
|
|
|
|
if (*s == '"')
|
|
|
|
|
break;
|
|
|
|
|
if (s < c + size)
|
|
|
|
|
++s;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
*s = 0;
|
|
|
|
|
return c;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
extern unsigned int __at_hashkey(const char *s);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
FILE *cfopen(const char *filename, const char *mode)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
FILE *F = fopen(filename, mode);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
if (F == 0) {
|
|
|
|
|
perror(filename);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
setvbuf(F, 0, _IOFBF, 32 * 1024); /* 32 kb buffer size */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return F;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int freadstr(FILE * F, int encoding, char *start, size_t size)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
char *str = start;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
boolean quote = false;
|
|
|
|
|
for (;;) {
|
|
|
|
|
int c = fgetc(F);
|
|
|
|
|
|
|
|
|
|
if (isxspace(c)) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (str == start) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (!quote) {
|
|
|
|
|
*str = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
return (int)(str - start);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
switch (c) {
|
2011-03-08 08:44:20 +01:00
|
|
|
|
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) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case EOF:
|
|
|
|
|
return EOF;
|
2011-03-08 08:44:20 +01:00
|
|
|
|
case 'n':
|
|
|
|
|
if ((size_t) (str - start + 1) < size) {
|
|
|
|
|
*str++ = '\n';
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if ((size_t) (str - start + 1) < size) {
|
|
|
|
|
if (encoding == XML_CHAR_ENCODING_8859_1 && c & 0x80) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char inbuf = (char)c;
|
|
|
|
|
size_t inbytes = 1;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
size_t outbytes = size - (str - start);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int ret = unicode_latin1_to_utf8(str, &outbytes, &inbuf, &inbytes);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (ret > 0)
|
|
|
|
|
str += ret;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
else {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("input data was not iso-8859-1! assuming utf-8\n");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
encoding = XML_CHAR_ENCODING_ERROR;
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-03-08 08:44:20 +01:00
|
|
|
|
}
|
|
|
|
|
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 {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("input data was not iso-8859-1! assuming utf-8\n");
|
2011-03-08 08:44:20 +01:00
|
|
|
|
encoding = XML_CHAR_ENCODING_ERROR;
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
*str++ = (char)c;
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** writes a quoted string to the file
|
|
|
|
|
* no trailing space, since this is used to make the creport.
|
|
|
|
|
*/
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int fwritestr(FILE * F, const char *str)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
int nwrite = 0;
|
|
|
|
|
fputc('\"', F);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (str)
|
|
|
|
|
while (*str) {
|
|
|
|
|
int c = (int)(unsigned char)*str++;
|
|
|
|
|
switch (c) {
|
2011-03-08 08:44:20 +01:00
|
|
|
|
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;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
fputc('\"', F);
|
|
|
|
|
return nwrite + 2;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static unit *unitorders(FILE * F, int enc, struct faction *f)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
unit *u;
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (!f)
|
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
i = getid();
|
|
|
|
|
u = findunitg(i, NULL);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (u && u->race == new_race[RC_SPELL])
|
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (u && u->faction == f) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
order **ordp;
|
2010-08-08 10:06:34 +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) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
order *ord = *ordp;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (!is_repeated(ord)) {
|
|
|
|
|
*ordp = ord->next;
|
|
|
|
|
ord->next = NULL;
|
|
|
|
|
free_order(ord);
|
|
|
|
|
} else {
|
|
|
|
|
ordp = &ord->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
free_orders(&u->orders);
|
|
|
|
|
}
|
|
|
|
|
u->orders = 0;
|
|
|
|
|
|
|
|
|
|
ordp = &u->orders;
|
|
|
|
|
|
|
|
|
|
for (;;) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const char *s;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* Erst wenn wir sicher sind, dass kein Befehl
|
2011-03-07 08:02:35 +01:00
|
|
|
|
* eingegeben wurde, checken wir, ob nun eine neue
|
|
|
|
|
* Einheit oder ein neuer Spieler drankommt */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
s = getbuf(F, enc);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (s == NULL)
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2012-05-19 08:57:20 +02:00
|
|
|
|
if (s[0]) {
|
|
|
|
|
if (s[0]!='@') {
|
|
|
|
|
const char *stok = s;
|
|
|
|
|
stok = parse_token(&stok);
|
|
|
|
|
|
|
|
|
|
if (stok) {
|
|
|
|
|
boolean quit = false;
|
|
|
|
|
param_t param = findparam(stok, u->faction->locale);
|
|
|
|
|
switch (param) {
|
|
|
|
|
case P_UNIT:
|
|
|
|
|
case P_REGION:
|
2010-08-08 10:06:34 +02:00
|
|
|
|
quit = true;
|
|
|
|
|
break;
|
2012-05-19 08:57:20 +02:00
|
|
|
|
case P_FACTION:
|
|
|
|
|
case P_NEXT:
|
|
|
|
|
case P_GAMENAME:
|
|
|
|
|
/* these terminate the orders, so we apply extra checking */
|
|
|
|
|
if (strlen(stok) >= 3) {
|
|
|
|
|
quit = true;
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
quit = false;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
/* TODO: syntax error message */
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (quit) {
|
|
|
|
|
break;
|
2011-03-08 08:44:20 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
/* Nun wird der Befehl erzeut und eingeh<65>ngt */
|
|
|
|
|
*ordp = parse_order(s, u->faction->locale);
|
2012-05-19 08:57:20 +02:00
|
|
|
|
if (*ordp) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
ordp = &(*ordp)->next;
|
2012-05-19 08:57:20 +02:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
/* cmistake(?, buf, 160, MSG_EVENT); */
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
return u;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static faction *factionorders(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f = NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int fid = getid();
|
|
|
|
|
|
|
|
|
|
f = findfaction(fid);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
|
|
if (f != NULL && !is_monsters(f)) {
|
|
|
|
|
const char *pass = getstrtoken();
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
if (!checkpasswd(f, (const char *)pass, true)) {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("Invalid password for faction %s\n", itoa36(fid));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
ADDMSG(&f->msgs, msg_message("wrongpasswd", "faction password",
|
2011-03-07 08:02:35 +01:00
|
|
|
|
f->no, pass));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
/* Die Partei hat sich zumindest gemeldet, so da<64> sie noch
|
|
|
|
|
* nicht als unt<EFBFBD>tig gilt */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* TODO: +1 ist ein Workaround, weil cturn erst in process_orders
|
|
|
|
|
* incrementiert wird. */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
f->lastorders = global.data_turn + 1;
|
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
} else {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("orders for invalid faction %s\n", itoa36(fid));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
double version(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
return RELEASE_VERSION * 0.1;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static param_t igetparam(const char *s, const struct locale *lang)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
return findparam(igetstrtoken(s), lang);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int readorders(const char *filename)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
FILE *F = NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
const char *b;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int nfactions = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
struct faction *f = NULL;
|
|
|
|
|
|
2012-05-20 01:54:19 +02:00
|
|
|
|
if (filename) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
F = cfopen(filename, "rb");
|
2012-05-20 01:54:19 +02:00
|
|
|
|
}
|
|
|
|
|
if (!F) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (verbosity >= 1)
|
|
|
|
|
puts(" - lese Befehlsdatei...\n");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
/* TODO: recognize UTF8 BOM */
|
|
|
|
|
b = getbuf(F, enc_gamedata);
|
|
|
|
|
|
|
|
|
|
/* Auffinden der ersten Partei, und danach abarbeiten bis zur letzten
|
|
|
|
|
* Partei */
|
|
|
|
|
|
|
|
|
|
while (b) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const struct locale *lang = f ? f->locale : default_locale;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int p;
|
|
|
|
|
switch (igetparam(b, lang)) {
|
|
|
|
|
#undef LOCALE_CHANGE
|
|
|
|
|
#ifdef LOCALE_CHANGE
|
2012-05-20 09:16:12 +02:00
|
|
|
|
case P_LOCALE:
|
|
|
|
|
{
|
|
|
|
|
const char *s = getstrtoken();
|
|
|
|
|
if (f && find_locale(s)) {
|
|
|
|
|
f->locale = find_locale(s);
|
|
|
|
|
}
|
2011-03-08 08:44:20 +01:00
|
|
|
|
}
|
|
|
|
|
b = getbuf(F, enc_gamedata);
|
|
|
|
|
break;
|
2012-05-20 09:16:12 +02:00
|
|
|
|
#endif
|
2011-03-08 08:44:20 +01:00
|
|
|
|
case P_GAMENAME:
|
|
|
|
|
case P_FACTION:
|
|
|
|
|
f = factionorders();
|
|
|
|
|
if (f) {
|
|
|
|
|
++nfactions;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-08 08:44:20 +01:00
|
|
|
|
b = getbuf(F, enc_gamedata);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-08 08:44:20 +01:00
|
|
|
|
/* in factionorders wird nur eine zeile gelesen:
|
|
|
|
|
* diejenige mit dem passwort. Die befehle der units
|
|
|
|
|
* werden geloescht, und die Partei wird als aktiv
|
|
|
|
|
* vermerkt. */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-08 08:44:20 +01:00
|
|
|
|
case P_UNIT:
|
|
|
|
|
if (!f || !unitorders(F, enc_gamedata, f))
|
|
|
|
|
do {
|
|
|
|
|
b = getbuf(F, enc_gamedata);
|
|
|
|
|
if (!b)
|
|
|
|
|
break;
|
2012-05-16 21:06:44 +02:00
|
|
|
|
p = (b[0]=='@') ? NOPARAM : igetparam(b, lang);
|
2011-03-08 08:44:20 +01:00
|
|
|
|
} while ((p != P_UNIT || !f) && p != P_FACTION && p != P_NEXT
|
|
|
|
|
&& p != P_GAMENAME);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-08 08:44:20 +01:00
|
|
|
|
/* 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>. */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-08 08:44:20 +01:00
|
|
|
|
case P_NEXT:
|
|
|
|
|
f = NULL;
|
|
|
|
|
b = getbuf(F, enc_gamedata);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-08 08:44:20 +01:00
|
|
|
|
default:
|
|
|
|
|
b = getbuf(F, enc_gamedata);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fclose(F);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (verbosity >= 1)
|
|
|
|
|
puts("\n");
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " %d Befehlsdateien gelesen\n", nfactions);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
/* #define INNER_WORLD */
|
|
|
|
|
/* f<>rs debuggen nur den inneren Teil der Welt laden */
|
|
|
|
|
/* -9;-27;-1;-19;Sumpfloch */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int inner_world(region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static int xy[2] = { 18, -45 };
|
|
|
|
|
static int size[2] = { 27, 27 };
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (r->x >= xy[0] && r->x < xy[0] + size[0] && r->y >= xy[1]
|
|
|
|
|
&& r->y < xy[1] + size[1])
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return 2;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
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)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return 1;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int maxregions = -1;
|
|
|
|
|
int loadplane = 0;
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
U_MAN,
|
|
|
|
|
U_UNDEAD,
|
|
|
|
|
U_ILLUSION,
|
|
|
|
|
U_FIREDRAGON,
|
|
|
|
|
U_DRAGON,
|
|
|
|
|
U_WYRM,
|
|
|
|
|
U_SPELL,
|
|
|
|
|
U_TAVERNE,
|
|
|
|
|
U_MONSTER,
|
|
|
|
|
U_BIRTHDAYDRAGON,
|
|
|
|
|
U_TREEMAN,
|
|
|
|
|
MAXTYPES
|
|
|
|
|
};
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
race_t typus2race(unsigned char typus)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (typus > 0 && typus <= 11)
|
|
|
|
|
return (race_t) (typus - 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return NORACE;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void create_backup(char *file)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
#ifdef HAVE_LINK
|
|
|
|
|
char bfile[MAX_PATH];
|
|
|
|
|
int c = 1;
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (access(file, R_OK) == 0)
|
|
|
|
|
return;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
do {
|
|
|
|
|
sprintf(bfile, "%s.backup%d", file, c);
|
|
|
|
|
c++;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} while (access(bfile, R_OK) == 0);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
link(file, bfile);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void read_items(struct storage *store, item ** ilist)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
for (;;) {
|
|
|
|
|
char ibuf[32];
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const item_type *itype;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int i;
|
|
|
|
|
store->r_str_buf(store, ibuf, sizeof(ibuf));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (!strcmp("end", ibuf))
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
itype = it_find(ibuf);
|
|
|
|
|
i = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (i <= 0) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("data contains an entry with %d %s\n", i, itype->rtype->_name[1]);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
} else {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(itype != NULL);
|
|
|
|
|
if (itype != NULL) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
i_change(ilist, itype, i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void read_alliances(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
char pbuf[8];
|
|
|
|
|
int id, terminator = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < SAVEALLIANCE_VERSION) {
|
|
|
|
|
if (!AllianceRestricted() && !AllianceAuto())
|
|
|
|
|
return;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < ALLIANCELEADER_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
terminator = atoi36("end");
|
|
|
|
|
store->r_str_buf(store, pbuf, sizeof(pbuf));
|
|
|
|
|
id = atoi36(pbuf);
|
|
|
|
|
} else {
|
|
|
|
|
id = store->r_id(store);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (id != terminator) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char aname[128];
|
2011-03-07 08:02:35 +01:00
|
|
|
|
alliance *al;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, aname, sizeof(aname));
|
|
|
|
|
al = makealliance(id, aname);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= OWNER_2_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
al->flags = store->r_int(store);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= ALLIANCELEADER_VERSION) {
|
|
|
|
|
read_reference(&al->_leader, store, read_faction_reference,
|
|
|
|
|
resolve_faction);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
id = store->r_id(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, pbuf, sizeof(pbuf));
|
|
|
|
|
id = atoi36(pbuf);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void write_alliances(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
alliance *al = alliances;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
while (al) {
|
|
|
|
|
if (al->_leader) {
|
|
|
|
|
store->w_id(store, al->id);
|
|
|
|
|
store->w_str(store, al->name);
|
|
|
|
|
store->w_int(store, (int)al->flags);
|
|
|
|
|
write_faction_reference(al->_leader, store);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
al = al->next;
|
|
|
|
|
}
|
|
|
|
|
store->w_id(store, 0);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void write_items(struct storage *store, item * ilist)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
item *itm;
|
|
|
|
|
for (itm = ilist; itm; itm = itm->next) {
|
|
|
|
|
assert(itm->number >= 0);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (itm->number) {
|
|
|
|
|
store->w_tok(store, resourcename(itm->type->rtype, 0));
|
|
|
|
|
store->w_int(store, itm->number);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static int resolve_owner(variant id, void *address)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
region_owner *owner = (region_owner *) address;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int result = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f = NULL;
|
|
|
|
|
if (id.i != 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f = findfaction(id.i);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f == NULL) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("region has an invalid owner (%s)\n", itoa36(id.i));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f = get_monsters();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
owner->owner = f;
|
|
|
|
|
if (f) {
|
|
|
|
|
owner->alliance = f->alliance;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void read_owner(struct storage *store, region_owner ** powner)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
int since_turn = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (since_turn >= 0) {
|
|
|
|
|
region_owner *owner = malloc(sizeof(region_owner));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
owner->since_turn = since_turn;
|
|
|
|
|
owner->morale_turn = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= MOURNING_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
owner->flags = store->r_int(store);
|
|
|
|
|
} else {
|
|
|
|
|
owner->flags = 0;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= OWNER_2_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int id = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
owner->alliance = id ? findalliance(id) : NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
} else {
|
|
|
|
|
owner->alliance = NULL;
|
|
|
|
|
}
|
|
|
|
|
read_reference(owner, store, &read_faction_reference, &resolve_owner);
|
|
|
|
|
*powner = owner;
|
|
|
|
|
} else {
|
|
|
|
|
*powner = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void write_owner(struct storage *store, region_owner * owner)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
if (owner) {
|
|
|
|
|
store->w_int(store, owner->since_turn);
|
|
|
|
|
store->w_int(store, owner->morale_turn);
|
|
|
|
|
store->w_int(store, owner->flags);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_id(store, owner->alliance ? owner->alliance->id : 0);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
write_faction_reference(owner->owner, store);
|
|
|
|
|
} else {
|
|
|
|
|
store->w_int(store, -1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int current_turn(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
char zText[MAX_PATH];
|
|
|
|
|
int cturn = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
FILE *f;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
sprintf(zText, "%s/turn", basepath());
|
|
|
|
|
f = cfopen(zText, "r");
|
|
|
|
|
if (f) {
|
|
|
|
|
fscanf(f, "%d\n", &cturn);
|
|
|
|
|
fclose(f);
|
|
|
|
|
}
|
|
|
|
|
return cturn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2011-03-07 08:02:35 +01:00
|
|
|
|
writeorder(struct storage *store, const struct order *ord,
|
|
|
|
|
const struct locale *lang)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
char obuf[1024];
|
|
|
|
|
write_order(ord, obuf, sizeof(obuf));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (obuf[0])
|
|
|
|
|
store->w_str(store, obuf);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
unit *read_unit(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
skill_t sk;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
unit *u;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int number, n, p;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
order **orderp;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char obuf[1024];
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char rname[32];
|
|
|
|
|
|
|
|
|
|
n = store->r_id(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (n <= 0)
|
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
u = findunit(n);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (u == NULL) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
u = calloc(sizeof(unit), 1);
|
|
|
|
|
u->no = n;
|
|
|
|
|
uhash(u);
|
|
|
|
|
} else {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (u->attribs)
|
|
|
|
|
a_remove(&u->attribs, u->attribs);
|
|
|
|
|
while (u->items)
|
|
|
|
|
i_free(i_remove(&u->items, u->items));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
free(u->skills);
|
|
|
|
|
u->skills = 0;
|
|
|
|
|
u->skill_size = 0;
|
|
|
|
|
u_setfaction(u, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = store->r_id(store);
|
|
|
|
|
f = findfaction(n);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f != u->faction)
|
|
|
|
|
u_setfaction(u, f);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
u->name = store->r_str(store);
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
|
|
|
|
} else {
|
|
|
|
|
u->display = store->r_str(store);
|
|
|
|
|
}
|
|
|
|
|
number = store->r_int(store);
|
|
|
|
|
u->age = (short)store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
|
|
if (store->version < STORAGE_VERSION) {
|
|
|
|
|
char *space;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, rname, sizeof(rname));
|
|
|
|
|
space = strchr(rname, ' ');
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (space != NULL) {
|
|
|
|
|
char *inc = space + 1;
|
|
|
|
|
char *outc = space;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
do {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (*inc == ' ')
|
|
|
|
|
++inc;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
while (*inc) {
|
|
|
|
|
*outc++ = *inc++;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (*inc == ' ')
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
} while (*inc);
|
|
|
|
|
*outc = 0;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
store->r_tok_buf(store, rname, sizeof(rname));
|
|
|
|
|
}
|
|
|
|
|
u->race = rc_find(rname);
|
|
|
|
|
assert(u->race);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < STORAGE_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, rname, sizeof(rname));
|
|
|
|
|
} else {
|
|
|
|
|
store->r_tok_buf(store, rname, sizeof(rname));
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (rname[0] && skill_enabled[SK_STEALTH])
|
|
|
|
|
u->irace = rc_find(rname);
|
|
|
|
|
else
|
|
|
|
|
u->irace = NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
if (u->race->describe) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const char *rcdisp = u->race->describe(u, u->faction->locale);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (u->display && rcdisp) {
|
|
|
|
|
/* see if the data file contains old descriptions */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp(rcdisp, u->display) == 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
free(u->display);
|
|
|
|
|
u->display = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (u->faction == NULL) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("unit %s has faction == NULL\n", unitname(u));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
u_setfaction(u, get_monsters());
|
|
|
|
|
set_number(u, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (count_unit(u) && u->faction)
|
|
|
|
|
u->faction->no_units++;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
set_number(u, number);
|
|
|
|
|
|
|
|
|
|
n = store->r_id(store);
|
2012-05-17 21:23:25 +02:00
|
|
|
|
if (n > 0) {
|
|
|
|
|
building * b = findbuilding(n);
|
|
|
|
|
if (b) {
|
|
|
|
|
u_set_building(u, b);
|
2012-05-19 07:34:04 +02:00
|
|
|
|
if (fval(u, UFL_OWNER)) {
|
2012-05-26 19:43:05 +02:00
|
|
|
|
building_set_owner(u);
|
2012-05-19 07:34:04 +02:00
|
|
|
|
}
|
2012-05-17 21:23:25 +02:00
|
|
|
|
} else {
|
|
|
|
|
log_error("read_unit: unit in unkown building '%s'\n", itoa36(n));
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
n = store->r_id(store);
|
2012-05-17 09:13:30 +02:00
|
|
|
|
if (n > 0) {
|
|
|
|
|
ship * sh = findship(n);
|
|
|
|
|
if (sh) {
|
|
|
|
|
u_set_ship(u, sh);
|
2012-05-19 07:34:04 +02:00
|
|
|
|
if (fval(u, UFL_OWNER)) {
|
2012-05-26 19:43:05 +02:00
|
|
|
|
ship_set_owner(u);
|
2012-05-19 07:34:04 +02:00
|
|
|
|
}
|
2012-05-17 21:23:25 +02:00
|
|
|
|
} else {
|
|
|
|
|
log_error("read_unit: unit in unkown ship '%s'\n", itoa36(n));
|
2012-05-17 09:13:30 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-17 21:23:25 +02:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
setstatus(u, store->r_int(store));
|
|
|
|
|
u->flags = store->r_int(store);
|
|
|
|
|
u->flags &= UFL_SAVEMASK;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if ((u->flags & UFL_ANON_FACTION) && !rule_stealth_faction()) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* if this rule is broken, then fix broken units */
|
|
|
|
|
u->flags -= UFL_ANON_FACTION;
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("%s was anonymous.\n", unitname(u));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
/* Persistente Befehle einlesen */
|
|
|
|
|
free_orders(&u->orders);
|
|
|
|
|
store->r_str_buf(store, obuf, sizeof(obuf));
|
|
|
|
|
p = n = 0;
|
|
|
|
|
orderp = &u->orders;
|
|
|
|
|
while (obuf[0]) {
|
|
|
|
|
if (!lomem) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
order *ord = parse_order(obuf, u->faction->locale);
|
|
|
|
|
if (ord != NULL) {
|
|
|
|
|
if (++n < MAXORDERS) {
|
|
|
|
|
if (!is_persistent(ord) || ++p < MAXPERSISTENT) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
*orderp = ord;
|
|
|
|
|
orderp = &ord->next;
|
|
|
|
|
ord = NULL;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else if (p == MAXPERSISTENT) {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("%s had %d or more persistent orders\n", unitname(u), MAXPERSISTENT);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else if (n == MAXORDERS) {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("%s had %d or more orders\n", unitname(u), MAXORDERS);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (ord != NULL)
|
|
|
|
|
free_order(ord);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
store->r_str_buf(store, obuf, sizeof(obuf));
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < NOLASTORDER_VERSION) {
|
|
|
|
|
order *ord;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, obuf, sizeof(obuf));
|
|
|
|
|
ord = parse_order(obuf, u->faction->locale);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (ord != NULL) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
addlist(&u->orders, ord);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
set_order(&u->thisorder, NULL);
|
|
|
|
|
|
|
|
|
|
assert(u->race);
|
|
|
|
|
while ((sk = (skill_t) store->r_int(store)) != NOSKILL) {
|
|
|
|
|
int level = store->r_int(store);
|
|
|
|
|
int weeks = store->r_int(store);
|
|
|
|
|
if (level) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
skill *sv = add_skill(u, sk);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
sv->level = sv->old = (unsigned char)level;
|
|
|
|
|
sv->weeks = (unsigned char)weeks;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
read_items(store, &u->items);
|
|
|
|
|
u->hp = store->r_int(store);
|
|
|
|
|
if (u->hp < u->number) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("Einheit %s hat %u Personen, und %u Trefferpunkte\n", itoa36(u->no), u->number, u->hp);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
u->hp = u->number;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
a_read(store, &u->attribs, u);
|
|
|
|
|
return u;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void write_unit(struct storage *store, const unit * u)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
order *ord;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int i, p = 0;
|
2012-05-19 07:34:04 +02:00
|
|
|
|
unsigned int flags = u->flags & UFL_SAVEMASK;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const race *irace = u_irace(u);
|
2012-05-19 07:34:04 +02:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
write_unit_reference(u, store);
|
|
|
|
|
write_faction_reference(u->faction, store);
|
|
|
|
|
store->w_str(store, (const char *)u->name);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_str(store, u->display ? (const char *)u->display : "");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, u->number);
|
|
|
|
|
store->w_int(store, u->age);
|
|
|
|
|
store->w_tok(store, u->race->_name[0]);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_tok(store, (irace && irace != u->race) ? irace->_name[0] : "");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
write_building_reference(u->building, store);
|
|
|
|
|
write_ship_reference(u->ship, store);
|
|
|
|
|
store->w_int(store, u->status);
|
2012-05-19 07:34:04 +02:00
|
|
|
|
if (u->building && u==building_owner(u->building)) flags |= UFL_OWNER;
|
|
|
|
|
if (u->ship && u==ship_owner(u->ship)) flags |= UFL_OWNER;
|
|
|
|
|
store->w_int(store, flags);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_brk(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
for (ord = u->old_orders; ord; ord = ord->next) {
|
|
|
|
|
if (++p < MAXPERSISTENT) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
writeorder(store, ord, u->faction->locale);
|
|
|
|
|
} else {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("%s had %d or more persistent orders\n", unitname(u), MAXPERSISTENT);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
for (ord = u->orders; ord; ord = ord->next) {
|
|
|
|
|
if (u->old_orders && is_repeated(ord))
|
|
|
|
|
continue; /* has new defaults */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (is_persistent(ord)) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (++p < MAXPERSISTENT) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
writeorder(store, ord, u->faction->locale);
|
|
|
|
|
} else {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("%s had %d or more persistent orders\n", unitname(u), MAXPERSISTENT);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* write an empty string to terminate the list */
|
|
|
|
|
store->w_str(store, "");
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
|
|
|
|
assert(u->race);
|
|
|
|
|
|
2011-03-07 08:02:35 +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) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, sv->id);
|
|
|
|
|
store->w_int(store, sv->level);
|
|
|
|
|
store->w_int(store, sv->weeks);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
store->w_int(store, -1);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_items(store, u->items);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
if (u->hp == 0) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("unit %s has 0 hitpoints, adjusting.\n", itoa36(u->no));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
((unit *) u)->hp = u->number;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
store->w_int(store, u->hp);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
a_write(store, u->attribs, u);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static region *readregion(struct storage *store, int x, int y)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
region *r = findregion(x, y);
|
|
|
|
|
const terrain_type *terrain;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char token[32];
|
|
|
|
|
unsigned int uid = 0;
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= UID_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
uid = store->r_int(store);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (r == NULL) {
|
|
|
|
|
plane *pl = findplane(x, y);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
r = new_region(x, y, pl, uid);
|
|
|
|
|
} else {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(uid == 0 || r->uid == uid);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
current_region = r;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (r->attribs)
|
|
|
|
|
a_remove(&r->attribs, r->attribs);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (r->land) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
free(r->land); /* mem leak */
|
|
|
|
|
r->land->demands = 0; /* mem leak */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
while (r->resources) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
rawmaterial *rm = r->resources;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
r->resources = rm->next;
|
|
|
|
|
free(rm);
|
|
|
|
|
}
|
|
|
|
|
r->land = 0;
|
|
|
|
|
}
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
|
|
|
|
} else {
|
|
|
|
|
char info[DISPLAYSIZE];
|
|
|
|
|
store->r_str_buf(store, info, sizeof(info));
|
|
|
|
|
region_setinfo(r, info);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (store->version < TERRAIN_VERSION) {
|
|
|
|
|
int ter = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
terrain = newterrain((terrain_t) ter);
|
|
|
|
|
if (terrain == NULL) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("while reading datafile from pre-TERRAIN_VERSION, could not find terrain #%d.\n", ter);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
terrain = newterrain(T_PLAIN);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
char name[64];
|
|
|
|
|
store->r_str_buf(store, name, sizeof(name));
|
|
|
|
|
terrain = get_terrain(name);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (terrain == NULL) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("Unknown terrain '%s'\n", name);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
assert(!"unknown terrain");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
r->terrain = terrain;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
r->flags = (char)store->r_int(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
r->age = (unsigned short)store->r_int(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
if (fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
r->land = calloc(1, sizeof(land_region));
|
|
|
|
|
r->land->name = store->r_str(store);
|
|
|
|
|
}
|
|
|
|
|
if (r->land) {
|
|
|
|
|
int i;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
rawmaterial **pres = &r->resources;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
i = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (i < 0) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("number of trees in %s is %d.\n", regionname(r, NULL), i);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
i = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 0, i);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
i = store->r_int(store);
|
|
|
|
|
if (i < 0) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("number of young trees in %s is %d.\n", regionname(r, NULL), i);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
i = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 1, i);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
i = store->r_int(store);
|
|
|
|
|
if (i < 0) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("number of seeds in %s is %d.\n", regionname(r, NULL), i);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
i = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 2, i);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
i = store->r_int(store);
|
|
|
|
|
rsethorses(r, i);
|
|
|
|
|
assert(*pres == NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
for (;;) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
rawmaterial *res;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp(token, "end") == 0)
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
res = malloc(sizeof(rawmaterial));
|
|
|
|
|
res->type = rmt_find(token);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (res->type == NULL) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("invalid resourcetype %s in data.\n", token);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(res->type != NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
res->level = store->r_int(store);
|
|
|
|
|
res->amount = store->r_int(store);
|
|
|
|
|
res->flags = 0;
|
|
|
|
|
|
|
|
|
|
res->startlevel = store->r_int(store);
|
|
|
|
|
res->base = store->r_int(store);
|
|
|
|
|
res->divisor = store->r_int(store);
|
|
|
|
|
|
|
|
|
|
*pres = res;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
pres = &res->next;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
*pres = NULL;
|
|
|
|
|
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
|
|
|
|
if (strcmp(token, "noherb") != 0) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const resource_type *rtype = rt_find(token);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
assert(rtype && rtype->itype && fval(rtype->itype, ITF_HERB));
|
|
|
|
|
rsetherbtype(r, rtype->itype);
|
|
|
|
|
} else {
|
|
|
|
|
rsetherbtype(r, NULL);
|
|
|
|
|
}
|
|
|
|
|
rsetherbs(r, (short)store->r_int(store));
|
|
|
|
|
rsetpeasants(r, store->r_int(store));
|
|
|
|
|
rsetmoney(r, store->r_int(store));
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(r->terrain != NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
assert(rhorses(r) >= 0);
|
|
|
|
|
assert(rpeasants(r) >= 0);
|
|
|
|
|
assert(rmoney(r) >= 0);
|
|
|
|
|
|
|
|
|
|
if (r->land) {
|
|
|
|
|
for (;;) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const struct item_type *itype;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (!strcmp(token, "end"))
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
itype = it_find(token);
|
|
|
|
|
assert(itype->rtype->ltype);
|
|
|
|
|
r_setdemand(r, itype->rtype->ltype, store->r_int(store));
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= REGIONITEMS_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
read_items(store, &r->land->items);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= REGIONOWNER_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
r->land->morale = (short)store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (r->land->morale < 0)
|
|
|
|
|
r->land->morale = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
read_owner(store, &r->land->ownership);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
a_read(store, &r->attribs, r);
|
|
|
|
|
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void writeregion(struct storage *store, const region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
store->w_int(store, r->uid);
|
|
|
|
|
store->w_str(store, region_getinfo(r));
|
|
|
|
|
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);
|
|
|
|
|
if (fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
const item_type *rht;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
struct demand *demand;
|
|
|
|
|
rawmaterial *res = r->resources;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_str(store, (const char *)r->land->name);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(rtrees(r, 0) >= 0);
|
|
|
|
|
assert(rtrees(r, 1) >= 0);
|
|
|
|
|
assert(rtrees(r, 2) >= 0);
|
|
|
|
|
store->w_int(store, rtrees(r, 0));
|
|
|
|
|
store->w_int(store, rtrees(r, 1));
|
|
|
|
|
store->w_int(store, rtrees(r, 2));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, rhorses(r));
|
|
|
|
|
|
|
|
|
|
while (res) {
|
|
|
|
|
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);
|
|
|
|
|
res = res->next;
|
|
|
|
|
}
|
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
|
|
|
|
|
rht = rherbtype(r);
|
|
|
|
|
if (rht) {
|
|
|
|
|
store->w_tok(store, resourcename(rht->rtype, 0));
|
|
|
|
|
} else {
|
|
|
|
|
store->w_tok(store, "noherb");
|
|
|
|
|
}
|
|
|
|
|
store->w_int(store, rherbs(r));
|
|
|
|
|
store->w_int(store, rpeasants(r));
|
|
|
|
|
store->w_int(store, rmoney(r));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (r->land)
|
|
|
|
|
for (demand = r->land->demands; demand; demand = demand->next) {
|
|
|
|
|
store->w_tok(store, resourcename(demand->type->itype->rtype, 0));
|
|
|
|
|
store->w_int(store, demand->value);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
#if RELEASE_VERSION>=REGIONITEMS_VERSION
|
|
|
|
|
write_items(store, r->land->items);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
#endif
|
|
|
|
|
#if RELEASE_VERSION>=REGIONOWNER_VERSION
|
|
|
|
|
store->w_int(store, r->land->morale);
|
|
|
|
|
write_owner(store, r->land->ownership);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
a_write(store, r->attribs, r);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static ally **addally(const faction * f, ally ** sfp, int aid, int state)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
struct faction *af = findfaction(aid);
|
|
|
|
|
ally *sf;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
state &= ~HELP_OBSERVE;
|
|
|
|
|
#ifndef REGIONOWNERS
|
|
|
|
|
state &= ~HELP_TRAVEL;
|
|
|
|
|
#endif
|
|
|
|
|
state &= HelpMask();
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (state == 0)
|
|
|
|
|
return sfp;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
sf = calloc(1, sizeof(ally));
|
|
|
|
|
sf->faction = af;
|
|
|
|
|
if (!sf->faction) {
|
|
|
|
|
variant id;
|
|
|
|
|
id.i = aid;
|
|
|
|
|
ur_add(id, &sf->faction, resolve_faction);
|
|
|
|
|
}
|
|
|
|
|
sf->status = state & HELP_ALL;
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (*sfp)
|
|
|
|
|
sfp = &(*sfp)->next;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
*sfp = sf;
|
|
|
|
|
return &sf->next;
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-26 23:19:35 +02:00
|
|
|
|
int get_spell_level_faction(const spell * sp, void * cbdata)
|
2012-05-24 09:56:54 +02:00
|
|
|
|
{
|
2012-05-26 23:19:35 +02:00
|
|
|
|
static spellbook * common = 0;
|
|
|
|
|
spellbook * book;
|
|
|
|
|
faction * f = (faction *)cbdata;
|
|
|
|
|
spellbook_entry * sbe;
|
|
|
|
|
|
|
|
|
|
book = get_spellbook(magic_school[f->magiegebiet]);
|
|
|
|
|
if (book) {
|
|
|
|
|
sbe = spellbook_get(book, sp);
|
|
|
|
|
if (sbe) return sbe->level;
|
|
|
|
|
}
|
|
|
|
|
if (!common) {
|
|
|
|
|
common = get_spellbook(magic_school[M_COMMON]);
|
|
|
|
|
}
|
|
|
|
|
sbe = spellbook_get(common, sp);
|
|
|
|
|
if (sbe) {
|
|
|
|
|
return sbe->level;
|
|
|
|
|
} else {
|
|
|
|
|
log_error("read_spellbook: faction '%s' has a spell with unknown level: '%s'", factionname(f), sp->sname);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void read_spellbook(spellbook **bookp, struct storage *store, int (*get_level)(const spell * sp, void *), void * cbdata)
|
|
|
|
|
{
|
|
|
|
|
for (;;) {
|
2012-05-26 18:36:32 +02:00
|
|
|
|
spell *sp = 0;
|
2012-05-24 09:56:54 +02:00
|
|
|
|
char spname[64];
|
2012-05-26 23:19:35 +02:00
|
|
|
|
int level = 0;
|
2012-05-24 09:56:54 +02:00
|
|
|
|
|
|
|
|
|
if (store->version < SPELLNAME_VERSION) {
|
|
|
|
|
int i = store->r_int(store);
|
|
|
|
|
if (i < 0)
|
|
|
|
|
break;
|
2012-05-26 18:36:32 +02:00
|
|
|
|
if (bookp) {
|
|
|
|
|
sp = find_spellbyid((unsigned int) i);
|
|
|
|
|
}
|
2012-05-24 09:56:54 +02:00
|
|
|
|
} else {
|
|
|
|
|
store->r_tok_buf(store, spname, sizeof(spname));
|
|
|
|
|
if (strcmp(spname, "end") == 0)
|
|
|
|
|
break;
|
2012-05-26 18:36:32 +02:00
|
|
|
|
if (bookp) {
|
|
|
|
|
sp = find_spell(spname);
|
|
|
|
|
if (!sp) {
|
|
|
|
|
log_error("read_spells: could not find spell '%s'\n", spname);
|
|
|
|
|
}
|
2012-05-24 09:56:54 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (store->version >= SPELLBOOK_VERSION) {
|
|
|
|
|
level = store->r_int(store);
|
|
|
|
|
}
|
|
|
|
|
if (sp) {
|
2012-05-27 00:15:18 +02:00
|
|
|
|
spellbook * sb = *bookp;
|
2012-05-26 23:19:35 +02:00
|
|
|
|
if (level<=0 && get_level) {
|
|
|
|
|
level = get_level(sp, cbdata);
|
|
|
|
|
}
|
2012-05-27 00:15:18 +02:00
|
|
|
|
if (!sb) {
|
2012-05-26 18:36:32 +02:00
|
|
|
|
*bookp = create_spellbook(0);
|
2012-05-27 00:15:18 +02:00
|
|
|
|
sb = *bookp;
|
|
|
|
|
}
|
|
|
|
|
if (store->version>=SPELLBOOK_VERSION || !spellbook_get(sb, sp)) {
|
|
|
|
|
spellbook_add(sb, sp, level);
|
2012-05-24 09:56:54 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-25 07:26:05 +02:00
|
|
|
|
void write_spellbook(const struct spellbook *book, struct storage *store)
|
2012-05-24 09:56:54 +02:00
|
|
|
|
{
|
|
|
|
|
quicklist *ql;
|
|
|
|
|
int qi;
|
|
|
|
|
|
|
|
|
|
if (book) {
|
|
|
|
|
for (ql = book->spells, qi = 0; ql; ql_advance(&ql, &qi, 1)) {
|
|
|
|
|
spellbook_entry *sbe = (spellbook_entry *) ql_get(ql, qi);
|
|
|
|
|
store->w_tok(store, sbe->sp->sname);
|
|
|
|
|
store->w_int(store, sbe->level);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-08 10:06:34 +02: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.
|
|
|
|
|
*/
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *readfaction(struct storage * store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
ally **sfp;
|
|
|
|
|
int planes;
|
|
|
|
|
int i = store->r_id(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f = findfaction(i);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char email[128];
|
|
|
|
|
char token[32];
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f == NULL) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f = (faction *) calloc(1, sizeof(faction));
|
|
|
|
|
f->no = i;
|
|
|
|
|
} else {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
f->allies = NULL; /* mem leak */
|
|
|
|
|
while (f->attribs)
|
|
|
|
|
a_remove(&f->attribs, f->attribs);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
f->subscription = store->r_int(store);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (alliances || store->version >= OWNER_2_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int allianceid = store->r_id(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (allianceid > 0)
|
|
|
|
|
f->alliance = findalliance(allianceid);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (f->alliance) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
alliance *al = f->alliance;
|
|
|
|
|
if (al->flags & ALF_NON_ALLIED) {
|
|
|
|
|
assert(!al->members
|
|
|
|
|
|| !"non-allied dummy-alliance has more than one member");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-02-25 16:23:02 +01:00
|
|
|
|
ql_push(&al->members, f);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else if (rule_region_owners()) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* compat fix for non-allied factions */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
alliance *al = makealliance(0, NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
setalliance(f, al);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= OWNER_2_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f->alliance_joindate = store->r_int(store);
|
|
|
|
|
} else {
|
|
|
|
|
f->alliance_joindate = turn - 10; /* we're guessing something safe here */
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f->name = store->r_str(store);
|
|
|
|
|
f->banner = store->r_str(store);
|
|
|
|
|
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Lese Partei %s (%s)\n", f->name, factionid(f));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
store->r_str_buf(store, email, sizeof(email));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (set_email(&f->email, email) != 0) {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("Invalid email address for faction %s: %s\n", itoa36(f->no), email);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
set_email(&f->email, "");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f->passw = store->r_str(store);
|
|
|
|
|
if (store->version >= OVERRIDE_VERSION) {
|
|
|
|
|
f->override = store->r_str(store);
|
|
|
|
|
} else {
|
|
|
|
|
f->override = strdup(itoa36(rng_int()));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
|
|
|
|
f->locale = find_locale(token);
|
|
|
|
|
f->lastorders = store->r_int(store);
|
|
|
|
|
f->age = store->r_int(store);
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
|
|
|
|
f->race = rc_find(token);
|
|
|
|
|
assert(f->race);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
f->magiegebiet = (magic_t) store->r_int(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < FOSS_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* ignore karma */
|
|
|
|
|
store->r_int(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f->flags = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f->no == 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f->flags |= FFL_NPC;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
a_read(store, &f->attribs, f);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= CLAIM_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
read_items(store, &f->items);
|
|
|
|
|
}
|
|
|
|
|
for (;;) {
|
|
|
|
|
store->r_tok_buf(store, token, sizeof(token));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp("end", token) == 0)
|
|
|
|
|
break;
|
2012-05-20 09:16:12 +02:00
|
|
|
|
store->r_int(store); /* there used to be a level here, which is now ignored */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
planes = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (--planes >= 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int id = store->r_int(store);
|
|
|
|
|
int ux = store->r_int(store);
|
|
|
|
|
int uy = store->r_int(store);
|
|
|
|
|
set_ursprung(f, id, ux, uy);
|
|
|
|
|
}
|
|
|
|
|
f->newbies = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
i = f->options = store->r_int(store);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if ((i & (want(O_REPORT) | want(O_COMPUTER))) == 0 && !is_monsters(f)) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* Kein Report eingestellt, Fehler */
|
|
|
|
|
f->options = f->options | want(O_REPORT) | want(O_ZUGVORLAGE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sfp = &f->allies;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < ALLIANCES_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int p = store->r_int(store);
|
|
|
|
|
while (--p >= 0) {
|
|
|
|
|
int aid = store->r_id(store);
|
|
|
|
|
int state = store->r_int(store);
|
|
|
|
|
sfp = addally(f, sfp, aid, state);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for (;;) {
|
|
|
|
|
int aid = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < STORAGE_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->r_tok_buf(store, token, sizeof(token));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp(token, "end") != 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
aid = atoi36(token);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
aid = store->r_id(store);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (aid > 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int state = store->r_int(store);
|
|
|
|
|
sfp = addally(f, sfp, aid, state);
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
read_groups(store, f);
|
2012-05-24 09:56:54 +02:00
|
|
|
|
f->spellbook = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= REGIONOWNER_VERSION) {
|
2012-05-26 23:19:35 +02:00
|
|
|
|
read_spellbook(FactionSpells() ? &f->spellbook : 0, store, get_spell_level_faction, (void *)f);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void writefaction(struct storage *store, const faction * f)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
ally *sf;
|
|
|
|
|
ursprung *ur;
|
|
|
|
|
|
|
|
|
|
write_faction_reference(f, store);
|
|
|
|
|
store->w_int(store, f->subscription);
|
|
|
|
|
if (f->alliance) {
|
|
|
|
|
store->w_id(store, f->alliance->id);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f->alliance->flags & ALF_NON_ALLIED) {
|
|
|
|
|
assert(f == f->alliance->_leader
|
|
|
|
|
|| !"non-allied faction is not leader of its own dummy-alliance.");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_id(store, 0);
|
|
|
|
|
}
|
|
|
|
|
store->w_int(store, f->alliance_joindate);
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_int(store, f->flags & FFL_SAVEMASK);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
a_write(store, f->attribs, f);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_items(store, f->items);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_tok(store, "end");
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, listlen(f->ursprung));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
for (ur = f->ursprung; ur; ur = ur->next) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, ur->id);
|
|
|
|
|
store->w_int(store, ur->x);
|
|
|
|
|
store->w_int(store, ur->y);
|
|
|
|
|
}
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, f->options & ~want(O_DEBUG));
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
|
|
|
|
for (sf = f->allies; sf; sf = sf->next) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int no = (sf->faction != NULL) ? sf->faction->no : 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int status = alliedfaction(NULL, f, sf->faction, HELP_ALL);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (status != 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_id(store, no);
|
|
|
|
|
store->w_int(store, sf->status);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
store->w_id(store, 0);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_groups(store, f->groups);
|
2012-05-24 09:56:54 +02:00
|
|
|
|
write_spellbook(f->spellbook, store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-04-08 03:11:58 +02:00
|
|
|
|
static void repair_unit(unit * u) {
|
|
|
|
|
static const race * rctoad;
|
|
|
|
|
if (!rctoad) rctoad = rc_find("toad");
|
|
|
|
|
|
|
|
|
|
if (u->race==rctoad) {
|
|
|
|
|
int found = 0;
|
|
|
|
|
attrib * a = a_find(u->attribs, &at_eventhandler);
|
|
|
|
|
while (!found && a && a->type==&at_eventhandler) {
|
|
|
|
|
struct trigger ** tlist = get_triggers(a, "timer");
|
|
|
|
|
while (!found && tlist && *tlist) {
|
|
|
|
|
trigger * t = *tlist;
|
|
|
|
|
if (strcmp("changerace", t->type->name)==0) {
|
|
|
|
|
found = 1;
|
|
|
|
|
}
|
|
|
|
|
tlist = &t->next;
|
|
|
|
|
}
|
|
|
|
|
a = a->next;
|
|
|
|
|
}
|
|
|
|
|
if (!found) {
|
|
|
|
|
u->race = u->faction->race;
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("This toad did not have a changerace trigger: %s\n", unitname(u));
|
2012-04-08 03:11:58 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int readgame(const char *filename, int mode, int backup)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
int i, n, p;
|
|
|
|
|
faction *f, **fp;
|
|
|
|
|
region *r;
|
|
|
|
|
building *b, **bp;
|
|
|
|
|
ship **shp;
|
|
|
|
|
unit *u;
|
|
|
|
|
int rmax = maxregions;
|
|
|
|
|
char path[MAX_PATH];
|
|
|
|
|
char token[32];
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const struct building_type *bt_lighthouse = bt_find("lighthouse");
|
|
|
|
|
storage my_store = (mode == IO_BINARY) ? binary_store : text_store;
|
|
|
|
|
storage *store = &my_store;
|
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
sprintf(path, "%s/%s", datapath(), filename);
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "- reading game data from %s\n", filename);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (backup)
|
|
|
|
|
create_backup(path);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
store->encoding = enc_gamedata;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->open(store, path, IO_READ) != 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
enc_gamedata = store->encoding;
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(store->version >= MIN_VERSION || !"unsupported data format");
|
|
|
|
|
assert(store->version <= RELEASE_VERSION || !"unsupported data format");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
if (store->version >= SAVEXMLNAME_VERSION) {
|
|
|
|
|
char basefile[1024];
|
|
|
|
|
|
|
|
|
|
store->r_str_buf(store, basefile, sizeof(basefile));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp(game_name, basefile) != 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
char buffer[64];
|
2012-05-29 21:17:25 +02:00
|
|
|
|
strlcpy(buffer, game_name, sizeof(buffer));
|
|
|
|
|
strlcat(buffer, ".xml", sizeof(buffer));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp(basefile, buffer) != 0) {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("game mismatch: datafile contains %s, game is %s\n", basefile, game_name);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
printf("WARNING: any key to continue, Ctrl-C to stop\n");
|
|
|
|
|
getchar();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
a_read(store, &global.attribs, NULL);
|
|
|
|
|
global.data_turn = turn = store->r_int(store);
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - reading turn %d\n", turn);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
rng_init(turn);
|
|
|
|
|
++global.cookie;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->r_int(store); /* max_unique_id = */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
nextborder = store->r_int(store);
|
|
|
|
|
|
|
|
|
|
/* Planes */
|
|
|
|
|
planes = NULL;
|
|
|
|
|
n = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (--n >= 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int id = store->r_int(store);
|
2011-02-16 08:28:09 +01:00
|
|
|
|
variant fno;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
plane *pl = getplanebyid(id);
|
2011-02-16 08:28:09 +01:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (pl == NULL) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
pl = calloc(1, sizeof(plane));
|
|
|
|
|
} else {
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("the plane with id=%d already exists.\n", id);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
pl->id = id;
|
|
|
|
|
pl->name = store->r_str(store);
|
|
|
|
|
pl->minx = store->r_int(store);
|
|
|
|
|
pl->maxx = store->r_int(store);
|
|
|
|
|
pl->miny = store->r_int(store);
|
|
|
|
|
pl->maxy = store->r_int(store);
|
|
|
|
|
pl->flags = store->r_int(store);
|
|
|
|
|
|
|
|
|
|
/* read watchers */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < FIX_WATCHERS_VERSION) {
|
2011-02-16 08:28:09 +01:00
|
|
|
|
char rname[64];
|
|
|
|
|
/* before this version, watcher storage was pretty broken. we are incompatible and don't read them */
|
|
|
|
|
for (;;) {
|
|
|
|
|
store->r_tok_buf(store, rname, sizeof(rname));
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (strcmp(rname, "end") == 0) {
|
|
|
|
|
break; /* this is most likely the end of the list */
|
2011-02-16 08:28:09 +01:00
|
|
|
|
} else {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
log_error(
|
|
|
|
|
("This datafile contains watchers, but we are unable to read them\n"));
|
2011-02-16 08:28:09 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
fno = read_faction_reference(store);
|
|
|
|
|
while (fno.i) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
watcher *w = (watcher *) malloc(sizeof(watcher));
|
2011-02-16 08:28:09 +01:00
|
|
|
|
ur_add(fno, &w->faction, resolve_faction);
|
|
|
|
|
w->mode = (unsigned char)store->r_int(store);
|
|
|
|
|
w->next = pl->watchers;
|
|
|
|
|
pl->watchers = w;
|
|
|
|
|
fno = read_faction_reference(store);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
a_read(store, &pl->attribs, pl);
|
|
|
|
|
addlist(&planes, pl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Read factions */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= ALLIANCES_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
read_alliances(store);
|
|
|
|
|
}
|
|
|
|
|
n = store->r_int(store);
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Einzulesende Parteien: %d\n", n);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
fp = &factions;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
while (*fp)
|
|
|
|
|
fp = &(*fp)->next;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
while (--n >= 0) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f = readfaction(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
*fp = f;
|
|
|
|
|
fp = &f->next;
|
|
|
|
|
fhash(f);
|
|
|
|
|
}
|
|
|
|
|
*fp = 0;
|
|
|
|
|
|
|
|
|
|
/* Benutzte Faction-Ids */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < STORAGE_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
i = store->r_int(store);
|
|
|
|
|
while (i--) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->r_int(store); /* used faction ids. ignore. */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Regionen */
|
|
|
|
|
|
|
|
|
|
n = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(n < MAXREGIONS);
|
2012-05-26 09:33:55 +02:00
|
|
|
|
if (rmax < 0) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
rmax = n;
|
2012-05-26 09:33:55 +02:00
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Einzulesende Regionen: %d/%d\r", rmax, n);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
while (--n >= 0) {
|
|
|
|
|
unit **up;
|
|
|
|
|
int x = store->r_int(store);
|
|
|
|
|
int y = store->r_int(store);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if ((n & 0x3FF) == 0) { /* das spart extrem Zeit */
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Einzulesende Regionen: %d/%d * %d,%d \r", rmax, n, x, y);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
--rmax;
|
|
|
|
|
|
|
|
|
|
r = readregion(store, x, y);
|
|
|
|
|
|
|
|
|
|
/* Burgen */
|
|
|
|
|
p = store->r_int(store);
|
|
|
|
|
bp = &r->buildings;
|
|
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
|
|
|
|
|
|
|
|
|
b = (building *) calloc(1, sizeof(building));
|
|
|
|
|
b->no = store->r_id(store);
|
|
|
|
|
*bp = b;
|
|
|
|
|
bp = &b->next;
|
|
|
|
|
bhash(b);
|
|
|
|
|
b->name = store->r_str(store);
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
b->display = store->r_str(store);
|
|
|
|
|
}
|
|
|
|
|
b->size = store->r_int(store);
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
|
|
|
|
b->type = bt_find(token);
|
|
|
|
|
b->region = r;
|
|
|
|
|
a_read(store, &b->attribs, b);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (b->type == bt_lighthouse) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
r->flags |= RF_LIGHTHOUSE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* Schiffe */
|
|
|
|
|
|
|
|
|
|
p = store->r_int(store);
|
|
|
|
|
shp = &r->ships;
|
|
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
ship *sh = (ship *) calloc(1, sizeof(ship));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
sh->region = r;
|
|
|
|
|
sh->no = store->r_id(store);
|
|
|
|
|
*shp = sh;
|
|
|
|
|
shp = &sh->next;
|
|
|
|
|
shash(sh);
|
|
|
|
|
sh->name = store->r_str(store);
|
|
|
|
|
if (lomem) {
|
|
|
|
|
store->r_str_buf(store, NULL, 0);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
} else {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
sh->display = store->r_str(store);
|
|
|
|
|
}
|
|
|
|
|
store->r_str_buf(store, token, sizeof(token));
|
|
|
|
|
sh->type = st_find(token);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (sh->type == NULL) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* old datafiles */
|
|
|
|
|
sh->type = st_find((const char *)locale_string(default_locale, token));
|
|
|
|
|
}
|
|
|
|
|
assert(sh->type || !"ship_type not registered!");
|
|
|
|
|
sh->size = store->r_int(store);
|
|
|
|
|
sh->damage = store->r_int(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version >= FOSS_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
sh->flags = store->r_int(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Attribute rekursiv einlesen */
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
sh->coast = (direction_t) store->r_int(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (sh->type->flags & SFL_NOCOAST) {
|
|
|
|
|
sh->coast = NODIRECTION;
|
|
|
|
|
}
|
|
|
|
|
a_read(store, &sh->attribs, sh);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*shp = 0;
|
|
|
|
|
|
|
|
|
|
/* Einheiten */
|
|
|
|
|
|
|
|
|
|
p = store->r_int(store);
|
|
|
|
|
up = &r->units;
|
|
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
unit *u = read_unit(store);
|
|
|
|
|
sc_mage *mage;
|
|
|
|
|
|
2012-04-08 03:11:58 +02:00
|
|
|
|
repair_unit(u);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert(u->region == NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
u->region = r;
|
|
|
|
|
*up = u;
|
|
|
|
|
up = &u->next;
|
|
|
|
|
|
|
|
|
|
update_interval(u->faction, u->region);
|
|
|
|
|
mage = get_mage(u);
|
|
|
|
|
if (mage) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f = u->faction;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
int skl = effskill(u, SK_MAGIC);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (!is_monsters(f) && f->magiegebiet == M_GRAY) {
|
2012-05-17 01:52:01 +02:00
|
|
|
|
log_error("faction %s had magic=gray, fixing (%s)\n", factionname(f), magic_school[mage->magietyp]);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f->magiegebiet = mage->magietyp;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f->max_spelllevel < skl) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f->max_spelllevel = skl;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (mage->spellcount < 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
mage->spellcount = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "\n");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
read_borders(store);
|
|
|
|
|
|
|
|
|
|
store->close(store);
|
|
|
|
|
|
|
|
|
|
/* Unaufgeloeste Zeiger initialisieren */
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "fixing unresolved references.\n");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
resolve();
|
|
|
|
|
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "updating area information for lighthouses.\n");
|
2011-03-07 08:02:35 +01:00
|
|
|
|
for (r = regions; r; r = r->next) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (r->flags & RF_LIGHTHOUSE) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
building *b;
|
|
|
|
|
for (b = r->buildings; b; b = b->next)
|
|
|
|
|
update_lighthouse(b);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "marking factions as alive.\n");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
for (f = factions; f; f = f->next) {
|
|
|
|
|
if (f->flags & FFL_NPC) {
|
|
|
|
|
f->alive = 1;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (f->no == 0) {
|
|
|
|
|
int no = 666;
|
|
|
|
|
while (findfaction(no))
|
|
|
|
|
++no;
|
2012-05-17 00:52:37 +02:00
|
|
|
|
log_warning("renum(monsters, %d)\n", no);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
renumber_faction(f, no);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for (u = f->units; u; u = u->nextF) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (u->number > 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
f->alive = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (loadplane || maxregions >= 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
remove_empty_factions();
|
|
|
|
|
}
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "Done loading turn %d.\n", turn);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void clear_monster_orders(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
faction *f = get_monsters();
|
2010-08-08 10:06:34 +02:00
|
|
|
|
if (f) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
unit *u;
|
|
|
|
|
for (u = f->units; u; u = u->nextF) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
free_orders(&u->orders);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int writegame(const char *filename, int mode)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
char *base;
|
|
|
|
|
int n;
|
|
|
|
|
faction *f;
|
|
|
|
|
region *r;
|
|
|
|
|
building *b;
|
|
|
|
|
ship *sh;
|
|
|
|
|
unit *u;
|
|
|
|
|
plane *pl;
|
|
|
|
|
char path[MAX_PATH];
|
2011-03-07 08:02:35 +01:00
|
|
|
|
storage my_store = (mode == IO_BINARY) ? binary_store : text_store;
|
|
|
|
|
storage *store = &my_store;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->version = RELEASE_VERSION;
|
|
|
|
|
|
|
|
|
|
clear_monster_orders();
|
|
|
|
|
sprintf(path, "%s/%s", datapath(), filename);
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
if (access(path, R_OK) == 0) {
|
|
|
|
|
/* make sure we don't overwrite some hardlinkedfile */
|
|
|
|
|
unlink(path);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
store->encoding = enc_gamedata;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->open(store, path, IO_WRITE) != 0) {
|
2011-02-27 07:55:27 +01:00
|
|
|
|
int err = os_mkdir(datapath(), 0700);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
if (store->open(store, path, IO_WRITE) != 0) {
|
2011-02-27 07:55:27 +01:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* globale Variablen */
|
|
|
|
|
|
|
|
|
|
base = strrchr(game_name, '/');
|
|
|
|
|
if (base) {
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_str(store, base + 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
} else {
|
|
|
|
|
store->w_str(store, game_name);
|
|
|
|
|
}
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
|
|
|
|
a_write(store, global.attribs, NULL);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
|
|
|
|
store->w_int(store, turn);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_int(store, 0 /*max_unique_id */ );
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, nextborder);
|
|
|
|
|
|
|
|
|
|
/* Write planes */
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, listlen(planes));
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
for (pl = planes; pl; pl = pl->next) {
|
|
|
|
|
watcher *w;
|
2010-08-08 10:06:34 +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);
|
|
|
|
|
w = pl->watchers;
|
|
|
|
|
while (w) {
|
|
|
|
|
if (w->faction) {
|
|
|
|
|
write_faction_reference(w->faction, store);
|
|
|
|
|
store->w_int(store, w->mode);
|
|
|
|
|
}
|
|
|
|
|
w = w->next;
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
write_faction_reference(NULL, store); /* mark the end of the list */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
a_write(store, pl->attribs, pl);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Write factions */
|
|
|
|
|
#if RELEASE_VERSION>=ALLIANCES_VERSION
|
|
|
|
|
write_alliances(store);
|
|
|
|
|
#endif
|
|
|
|
|
n = listlen(factions);
|
|
|
|
|
store->w_int(store, n);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Schreibe %d Parteien...\n", n);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
for (f = factions; f; f = f->next) {
|
|
|
|
|
writefaction(store, f);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Write regions */
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
n = listlen(regions);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, n);
|
|
|
|
|
store->w_brk(store);
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Schreibe Regionen: %d \r", n);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next, --n) {
|
|
|
|
|
/* plus leerzeile */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if ((n % 1024) == 0) { /* das spart extrem Zeit */
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, " - Schreibe Regionen: %d \r", n);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
fflush(stdout);
|
|
|
|
|
}
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
store->w_int(store, r->x);
|
|
|
|
|
store->w_int(store, r->y);
|
|
|
|
|
writeregion(store, r);
|
|
|
|
|
|
|
|
|
|
store->w_int(store, listlen(r->buildings));
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
for (b = r->buildings; b; b = b->next) {
|
|
|
|
|
write_building_reference(b, store);
|
|
|
|
|
store->w_str(store, b->name);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_str(store, b->display ? b->display : "");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, b->size);
|
|
|
|
|
store->w_tok(store, b->type->_name);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
a_write(store, b->attribs, b);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
store->w_int(store, listlen(r->ships));
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
for (sh = r->ships; sh; sh = sh->next) {
|
|
|
|
|
assert(sh->region == r);
|
|
|
|
|
write_ship_reference(sh, store);
|
|
|
|
|
store->w_str(store, (const char *)sh->name);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
store->w_str(store, sh->display ? (const char *)sh->display : "");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
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->flags & SFL_SAVEMASK);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
assert((sh->type->flags & SFL_NOCOAST) == 0 || sh->coast == NODIRECTION);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, sh->coast);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
a_write(store, sh->attribs, sh);
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
store->w_int(store, listlen(r->units));
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
write_unit(store, u);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
store->w_brk(store);
|
|
|
|
|
write_borders(store);
|
|
|
|
|
store->w_brk(store);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->close(store);
|
|
|
|
|
|
2012-05-16 07:21:59 +02:00
|
|
|
|
log_printf(stdout, "\nOk.\n");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int a_readint(attrib * a, void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
/* assert(sizeof(int)==sizeof(a->data)); */
|
|
|
|
|
a->data.i = store->r_int(store);
|
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void a_writeint(const attrib * a, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
store->w_int(store, a->data.i);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int a_readshorts(attrib * a, void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < ATTRIBREAD_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return a_readint(a, store, owner);
|
|
|
|
|
}
|
|
|
|
|
a->data.sa[0] = (short)store->r_int(store);
|
|
|
|
|
a->data.sa[1] = (short)store->r_int(store);
|
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void a_writeshorts(const attrib * a, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
store->w_int(store, a->data.sa[0]);
|
|
|
|
|
store->w_int(store, a->data.sa[1]);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int a_readchars(attrib * a, void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
int i;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < ATTRIBREAD_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return a_readint(a, store, owner);
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
for (i = 0; i != 4; ++i) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
a->data.ca[i] = (char)store->r_int(store);
|
|
|
|
|
}
|
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void a_writechars(const attrib * a, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
int i;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
|
|
for (i = 0; i != 4; ++i) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
store->w_int(store, a->data.ca[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int a_readvoid(attrib * a, void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
|
if (store->version < ATTRIBREAD_VERSION) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
return a_readint(a, store, owner);
|
|
|
|
|
}
|
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void a_writevoid(const attrib * a, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int a_readstring(attrib * a, void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
a->data.v = store->r_str(store);
|
|
|
|
|
return AT_READ_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void a_writestring(const attrib * a, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
assert(a->data.v);
|
|
|
|
|
store->w_str(store, (const char *)a->data.v);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void a_finalizestring(attrib * a)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
|
|
|
|
free(a->data.v);
|
|
|
|
|
}
|