2010-08-08 10:06:34 +02:00
|
|
|
#include <platform.h>
|
|
|
|
#include <kernel/config.h>
|
2016-09-10 17:41:44 +02:00
|
|
|
#include <kernel/version.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include "save.h"
|
|
|
|
|
|
|
|
#include "alchemy.h"
|
|
|
|
#include "alliance.h"
|
2012-06-30 20:07:28 +02:00
|
|
|
#include "ally.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
#include "building.h"
|
2018-02-14 20:00:48 +01:00
|
|
|
#include "calendar.h"
|
2017-10-07 03:22:35 +02:00
|
|
|
#include "connection.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
#include "faction.h"
|
|
|
|
#include "group.h"
|
|
|
|
#include "item.h"
|
2017-08-31 21:19:25 +02:00
|
|
|
#include "magic.h"
|
2014-06-09 18:54:48 +02:00
|
|
|
#include "messages.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
#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"
|
2014-12-17 17:22:26 +01:00
|
|
|
#include "lighthouse.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
/* attributes includes */
|
2017-05-24 08:18:55 +02:00
|
|
|
#include <attributes/attributes.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <attributes/key.h>
|
2020-08-03 10:31:58 +02:00
|
|
|
#include <attributes/racename.h>
|
2012-04-08 03:11:58 +02:00
|
|
|
#include <triggers/timeout.h>
|
2017-08-31 21:19:25 +02:00
|
|
|
#include <triggers/shock.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
/* util includes */
|
2017-03-04 20:59:43 +01:00
|
|
|
#include <util/assert.h>
|
2018-09-29 11:37:17 +02:00
|
|
|
#include <kernel/attrib.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/base36.h>
|
2018-09-29 11:37:17 +02:00
|
|
|
#include <kernel/event.h>
|
2018-09-29 13:21:46 +02:00
|
|
|
#include <kernel/gamedata.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/goodies.h>
|
2018-09-29 13:21:46 +02:00
|
|
|
#include <kernel/gamedata.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/language.h>
|
|
|
|
#include <util/lists.h>
|
|
|
|
#include <util/log.h>
|
|
|
|
#include <util/parser.h>
|
2016-01-12 23:52:30 +01:00
|
|
|
#include <util/password.h>
|
2017-12-29 11:44:14 +01:00
|
|
|
#include <util/path.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/rand.h>
|
|
|
|
#include <util/resolve.h>
|
|
|
|
#include <util/rng.h>
|
2017-12-28 18:29:40 +01:00
|
|
|
#include <util/strings.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/umlaut.h>
|
2016-11-11 00:30:49 +01:00
|
|
|
#include <util/unicode.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-01-26 17:41:21 +01:00
|
|
|
#include <selist.h>
|
2014-11-03 22:29:04 +01:00
|
|
|
#include <stream.h>
|
|
|
|
#include <filestream.h>
|
2017-12-29 11:58:39 +01:00
|
|
|
#include <limits.h>
|
2013-12-31 10:06:28 +01:00
|
|
|
#include <storage.h>
|
|
|
|
#include <binarystore.h>
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
/* libc includes */
|
|
|
|
#include <string.h>
|
2013-12-31 10:25:25 +01:00
|
|
|
#include <errno.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#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 */
|
|
|
|
int firstx = 0, firsty = 0;
|
2016-11-13 19:40:38 +01:00
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
static void read_alliances(gamedata *data)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2016-02-13 15:44:15 +01:00
|
|
|
storage *store = data->store;
|
2014-07-23 11:54:51 +02:00
|
|
|
char pbuf[8];
|
|
|
|
int id, terminator = 0;
|
2016-02-13 15:44:15 +01:00
|
|
|
if (data->version < ALLIANCELEADER_VERSION) {
|
2014-07-23 11:54:51 +02:00
|
|
|
terminator = atoi36("end");
|
|
|
|
READ_STR(store, pbuf, sizeof(pbuf));
|
|
|
|
id = atoi36(pbuf);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
READ_INT(store, &id);
|
|
|
|
}
|
|
|
|
while (id != terminator) {
|
|
|
|
char aname[128];
|
|
|
|
alliance *al;
|
|
|
|
READ_STR(store, aname, sizeof(aname));
|
2016-08-17 21:22:50 +02:00
|
|
|
al = new_alliance(id, aname);
|
2016-02-13 15:44:15 +01:00
|
|
|
if (data->version >= OWNER_2_VERSION) {
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(store, &al->flags);
|
|
|
|
}
|
2016-02-13 15:44:15 +01:00
|
|
|
if (data->version >= ALLIANCELEADER_VERSION) {
|
2018-11-01 09:53:23 +01:00
|
|
|
read_faction_reference(data, &al->_leader);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(store, &id);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
READ_STR(store, pbuf, sizeof(pbuf));
|
|
|
|
id = atoi36(pbuf);
|
|
|
|
}
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2016-08-18 18:59:30 +02:00
|
|
|
void read_planes(gamedata *data) {
|
|
|
|
struct storage *store = data->store;
|
|
|
|
int nread;
|
|
|
|
char name[32];
|
|
|
|
|
|
|
|
/* Planes */
|
|
|
|
planes = NULL;
|
|
|
|
READ_INT(store, &nread);
|
|
|
|
while (--nread >= 0) {
|
|
|
|
int id;
|
|
|
|
plane *pl;
|
|
|
|
|
|
|
|
READ_INT(store, &id);
|
|
|
|
pl = getplanebyid(id);
|
|
|
|
|
|
|
|
if (pl == NULL) {
|
|
|
|
pl = calloc(1, sizeof(plane));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (!pl) abort();
|
2016-08-18 18:59:30 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_warning("the plane with id=%d already exists.", id);
|
|
|
|
}
|
|
|
|
pl->id = id;
|
|
|
|
READ_STR(store, name, sizeof(name));
|
2017-12-28 18:29:40 +01:00
|
|
|
pl->name = str_strdup(name);
|
2016-08-18 18:59:30 +02:00
|
|
|
READ_INT(store, &pl->minx);
|
|
|
|
READ_INT(store, &pl->maxx);
|
|
|
|
READ_INT(store, &pl->miny);
|
|
|
|
READ_INT(store, &pl->maxy);
|
|
|
|
READ_INT(store, &pl->flags);
|
|
|
|
|
|
|
|
/* read watchers */
|
|
|
|
if (data->version < FIX_WATCHERS_VERSION) {
|
|
|
|
char rname[64];
|
|
|
|
/* before this version, watcher storage was pretty broken. we are incompatible and don't read them */
|
|
|
|
for (;;) {
|
|
|
|
READ_TOK(store, rname, sizeof(rname));
|
|
|
|
if (strcmp(rname, "end") == 0) {
|
|
|
|
break; /* this is most likely the end of the list */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error(
|
|
|
|
("This datafile contains watchers, but we are unable to read them."));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* WATCHERS - eliminated in February 2016, ca. turn 966 */
|
|
|
|
if (data->version < NOWATCH_VERSION) {
|
2017-09-21 16:26:53 +02:00
|
|
|
int fno;
|
|
|
|
READ_INT(data->store, &fno);
|
2017-09-19 11:42:02 +02:00
|
|
|
while (fno) {
|
2017-09-21 16:26:53 +02:00
|
|
|
READ_INT(data->store, &fno);
|
2016-08-18 18:59:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
read_attribs(data, &pl->attribs, pl);
|
2017-02-18 21:15:14 +01:00
|
|
|
if (pl->id != 1094969858) { /* Regatta */
|
2016-08-18 18:59:30 +02:00
|
|
|
addlist(&planes, pl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void write_planes(storage *store) {
|
|
|
|
plane *pl;
|
|
|
|
/* Write planes */
|
|
|
|
WRITE_INT(store, listlen(planes));
|
|
|
|
for (pl = planes; pl; pl = pl->next) {
|
|
|
|
WRITE_INT(store, pl->id);
|
|
|
|
WRITE_STR(store, pl->name);
|
|
|
|
WRITE_INT(store, pl->minx);
|
|
|
|
WRITE_INT(store, pl->maxx);
|
|
|
|
WRITE_INT(store, pl->miny);
|
|
|
|
WRITE_INT(store, pl->maxy);
|
|
|
|
WRITE_INT(store, pl->flags);
|
|
|
|
#if RELEASE_VERSION < NOWATCH_VERSION
|
|
|
|
write_faction_reference(NULL, store); /* mark the end of pl->watchers (gone since T966) */
|
|
|
|
#endif
|
|
|
|
a_write(store, pl->attribs, pl);
|
|
|
|
WRITE_SECTION(store);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
void write_alliances(gamedata *data)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
alliance *al = alliances;
|
|
|
|
while (al) {
|
|
|
|
if (al->_leader) {
|
|
|
|
WRITE_INT(data->store, al->id);
|
|
|
|
WRITE_STR(data->store, al->name);
|
|
|
|
WRITE_INT(data->store, (int)al->flags);
|
|
|
|
write_faction_reference(al->_leader, data->store);
|
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
}
|
|
|
|
al = al->next;
|
|
|
|
}
|
|
|
|
WRITE_INT(data->store, 0);
|
|
|
|
WRITE_SECTION(data->store);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
static void read_owner(gamedata *data, region_owner ** powner)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
int since_turn;
|
|
|
|
|
|
|
|
READ_INT(data->store, &since_turn);
|
|
|
|
if (since_turn >= 0) {
|
|
|
|
region_owner *owner = malloc(sizeof(region_owner));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (!owner) abort();
|
2014-07-23 11:54:51 +02:00
|
|
|
owner->since_turn = since_turn;
|
|
|
|
READ_INT(data->store, &owner->morale_turn);
|
|
|
|
if (data->version >= MOURNING_VERSION) {
|
|
|
|
READ_INT(data->store, &owner->flags);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
owner->flags = 0;
|
|
|
|
}
|
2015-11-12 16:09:26 +01:00
|
|
|
if (data->version >= OWNER_3_VERSION) {
|
2018-11-01 09:53:23 +01:00
|
|
|
read_faction_reference(data, &owner->last_owner);
|
2016-02-23 08:35:31 +01:00
|
|
|
}
|
|
|
|
else if (data->version >= OWNER_2_VERSION) {
|
2014-07-23 11:54:51 +02:00
|
|
|
int id;
|
2015-11-12 16:09:26 +01:00
|
|
|
alliance *a;
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &id);
|
2015-11-12 16:09:26 +01:00
|
|
|
a = id ? findalliance(id) : NULL;
|
|
|
|
/* don't know which faction, take the leader */
|
2016-02-23 08:35:31 +01:00
|
|
|
owner->last_owner = a ? a->_leader : NULL;
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
else {
|
2015-11-12 16:09:26 +01:00
|
|
|
owner->last_owner = NULL;
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2018-11-01 09:53:23 +01:00
|
|
|
read_faction_reference(data, &owner->owner);
|
2014-07-23 11:54:51 +02:00
|
|
|
*powner = owner;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*powner = 0;
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
static void write_owner(gamedata *data, region_owner * owner)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
if (owner) {
|
|
|
|
WRITE_INT(data->store, owner->since_turn);
|
2016-09-08 19:48:36 +02:00
|
|
|
if (owner->since_turn >= 0) {
|
2018-12-09 03:42:08 +01:00
|
|
|
faction *f = owner->last_owner;
|
2016-09-08 19:48:36 +02:00
|
|
|
WRITE_INT(data->store, owner->morale_turn);
|
|
|
|
WRITE_INT(data->store, owner->flags);
|
|
|
|
write_faction_reference((f && f->_alive) ? f : NULL, data->store);
|
|
|
|
f = owner->owner;
|
|
|
|
write_faction_reference((f && f->_alive) ? f : NULL, data->store);
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
WRITE_INT(data->store, -1);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int current_turn(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-12-29 11:44:14 +01:00
|
|
|
char zText[PATH_MAX];
|
2014-07-23 11:54:51 +02:00
|
|
|
int cturn = 0;
|
|
|
|
FILE *F;
|
|
|
|
|
2017-12-29 11:44:14 +01:00
|
|
|
path_join(basepath(), "turn", zText, sizeof(zText));
|
2014-07-23 11:54:51 +02:00
|
|
|
F = fopen(zText, "r");
|
|
|
|
if (!F) {
|
|
|
|
perror(zText);
|
|
|
|
}
|
|
|
|
else {
|
2017-12-08 21:08:11 +01:00
|
|
|
int c = fscanf(F, "%4d\n", &cturn);
|
2014-07-23 11:54:51 +02:00
|
|
|
fclose(F);
|
2015-11-04 11:45:13 +01:00
|
|
|
if (c != 1) {
|
|
|
|
return -1;
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
return cturn;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-09-15 21:55:57 +02:00
|
|
|
static void writeorder(gamedata *data, const struct order *ord,
|
2016-02-23 08:35:31 +01:00
|
|
|
const struct locale *lang)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
char obuf[1024];
|
2017-10-08 17:07:42 +02:00
|
|
|
write_order(ord, lang, obuf, sizeof(obuf));
|
2014-07-23 11:54:51 +02:00
|
|
|
if (obuf[0])
|
|
|
|
WRITE_STR(data->store, obuf);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2018-02-03 12:57:43 +01:00
|
|
|
static void read_skill(gamedata *data, skill *sv) {
|
|
|
|
int val;
|
|
|
|
READ_INT(data->store, &val);
|
2018-02-03 14:16:01 +01:00
|
|
|
assert(val < MAXSKILLS);
|
2018-02-03 12:57:43 +01:00
|
|
|
sv->id = (skill_t)val;
|
|
|
|
if (sv->id != NOSKILL) {
|
|
|
|
READ_INT(data->store, &val);
|
2018-02-03 14:16:01 +01:00
|
|
|
assert(val < CHAR_MAX);
|
2018-02-03 12:57:43 +01:00
|
|
|
sv->old = sv->level = val;
|
|
|
|
READ_INT(data->store, &val);
|
2018-02-03 14:16:01 +01:00
|
|
|
assert(val < CHAR_MAX);
|
2018-02-03 12:57:43 +01:00
|
|
|
sv->weeks = val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-03 14:16:01 +01:00
|
|
|
static int skill_cmp(const void *a, const void *b) {
|
|
|
|
const skill * sa = (const skill *)a;
|
|
|
|
const skill * sb = (const skill *)b;
|
|
|
|
return sa->id - sb->id;
|
|
|
|
}
|
|
|
|
|
2017-09-15 21:55:57 +02:00
|
|
|
static void read_skills(gamedata *data, unit *u)
|
|
|
|
{
|
2017-09-15 22:03:54 +02:00
|
|
|
if (data->version < SKILLSORT_VERSION) {
|
2018-02-03 12:57:43 +01:00
|
|
|
skill skills[MAXSKILLS], *sv = skills;
|
2018-02-03 14:16:01 +01:00
|
|
|
|
2018-02-03 12:57:43 +01:00
|
|
|
u->skill_size = 0;
|
2017-09-15 22:03:54 +02:00
|
|
|
for (;;) {
|
2018-02-03 12:57:43 +01:00
|
|
|
read_skill(data, sv);
|
|
|
|
if (sv->id == NOSKILL) break;
|
|
|
|
if (sv->level > 0) {
|
|
|
|
++sv;
|
|
|
|
++u->skill_size;
|
2017-09-15 22:03:54 +02:00
|
|
|
}
|
|
|
|
}
|
2018-02-03 14:16:01 +01:00
|
|
|
if (u->skill_size > 0) {
|
|
|
|
size_t sz = u->skill_size * sizeof(skill);
|
|
|
|
|
|
|
|
qsort(skills, u->skill_size, sizeof(skill), skill_cmp);
|
2018-10-22 21:51:11 +02:00
|
|
|
u->skills = (skill *)malloc(sz);
|
2018-02-03 14:16:01 +01:00
|
|
|
memcpy(u->skills, skills, sz);
|
|
|
|
}
|
2017-09-15 22:03:54 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
int i;
|
|
|
|
READ_INT(data->store, &u->skill_size);
|
2018-10-22 21:51:11 +02:00
|
|
|
u->skills = (skill *)malloc(sizeof(skill)*u->skill_size);
|
2017-09-15 22:03:54 +02:00
|
|
|
for (i = 0; i != u->skill_size; ++i) {
|
|
|
|
skill *sv = u->skills + i;
|
2018-02-03 12:57:43 +01:00
|
|
|
read_skill(data, sv);
|
2017-09-15 21:55:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void write_skills(gamedata *data, const unit *u) {
|
|
|
|
int i;
|
2017-09-16 07:37:33 +02:00
|
|
|
skill_t sk = NOSKILL;
|
2017-09-15 22:03:54 +02:00
|
|
|
WRITE_INT(data->store, u->skill_size);
|
2017-09-15 21:55:57 +02:00
|
|
|
for (i = 0; i != u->skill_size; ++i) {
|
|
|
|
skill *sv = u->skills + i;
|
2017-09-16 07:37:33 +02:00
|
|
|
#ifndef NDEBUG
|
|
|
|
assert(sv->id > sk);
|
|
|
|
sk = sv->id;
|
2017-09-15 21:55:57 +02:00
|
|
|
assert(sv->weeks <= sv->level * 2 + 1);
|
2017-09-16 07:37:33 +02:00
|
|
|
#endif
|
2017-09-15 22:03:54 +02:00
|
|
|
WRITE_INT(data->store, sv->id);
|
|
|
|
WRITE_INT(data->store, sv->level);
|
|
|
|
WRITE_INT(data->store, sv->weeks);
|
2017-09-15 21:55:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unit *read_unit(gamedata *data)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-05-25 05:25:06 +02:00
|
|
|
unit *u;
|
2016-08-28 21:44:09 +02:00
|
|
|
const race *rc;
|
2020-05-09 21:06:25 +02:00
|
|
|
int number, n, p, bn, sn;
|
2014-05-25 05:25:06 +02:00
|
|
|
order **orderp;
|
|
|
|
char obuf[DISPLAYSIZE];
|
|
|
|
faction *f;
|
|
|
|
char rname[32];
|
2020-08-03 10:31:58 +02:00
|
|
|
static const struct race * rc_demon;
|
|
|
|
static int config;
|
|
|
|
|
|
|
|
if (rc_changed(&config)) {
|
|
|
|
rc_demon = get_race(RC_DAEMON);
|
|
|
|
}
|
2014-05-25 05:25:06 +02:00
|
|
|
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
if (n <= 0) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("data contains invalid unit %d.", n);
|
2014-07-23 11:54:51 +02:00
|
|
|
assert(n > 0);
|
2014-05-25 05:25:06 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
u = findunit(n);
|
|
|
|
if (u) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("reading unit %s that already exists.", unitname(u));
|
2014-05-25 05:25:06 +02:00
|
|
|
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);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
else {
|
2018-11-01 10:16:49 +01:00
|
|
|
u = unit_create(n);
|
2014-05-25 05:25:06 +02:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-05-25 05:25:06 +02:00
|
|
|
READ_INT(data->store, &n);
|
|
|
|
f = findfaction(n);
|
|
|
|
if (f != u->faction) {
|
|
|
|
u_setfaction(u, f);
|
|
|
|
}
|
2017-03-11 19:36:26 +01:00
|
|
|
if (!u->faction) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("unit %s has faction == NULL", itoa36(u->no));
|
2014-05-25 05:25:06 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2013-12-31 10:06:28 +01:00
|
|
|
READ_STR(data->store, obuf, sizeof(obuf));
|
2016-11-11 00:30:49 +01:00
|
|
|
if (unicode_utf8_trim(obuf)!=0) {
|
2016-11-11 11:09:57 +01:00
|
|
|
log_warning("trim unit %s name to '%s'", itoa36(u->no), obuf);
|
2016-11-11 21:46:56 +01:00
|
|
|
}
|
2018-10-24 20:16:17 +02:00
|
|
|
unit_setname(u, obuf[0] ? obuf : NULL);
|
2017-11-25 20:24:57 +01:00
|
|
|
READ_STR(data->store, obuf, sizeof(obuf));
|
|
|
|
if (unicode_utf8_trim(obuf)!=0) {
|
|
|
|
log_warning("trim unit %s info to '%s'", itoa36(u->no), obuf);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2018-10-24 20:16:17 +02:00
|
|
|
unit_setinfo(u, obuf[0] ? obuf : NULL);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &number);
|
2014-05-25 05:25:06 +02:00
|
|
|
set_number(u, number);
|
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &n);
|
2019-01-24 17:50:58 +01:00
|
|
|
u->age = n;
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_TOK(data->store, rname, sizeof(rname));
|
2016-08-28 21:44:09 +02:00
|
|
|
rc = rc_find(rname);
|
|
|
|
assert(rc);
|
|
|
|
u_setrace(u, rc);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_TOK(data->store, rname, sizeof(rname));
|
2017-11-18 17:57:30 +01:00
|
|
|
if (rname[0])
|
2014-07-23 11:54:51 +02:00
|
|
|
u->irace = rc_find(rname);
|
|
|
|
else
|
|
|
|
u->irace = NULL;
|
|
|
|
|
2020-05-09 21:06:25 +02:00
|
|
|
READ_INT(data->store, &bn);
|
|
|
|
READ_INT(data->store, &sn);
|
|
|
|
if (sn <= 0 && bn > 0) {
|
|
|
|
building * b = findbuilding(bn);
|
2014-07-23 11:54:51 +02:00
|
|
|
if (b) {
|
|
|
|
u_set_building(u, b);
|
|
|
|
if (fval(u, UFL_OWNER)) {
|
|
|
|
building_set_owner(u);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2020-05-09 21:06:25 +02:00
|
|
|
log_error("read_unit: unit in unkown building '%s'", itoa36(bn));
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-09 21:06:25 +02:00
|
|
|
if (sn > 0) {
|
|
|
|
ship * sh = findship(sn);
|
2014-07-23 11:54:51 +02:00
|
|
|
if (sh) {
|
|
|
|
u_set_ship(u, sh);
|
|
|
|
if (fval(u, UFL_OWNER)) {
|
|
|
|
ship_set_owner(u);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2020-05-09 21:06:25 +02:00
|
|
|
log_error("read_unit: unit in unkown ship '%s'", itoa36(sn));
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2013-12-31 10:06:28 +01:00
|
|
|
READ_INT(data->store, &n);
|
2018-02-15 20:25:58 +01:00
|
|
|
unit_setstatus(u, (status_t)n);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &u->flags);
|
|
|
|
u->flags &= UFL_SAVEMASK;
|
2015-11-09 13:36:52 +01:00
|
|
|
if ((u->flags & UFL_ANON_FACTION) && !rule_stealth_anon()) {
|
2014-07-23 11:54:51 +02:00
|
|
|
/* if this rule is broken, then fix broken units */
|
|
|
|
u->flags -= UFL_ANON_FACTION;
|
2016-05-29 10:58:49 +02:00
|
|
|
log_warning("%s was anonymous.", unitname(u));
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
/* Persistente Befehle einlesen */
|
|
|
|
free_orders(&u->orders);
|
|
|
|
READ_STR(data->store, obuf, sizeof(obuf));
|
|
|
|
p = n = 0;
|
|
|
|
orderp = &u->orders;
|
|
|
|
while (obuf[0]) {
|
2017-11-25 20:24:57 +01:00
|
|
|
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;
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2017-11-25 20:24:57 +01:00
|
|
|
else if (p == MAXPERSISTENT) {
|
|
|
|
log_info("%s had %d or more persistent orders", unitname(u), MAXPERSISTENT);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2017-11-25 20:24:57 +01:00
|
|
|
}
|
|
|
|
else if (n == MAXORDERS) {
|
|
|
|
log_info("%s had %d or more orders", unitname(u), MAXORDERS);
|
|
|
|
}
|
|
|
|
if (ord != NULL) {
|
|
|
|
free_order(ord);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
READ_STR(data->store, obuf, sizeof(obuf));
|
|
|
|
}
|
|
|
|
set_order(&u->thisorder, NULL);
|
|
|
|
|
|
|
|
assert(u_race(u));
|
2017-09-15 21:55:57 +02:00
|
|
|
read_skills(data, u);
|
2014-07-23 11:54:51 +02:00
|
|
|
read_items(data->store, &u->items);
|
|
|
|
READ_INT(data->store, &u->hp);
|
|
|
|
if (u->hp < u->number) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("Einheit %s hat %u Personen, und %u Trefferpunkte", itoa36(u->no), u->number, u->hp);
|
2014-07-23 11:54:51 +02:00
|
|
|
u->hp = u->number;
|
|
|
|
}
|
2016-02-13 13:42:02 +01:00
|
|
|
read_attribs(data, &u->attribs, u);
|
2020-08-03 10:31:58 +02:00
|
|
|
if (rc_demon && data->version < FIX_SHAPESHIFT_VERSION) {
|
|
|
|
if (u_race(u) == rc_demon) {
|
|
|
|
const char *zRace = get_racename(u->attribs);
|
|
|
|
if (zRace) {
|
|
|
|
const struct race *rc = rc_find(zRace);
|
|
|
|
if (rc) {
|
|
|
|
set_racename(&u->attribs, NULL);
|
|
|
|
u->irace = rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-09-21 16:26:53 +02:00
|
|
|
resolve_unit(u);
|
2014-07-23 11:54:51 +02:00
|
|
|
return u;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
void write_unit(gamedata *data, const unit * u)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-10-24 20:16:17 +02:00
|
|
|
const char *str;
|
2014-07-23 11:54:51 +02:00
|
|
|
order *ord;
|
2017-09-15 21:55:57 +02:00
|
|
|
int p = 0;
|
2014-07-23 11:54:51 +02:00
|
|
|
unsigned int flags = u->flags & UFL_SAVEMASK;
|
|
|
|
const race *irace = u_irace(u);
|
|
|
|
|
|
|
|
write_unit_reference(u, data->store);
|
2016-02-17 13:55:48 +01:00
|
|
|
assert(u->faction->_alive);
|
2014-07-23 11:54:51 +02:00
|
|
|
write_faction_reference(u->faction, data->store);
|
2015-02-11 07:55:35 +01:00
|
|
|
WRITE_STR(data->store, u->_name);
|
2018-10-24 20:16:17 +02:00
|
|
|
str = unit_getinfo(u);
|
|
|
|
WRITE_STR(data->store, str ? str : "");
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_INT(data->store, u->number);
|
|
|
|
WRITE_INT(data->store, u->age);
|
2014-08-24 21:49:55 +02:00
|
|
|
WRITE_TOK(data->store, u_race(u)->_name);
|
|
|
|
WRITE_TOK(data->store, (irace && irace != u_race(u)) ? irace->_name : "");
|
2014-07-23 11:54:51 +02:00
|
|
|
write_building_reference(u->building, data->store);
|
|
|
|
write_ship_reference(u->ship, data->store);
|
|
|
|
WRITE_INT(data->store, u->status);
|
|
|
|
if (u->building && u == building_owner(u->building)) flags |= UFL_OWNER;
|
|
|
|
if (u->ship && u == ship_owner(u->ship)) flags |= UFL_OWNER;
|
|
|
|
WRITE_INT(data->store, flags);
|
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
for (ord = u->old_orders; ord; ord = ord->next) {
|
|
|
|
if (++p < MAXPERSISTENT) {
|
|
|
|
writeorder(data, ord, u->faction->locale);
|
|
|
|
}
|
|
|
|
else {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_info("%s had %d or more persistent orders", unitname(u), MAXPERSISTENT);
|
2014-07-23 11:54:51 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (ord = u->orders; ord; ord = ord->next) {
|
2015-08-04 22:47:55 +02:00
|
|
|
keyword_t kwd = getkeyword(ord);
|
|
|
|
if (u->old_orders && is_repeated(kwd))
|
2014-07-23 11:54:51 +02:00
|
|
|
continue; /* has new defaults */
|
|
|
|
if (is_persistent(ord)) {
|
|
|
|
if (++p < MAXPERSISTENT) {
|
|
|
|
writeorder(data, ord, u->faction->locale);
|
|
|
|
}
|
|
|
|
else {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_info("%s had %d or more persistent orders", unitname(u), MAXPERSISTENT);
|
2014-07-23 11:54:51 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* write an empty string to terminate the list */
|
|
|
|
WRITE_STR(data->store, "");
|
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
|
|
|
|
assert(u_race(u));
|
2017-09-15 21:55:57 +02:00
|
|
|
write_skills(data, u);
|
2014-07-23 11:54:51 +02:00
|
|
|
write_items(data->store, u->items);
|
|
|
|
WRITE_SECTION(data->store);
|
2017-05-24 08:52:19 +02:00
|
|
|
if (u->hp == 0 && data->version < NORCSPELL_VERSION) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("unit %s has 0 hitpoints, adjusting.", itoa36(u->no));
|
2014-07-23 11:54:51 +02:00
|
|
|
((unit *)u)->hp = u->number;
|
|
|
|
}
|
|
|
|
WRITE_INT(data->store, u->hp);
|
|
|
|
WRITE_SECTION(data->store);
|
2016-02-09 00:28:23 +01:00
|
|
|
write_attribs(data->store, u->attribs, u);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_SECTION(data->store);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-09-16 09:03:42 +02:00
|
|
|
static void read_regioninfo(gamedata *data, const region *r, char *info, size_t len) {
|
2017-11-25 20:24:57 +01:00
|
|
|
READ_STR(data->store, info, len);
|
|
|
|
if (unicode_utf8_trim(info) != 0) {
|
|
|
|
log_warning("trim region %d info to '%s'", r->uid, info);
|
2017-09-16 07:58:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-30 18:56:37 +02:00
|
|
|
static void fix_resource_levels(region *r) {
|
2019-03-24 21:27:43 +01:00
|
|
|
struct terrain_production *p;
|
|
|
|
for (p = r->terrain->production; p->type; ++p) {
|
|
|
|
char *end;
|
|
|
|
long start = (int)strtol(p->startlevel, &end, 10);
|
|
|
|
if (*end == '\0') {
|
|
|
|
rawmaterial *res;
|
|
|
|
for (res = r->resources; res; res = res->next) {
|
|
|
|
if (p->type == res->rtype) {
|
|
|
|
if (start != res->startlevel) {
|
|
|
|
log_debug("setting resource start level for %s in %s to %d",
|
|
|
|
res->rtype->_name, regionname(r, NULL), start);
|
|
|
|
res->startlevel = start;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-30 18:56:37 +02:00
|
|
|
static void fix_resource_bases(region *r) {
|
|
|
|
struct terrain_production *p;
|
|
|
|
for (p = r->terrain->production; p->type; ++p) {
|
|
|
|
char *end;
|
|
|
|
long base = (int)strtol(p->base, &end, 10);
|
|
|
|
if (*end == '\0') {
|
|
|
|
rawmaterial *res;
|
|
|
|
for (res = r->resources; res; res = res->next) {
|
|
|
|
if (p->type == res->rtype) {
|
|
|
|
if (base != res->base) {
|
|
|
|
log_debug("setting resource base for %s in %s to %d",
|
|
|
|
res->rtype->_name, regionname(r, NULL), base);
|
|
|
|
res->base = base;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
static region *readregion(gamedata *data, int x, int y)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-11-03 14:28:32 +01:00
|
|
|
region *r;
|
2014-07-23 11:54:51 +02:00
|
|
|
const terrain_type *terrain;
|
2014-12-12 22:52:23 +01:00
|
|
|
char name[NAMESIZE];
|
2017-09-16 08:42:49 +02:00
|
|
|
char info[DISPLAYSIZE];
|
2014-07-23 11:54:51 +02:00
|
|
|
int uid = 0;
|
|
|
|
int n;
|
|
|
|
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_INT(data->store, &uid);
|
2018-11-03 14:28:32 +01:00
|
|
|
r = findregionbyid(uid);
|
2014-07-23 11:54:51 +02:00
|
|
|
if (r == NULL) {
|
2018-11-03 15:48:35 +01:00
|
|
|
r = region_create(uid);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
else {
|
2018-11-03 15:48:35 +01:00
|
|
|
/* make sure this was not read earlier */
|
|
|
|
assert(r->next == NULL);
|
|
|
|
assert(r->attribs == NULL);
|
|
|
|
assert(r->land == NULL);
|
|
|
|
assert(r->resources == NULL);
|
|
|
|
}
|
|
|
|
/* add region to the global list: */
|
|
|
|
add_region(r, x, y);
|
2017-09-16 07:58:57 +02:00
|
|
|
if (data->version < LANDDISPLAY_VERSION) {
|
2017-09-16 09:03:42 +02:00
|
|
|
read_regioninfo(data, r, info, sizeof(info));
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2017-09-16 09:34:47 +02:00
|
|
|
else {
|
|
|
|
info[0] = '\0';
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_STR(data->store, name, sizeof(name));
|
|
|
|
terrain = get_terrain(name);
|
|
|
|
if (terrain == NULL) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("Unknown terrain '%s'", name);
|
2014-12-12 22:52:23 +01:00
|
|
|
assert(!"unknown terrain");
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
r->terrain = terrain;
|
|
|
|
READ_INT(data->store, &r->flags);
|
2013-12-31 10:06:28 +01:00
|
|
|
READ_INT(data->store, &n);
|
2014-07-23 11:54:51 +02:00
|
|
|
r->age = (unsigned short)n;
|
|
|
|
|
|
|
|
if (fval(r->terrain, LAND_REGION)) {
|
|
|
|
r->land = calloc(1, sizeof(land_region));
|
|
|
|
READ_STR(data->store, name, sizeof(name));
|
2017-09-16 07:58:57 +02:00
|
|
|
if (unicode_utf8_trim(name) != 0) {
|
|
|
|
log_warning("trim region %d name to '%s'", uid, name);
|
|
|
|
};
|
2017-12-28 18:29:40 +01:00
|
|
|
r->land->name = str_strdup(name);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
if (r->land) {
|
|
|
|
int i;
|
|
|
|
rawmaterial **pres = &r->resources;
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
if (data->version >= LANDDISPLAY_VERSION) {
|
2017-09-16 09:03:42 +02:00
|
|
|
read_regioninfo(data, r, info, sizeof(info));
|
2017-09-16 07:58:57 +02:00
|
|
|
}
|
2017-09-16 08:42:49 +02:00
|
|
|
region_setinfo(r, info);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &i);
|
|
|
|
if (i < 0) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("number of trees in %s is %d.", regionname(r, NULL), i);
|
2014-07-23 11:54:51 +02:00
|
|
|
i = 0;
|
|
|
|
}
|
|
|
|
rsettrees(r, 0, i);
|
|
|
|
READ_INT(data->store, &i);
|
|
|
|
if (i < 0) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("number of young trees in %s is %d.", regionname(r, NULL), i);
|
2014-07-23 11:54:51 +02:00
|
|
|
i = 0;
|
|
|
|
}
|
|
|
|
rsettrees(r, 1, i);
|
|
|
|
READ_INT(data->store, &i);
|
|
|
|
if (i < 0) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("number of seeds in %s is %d.", regionname(r, NULL), i);
|
2014-07-23 11:54:51 +02:00
|
|
|
i = 0;
|
|
|
|
}
|
|
|
|
rsettrees(r, 2, i);
|
|
|
|
|
|
|
|
READ_INT(data->store, &i);
|
|
|
|
rsethorses(r, i);
|
|
|
|
assert(*pres == NULL);
|
|
|
|
for (;;) {
|
|
|
|
rawmaterial *res;
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_STR(data->store, name, sizeof(name));
|
|
|
|
if (strcmp(name, "end") == 0)
|
2014-07-23 11:54:51 +02:00
|
|
|
break;
|
|
|
|
res = malloc(sizeof(rawmaterial));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (!res) abort();
|
2017-03-22 20:37:09 +01:00
|
|
|
res->rtype = rt_find(name);
|
2017-03-22 20:46:29 +01:00
|
|
|
if (!res->rtype && strncmp("rm_", name, 3) == 0) {
|
|
|
|
res->rtype = rt_find(name + 3);
|
|
|
|
}
|
2017-03-22 20:37:09 +01:00
|
|
|
if (!res->rtype || !res->rtype->raw) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("invalid resourcetype %s in data.", name);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2017-03-22 20:37:09 +01:00
|
|
|
assert(res->rtype);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &n);
|
|
|
|
res->level = n;
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
res->amount = n;
|
|
|
|
res->flags = 0;
|
|
|
|
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
res->startlevel = n;
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
res->base = n;
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
res->divisor = n;
|
|
|
|
|
|
|
|
*pres = res;
|
|
|
|
pres = &res->next;
|
|
|
|
}
|
|
|
|
*pres = NULL;
|
|
|
|
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_STR(data->store, name, sizeof(name));
|
|
|
|
if (strcmp(name, "noherb") != 0) {
|
|
|
|
const resource_type *rtype = rt_find(name);
|
2014-07-23 11:54:51 +02:00
|
|
|
assert(rtype && rtype->itype && fval(rtype->itype, ITF_HERB));
|
|
|
|
rsetherbtype(r, rtype->itype);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rsetherbtype(r, NULL);
|
|
|
|
}
|
|
|
|
READ_INT(data->store, &n);
|
2016-08-18 19:04:48 +02:00
|
|
|
rsetherbs(r, n);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &n);
|
2016-01-24 14:31:24 +01:00
|
|
|
if (n < 0) {
|
|
|
|
/* bug 2182 */
|
|
|
|
log_error("data has negative peasants: %d in %s", n, regionname(r, 0));
|
|
|
|
rsetpeasants(r, 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rsetpeasants(r, n);
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &n);
|
|
|
|
rsetmoney(r, n);
|
|
|
|
}
|
2017-09-16 08:42:49 +02:00
|
|
|
else {
|
|
|
|
if (info[0]) {
|
|
|
|
log_error("%s %d has a description: %s", r->terrain->_name, r->uid, info);
|
|
|
|
}
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
assert(r->terrain != NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-06-25 07:44:05 +02:00
|
|
|
if (r->land) {
|
|
|
|
int n;
|
|
|
|
for (;;) {
|
|
|
|
const struct resource_type *rtype;
|
2014-12-12 22:52:23 +01:00
|
|
|
READ_STR(data->store, name, sizeof(name));
|
|
|
|
if (!strcmp(name, "end"))
|
2014-06-25 07:44:05 +02:00
|
|
|
break;
|
2014-12-12 22:52:23 +01:00
|
|
|
rtype = rt_find(name);
|
2014-06-25 07:44:05 +02:00
|
|
|
assert(rtype && rtype->ltype);
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
r_setdemand(r, rtype->ltype, n);
|
|
|
|
}
|
2016-10-24 13:54:53 +02:00
|
|
|
if (!r->land->demands) {
|
|
|
|
fix_demand(r);
|
|
|
|
}
|
2017-05-09 18:49:10 +02:00
|
|
|
if (data->version < NOLANDITEM_VERSION) {
|
|
|
|
read_items(data->store, NULL);
|
|
|
|
}
|
2014-06-25 07:44:05 +02:00
|
|
|
if (data->version >= REGIONOWNER_VERSION) {
|
|
|
|
READ_INT(data->store, &n);
|
2019-01-24 17:50:58 +01:00
|
|
|
if (n < 0) n = 0;
|
|
|
|
region_set_morale(r, n, -1);
|
2014-06-25 07:44:05 +02:00
|
|
|
read_owner(data, &r->land->ownership);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2016-02-13 13:42:02 +01:00
|
|
|
read_attribs(data, &r->attribs, r);
|
2019-03-24 21:27:43 +01:00
|
|
|
|
2019-04-30 18:56:37 +02:00
|
|
|
if (r->resources) {
|
|
|
|
if (data->version < FIX_STARTLEVEL_VERSION) {
|
|
|
|
/* we had some badly made rawmaterials before this */
|
|
|
|
fix_resource_levels(r);
|
|
|
|
}
|
|
|
|
if (data->version < FIX_RES_BASE_VERSION) {
|
|
|
|
/* we had some badly made rawmaterials before this */
|
|
|
|
fix_resource_bases(r);
|
|
|
|
}
|
2019-03-24 21:27:43 +01:00
|
|
|
}
|
2014-06-25 07:44:05 +02:00
|
|
|
return r;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2016-11-12 01:03:07 +01:00
|
|
|
region *read_region(gamedata *data)
|
|
|
|
{
|
|
|
|
storage *store = data->store;
|
|
|
|
region *r;
|
|
|
|
int x, y;
|
|
|
|
READ_INT(store, &x);
|
|
|
|
READ_INT(store, &y);
|
|
|
|
r = readregion(data, x, y);
|
2017-09-21 16:26:53 +02:00
|
|
|
resolve_region(r);
|
2016-11-12 01:03:07 +01:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
void writeregion(gamedata *data, const region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-07-07 00:49:12 +02:00
|
|
|
assert(r);
|
|
|
|
assert(data);
|
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_INT(data->store, r->uid);
|
|
|
|
WRITE_TOK(data->store, r->terrain->_name);
|
|
|
|
WRITE_INT(data->store, r->flags & RF_SAVEMASK);
|
|
|
|
WRITE_INT(data->store, r->age);
|
|
|
|
WRITE_SECTION(data->store);
|
2017-09-16 07:58:57 +02:00
|
|
|
if (r->land) {
|
2014-07-23 11:54:51 +02:00
|
|
|
const item_type *rht;
|
|
|
|
struct demand *demand;
|
|
|
|
rawmaterial *res = r->resources;
|
2016-02-23 08:35:31 +01:00
|
|
|
|
2015-07-07 00:49:12 +02:00
|
|
|
WRITE_STR(data->store, (const char *)r->land->name);
|
2017-09-16 07:58:57 +02:00
|
|
|
WRITE_STR(data->store, region_getinfo(r));
|
2014-07-23 11:54:51 +02:00
|
|
|
assert(rtrees(r, 0) >= 0);
|
|
|
|
assert(rtrees(r, 1) >= 0);
|
|
|
|
assert(rtrees(r, 2) >= 0);
|
|
|
|
WRITE_INT(data->store, rtrees(r, 0));
|
|
|
|
WRITE_INT(data->store, rtrees(r, 1));
|
|
|
|
WRITE_INT(data->store, rtrees(r, 2));
|
|
|
|
WRITE_INT(data->store, rhorses(r));
|
|
|
|
|
|
|
|
while (res) {
|
2017-03-22 20:37:09 +01:00
|
|
|
WRITE_TOK(data->store, res->rtype->_name);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_INT(data->store, res->level);
|
|
|
|
WRITE_INT(data->store, res->amount);
|
|
|
|
WRITE_INT(data->store, res->startlevel);
|
|
|
|
WRITE_INT(data->store, res->base);
|
|
|
|
WRITE_INT(data->store, res->divisor);
|
|
|
|
res = res->next;
|
|
|
|
}
|
|
|
|
WRITE_TOK(data->store, "end");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
rht = rherbtype(r);
|
|
|
|
if (rht) {
|
|
|
|
WRITE_TOK(data->store, resourcename(rht->rtype, 0));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
WRITE_TOK(data->store, "noherb");
|
|
|
|
}
|
|
|
|
WRITE_INT(data->store, rherbs(r));
|
|
|
|
WRITE_INT(data->store, rpeasants(r));
|
|
|
|
WRITE_INT(data->store, rmoney(r));
|
2015-07-07 00:49:12 +02:00
|
|
|
for (demand = r->land->demands; demand; demand = demand->next) {
|
|
|
|
WRITE_TOK(data->store, resourcename(demand->type->itype->rtype, 0));
|
|
|
|
WRITE_INT(data->store, demand->value);
|
2014-12-25 22:38:01 +01:00
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_TOK(data->store, "end");
|
|
|
|
WRITE_SECTION(data->store);
|
2015-12-07 17:59:50 +01:00
|
|
|
WRITE_INT(data->store, region_get_morale(r));
|
2014-07-23 11:54:51 +02:00
|
|
|
write_owner(data, r->land->ownership);
|
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
}
|
2016-02-09 00:28:23 +01:00
|
|
|
write_attribs(data->store, r->attribs, r);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_SECTION(data->store);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2016-11-12 01:03:07 +01:00
|
|
|
void write_region(gamedata *data, const region *r)
|
|
|
|
{
|
|
|
|
storage *store = data->store;
|
|
|
|
WRITE_INT(store, r->x);
|
|
|
|
WRITE_INT(store, r->y);
|
|
|
|
writeregion(data, r);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
2014-07-23 11:54:51 +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;
|
2012-05-26 23:19:35 +02:00
|
|
|
}
|
|
|
|
|
2016-02-20 22:23:19 +01:00
|
|
|
static char * getpasswd(int fno) {
|
2016-02-20 18:21:39 +01:00
|
|
|
FILE * F = fopen("passwords.txt", "r");
|
|
|
|
if (F) {
|
2018-12-09 03:42:08 +01:00
|
|
|
const char *prefix = itoa36(fno);
|
|
|
|
size_t len = strlen(prefix);
|
2016-02-20 18:21:39 +01:00
|
|
|
while (!feof(F)) {
|
2018-12-09 03:42:08 +01:00
|
|
|
char line[80];
|
2016-02-20 18:21:39 +01:00
|
|
|
fgets(line, sizeof(line), F);
|
2016-02-23 08:35:31 +01:00
|
|
|
if (line[len] == ':' && strncmp(prefix, line, len) == 0) {
|
|
|
|
size_t slen = strlen(line) - 1;
|
|
|
|
assert(line[slen] == '\n');
|
2016-02-20 22:23:19 +01:00
|
|
|
line[slen] = 0;
|
2016-02-20 18:21:39 +01:00
|
|
|
fclose(F);
|
2017-12-28 18:29:40 +01:00
|
|
|
return str_strdup(line + len + 1);
|
2016-02-20 18:21:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(F);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-02-21 10:10:26 +01:00
|
|
|
|
|
|
|
static void read_password(gamedata *data, faction *f) {
|
|
|
|
char name[128];
|
|
|
|
READ_STR(data->store, name, sizeof(name));
|
2016-05-29 10:58:49 +02:00
|
|
|
if (name[0] == '$' && data->version == BADCRYPT_VERSION) {
|
2016-02-21 11:01:50 +01:00
|
|
|
char * pass = getpasswd(f->no);
|
|
|
|
if (pass) {
|
2018-09-26 21:06:56 +02:00
|
|
|
faction_setpassword(f, password_hash(pass, PASSWORD_DEFAULT));
|
2017-02-18 21:15:14 +01:00
|
|
|
free(pass); /* TODO: remove this allocation! */
|
2016-02-21 11:01:50 +01:00
|
|
|
}
|
|
|
|
else {
|
2016-05-20 20:49:47 +02:00
|
|
|
log_error("data version is BADCRYPT but %s not in password.txt", itoa36(f->no));
|
2016-02-21 11:01:50 +01:00
|
|
|
}
|
2016-02-21 10:10:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2018-09-26 21:06:56 +02:00
|
|
|
faction_setpassword(f, (data->version >= CRYPT_VERSION) ? name : password_hash(name, PASSWORD_DEFAULT));
|
2016-02-21 10:10:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _test_read_password(gamedata *data, faction *f) {
|
|
|
|
read_password(data, f);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void write_password(gamedata *data, const faction *f) {
|
2018-10-24 09:27:48 +02:00
|
|
|
WRITE_TOK(data->store, faction_getpassword(f));
|
2016-02-21 10:10:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void _test_write_password(gamedata *data, const faction *f) {
|
|
|
|
write_password(data, f);
|
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
faction *read_faction(gamedata * data)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
int planes, n;
|
|
|
|
faction *f;
|
|
|
|
char name[DISPLAYSIZE];
|
|
|
|
|
|
|
|
READ_INT(data->store, &n);
|
2016-02-05 23:10:05 +01:00
|
|
|
assert(n > 0);
|
2014-07-23 11:54:51 +02:00
|
|
|
f = findfaction(n);
|
|
|
|
if (f == NULL) {
|
2018-10-30 06:30:32 +01:00
|
|
|
f = faction_create(n);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
else {
|
2016-11-12 00:47:25 +01:00
|
|
|
f->allies = NULL; /* FIXME: mem leak */
|
|
|
|
while (f->attribs) {
|
2014-07-23 11:54:51 +02:00
|
|
|
a_remove(&f->attribs, f->attribs);
|
2016-11-12 00:47:25 +01:00
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2018-09-24 20:18:21 +02:00
|
|
|
READ_INT(data->store, &f->uid);
|
|
|
|
if (data->version < FACTION_UID_VERSION) {
|
|
|
|
f->uid = 0;
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2015-08-19 19:43:47 +02:00
|
|
|
if (data->version >= SPELL_LEVEL_VERSION) {
|
|
|
|
READ_INT(data->store, &f->max_spelllevel);
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
if (alliances || data->version >= OWNER_2_VERSION) {
|
|
|
|
int allianceid;
|
|
|
|
READ_INT(data->store, &allianceid);
|
|
|
|
if (allianceid > 0)
|
|
|
|
f->alliance = findalliance(allianceid);
|
|
|
|
if (f->alliance) {
|
|
|
|
alliance *al = f->alliance;
|
|
|
|
if (al->flags & ALF_NON_ALLIED) {
|
|
|
|
assert(!al->members
|
|
|
|
|| !"non-allied dummy-alliance has more than one member");
|
|
|
|
}
|
2017-01-26 17:41:21 +01:00
|
|
|
selist_push(&al->members, f);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
else if (rule_region_owners()) {
|
|
|
|
/* compat fix for non-allied factions */
|
|
|
|
alliance *al = makealliance(0, NULL);
|
|
|
|
setalliance(f, al);
|
|
|
|
}
|
|
|
|
if (data->version >= OWNER_2_VERSION) {
|
|
|
|
READ_INT(data->store, &f->alliance_joindate);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
f->alliance_joindate = turn - 10; /* we're guessing something safe here */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
READ_STR(data->store, name, sizeof(name));
|
2016-11-11 00:30:49 +01:00
|
|
|
if (unicode_utf8_trim(name)!=0) {
|
2016-11-11 11:09:57 +01:00
|
|
|
log_warning("trim faction %s name to '%s'", itoa36(f->no), name);
|
2016-11-11 00:30:49 +01:00
|
|
|
};
|
2017-12-28 18:29:40 +01:00
|
|
|
f->name = str_strdup(name);
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_STR(data->store, name, sizeof(name));
|
2016-11-11 00:30:49 +01:00
|
|
|
if (unicode_utf8_trim(name)!=0) {
|
2016-11-11 11:09:57 +01:00
|
|
|
log_warning("trim faction %s banner to '%s'", itoa36(f->no), name);
|
2016-11-11 00:30:49 +01:00
|
|
|
};
|
2018-10-24 19:54:07 +02:00
|
|
|
faction_setbanner(f, name);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-11-17 21:29:15 +01:00
|
|
|
log_debug(" - Lese Partei %s (%s)", f->name, itoa36(f->no));
|
2014-07-23 11:54:51 +02:00
|
|
|
|
|
|
|
READ_STR(data->store, name, sizeof(name));
|
2017-11-14 16:20:31 +01:00
|
|
|
if (check_email(name) == 0) {
|
|
|
|
faction_setemail(f, name);
|
2018-02-25 13:22:32 +01:00
|
|
|
} else if (name[0]) {
|
2017-11-14 16:20:31 +01:00
|
|
|
log_warning("Invalid email address for faction %s: %s", itoa36(f->no), name);
|
|
|
|
faction_setemail(f, NULL);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
|
2016-02-21 10:10:26 +01:00
|
|
|
read_password(data, f);
|
2014-08-14 05:06:36 +02:00
|
|
|
if (data->version < NOOVERRIDE_VERSION) {
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_STR(data->store, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
READ_STR(data->store, name, sizeof(name));
|
|
|
|
f->locale = get_locale(name);
|
2014-11-23 21:15:09 +01:00
|
|
|
if (!f->locale) f->locale = default_locale;
|
2014-07-23 11:54:51 +02:00
|
|
|
READ_INT(data->store, &f->lastorders);
|
|
|
|
READ_INT(data->store, &f->age);
|
|
|
|
READ_STR(data->store, name, sizeof(name));
|
|
|
|
f->race = rc_find(name);
|
2014-06-01 11:07:22 +02:00
|
|
|
if (!f->race) {
|
2016-05-29 10:58:49 +02:00
|
|
|
log_error("unknown race in data: %s", name);
|
2014-06-01 11:07:22 +02:00
|
|
|
}
|
|
|
|
assert(f->race);
|
2013-12-31 10:06:28 +01:00
|
|
|
READ_INT(data->store, &n);
|
2014-07-23 11:54:51 +02:00
|
|
|
f->magiegebiet = (magic_t)n;
|
|
|
|
|
|
|
|
if (data->version < FOSS_VERSION) {
|
|
|
|
/* ignore karma */
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-05-10 18:57:04 +02:00
|
|
|
READ_INT(data->store, &f->flags);
|
|
|
|
if (data->version < INTFLAGS_VERSION) {
|
2014-07-23 11:54:51 +02:00
|
|
|
if (f->no == 0 || f->no == 666) {
|
2014-10-16 07:46:08 +02:00
|
|
|
f->flags = FFL_NPC | FFL_NOIDLEOUT;
|
2014-05-10 18:57:04 +02:00
|
|
|
}
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-02-13 13:42:02 +01:00
|
|
|
read_attribs(data, &f->attribs, f);
|
2014-12-12 23:00:30 +01:00
|
|
|
read_items(data->store, &f->items);
|
2010-08-08 10:06:34 +02:00
|
|
|
for (;;) {
|
2013-12-31 10:06:28 +01:00
|
|
|
READ_TOK(data->store, name, sizeof(name));
|
2014-07-23 11:54:51 +02:00
|
|
|
if (strcmp("end", name) == 0)
|
|
|
|
break;
|
|
|
|
READ_INT(data->store, &n); /* there used to be a level here, which is now ignored */
|
|
|
|
}
|
|
|
|
READ_INT(data->store, &planes);
|
|
|
|
while (--planes >= 0) {
|
|
|
|
int id, ux, uy;
|
|
|
|
READ_INT(data->store, &id);
|
|
|
|
READ_INT(data->store, &ux);
|
|
|
|
READ_INT(data->store, &uy);
|
2015-05-19 08:02:32 +02:00
|
|
|
faction_setorigin(f, id, ux, uy);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
f->newbies = 0;
|
|
|
|
|
|
|
|
READ_INT(data->store, &n);
|
|
|
|
f->options = n;
|
|
|
|
|
2018-02-03 14:49:35 +01:00
|
|
|
n = WANT_OPTION(O_REPORT) | WANT_OPTION(O_COMPUTER);
|
2014-11-11 16:53:56 +01:00
|
|
|
if ((f->options & n) == 0) {
|
2014-07-23 11:54:51 +02:00
|
|
|
/* Kein Report eingestellt, Fehler */
|
2014-11-11 16:53:56 +01:00
|
|
|
f->options |= n;
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2015-03-29 14:03:56 +02:00
|
|
|
if (data->version < JSON_REPORT_VERSION) {
|
|
|
|
/* mistakes were made in the past*/
|
2018-02-03 14:49:35 +01:00
|
|
|
f->options &= ~WANT_OPTION(O_JSON);
|
2015-03-29 14:03:56 +02:00
|
|
|
}
|
2018-10-26 19:47:50 +02:00
|
|
|
read_allies(data, &f->allies);
|
2016-02-13 13:42:02 +01:00
|
|
|
read_groups(data, f);
|
2014-07-23 11:54:51 +02:00
|
|
|
f->spellbook = 0;
|
|
|
|
if (data->version >= REGIONOWNER_VERSION) {
|
2016-02-13 15:48:48 +01:00
|
|
|
read_spellbook(FactionSpells() ? &f->spellbook : 0, data, get_spell_level_faction, (void *)f);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
return f;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
void write_faction(gamedata *data, const faction * f)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-10-22 21:51:11 +02:00
|
|
|
origin *ur;
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-02-18 08:11:39 +01:00
|
|
|
assert(f->_alive);
|
2018-01-21 10:32:39 +01:00
|
|
|
assert(f->no > 0 && f->no <= MAX_UNIT_NR);
|
|
|
|
WRITE_INT(data->store, f->no);
|
2018-09-24 20:18:21 +02:00
|
|
|
WRITE_INT(data->store, f->uid);
|
2015-08-19 19:43:47 +02:00
|
|
|
#if RELEASE_VERSION >= SPELL_LEVEL_VERSION
|
|
|
|
WRITE_INT(data->store, f->max_spelllevel);
|
|
|
|
#endif
|
2014-07-23 11:54:51 +02:00
|
|
|
if (f->alliance) {
|
|
|
|
WRITE_INT(data->store, f->alliance->id);
|
|
|
|
if (f->alliance->flags & ALF_NON_ALLIED) {
|
|
|
|
assert(f == f->alliance->_leader
|
|
|
|
|| !"non-allied faction is not leader of its own dummy-alliance.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
WRITE_INT(data->store, 0);
|
|
|
|
}
|
|
|
|
WRITE_INT(data->store, f->alliance_joindate);
|
|
|
|
|
2016-01-12 06:46:51 +01:00
|
|
|
WRITE_STR(data->store, f->name);
|
2018-10-24 19:54:07 +02:00
|
|
|
WRITE_STR(data->store, faction_getbanner(f));
|
2017-11-14 16:20:31 +01:00
|
|
|
WRITE_STR(data->store, f->email?f->email:"");
|
2016-02-21 10:10:26 +01:00
|
|
|
write_password(data, f);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_TOK(data->store, locale_name(f->locale));
|
|
|
|
WRITE_INT(data->store, f->lastorders);
|
|
|
|
WRITE_INT(data->store, f->age);
|
2014-08-24 21:49:55 +02:00
|
|
|
WRITE_TOK(data->store, f->race->_name);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
WRITE_INT(data->store, f->magiegebiet);
|
|
|
|
|
|
|
|
WRITE_INT(data->store, f->flags & FFL_SAVEMASK);
|
2016-02-09 00:28:23 +01:00
|
|
|
write_attribs(data->store, f->attribs, f);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
write_items(data->store, f->items);
|
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
WRITE_TOK(data->store, "end");
|
|
|
|
WRITE_SECTION(data->store);
|
2018-10-22 21:51:11 +02:00
|
|
|
WRITE_INT(data->store, listlen(f->origin));
|
|
|
|
for (ur = f->origin; ur; ur = ur->next) {
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_INT(data->store, ur->id);
|
|
|
|
WRITE_INT(data->store, ur->x);
|
|
|
|
WRITE_INT(data->store, ur->y);
|
|
|
|
}
|
|
|
|
WRITE_SECTION(data->store);
|
2018-02-03 14:49:35 +01:00
|
|
|
WRITE_INT(data->store, f->options & ~WANT_OPTION(O_DEBUG));
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_SECTION(data->store);
|
|
|
|
|
2018-10-26 21:49:58 +02:00
|
|
|
write_allies(data, f->allies);
|
2014-07-23 11:54:51 +02:00
|
|
|
WRITE_SECTION(data->store);
|
2018-10-26 21:49:58 +02:00
|
|
|
write_groups(data, f);
|
2014-07-23 11:54:51 +02:00
|
|
|
write_spellbook(f->spellbook, data->store);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2015-08-19 19:43:47 +02:00
|
|
|
static int cb_sb_maxlevel(spellbook_entry *sbe, void *cbdata) {
|
|
|
|
faction *f = (faction *)cbdata;
|
|
|
|
if (sbe->level > f->max_spelllevel) {
|
|
|
|
f->max_spelllevel = sbe->level;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-11-14 01:35:45 +01:00
|
|
|
int readgame(const char *filename)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-12-09 03:42:08 +01:00
|
|
|
int n = -2, stream_version;
|
2017-12-29 11:44:14 +01:00
|
|
|
char path[PATH_MAX];
|
2014-07-23 11:54:51 +02:00
|
|
|
gamedata gdata = { 0 };
|
|
|
|
storage store;
|
2014-11-03 22:29:04 +01:00
|
|
|
stream strm;
|
2014-07-23 11:54:51 +02:00
|
|
|
FILE *F;
|
2015-11-11 15:54:14 +01:00
|
|
|
size_t sz;
|
2013-12-31 10:06:28 +01:00
|
|
|
|
2016-05-29 10:58:49 +02:00
|
|
|
log_debug("- reading game data from %s", filename);
|
2017-12-29 11:44:14 +01:00
|
|
|
path_join(datapath(), filename, path, sizeof(path));
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-02-05 23:10:05 +01:00
|
|
|
F = fopen(path, "rb");
|
2014-07-23 11:54:51 +02:00
|
|
|
if (!F) {
|
|
|
|
perror(path);
|
|
|
|
return -1;
|
|
|
|
}
|
2015-11-11 15:54:14 +01:00
|
|
|
sz = fread(&gdata.version, sizeof(int), 1, F);
|
2018-12-09 03:42:08 +01:00
|
|
|
if (sz == 1) {
|
|
|
|
sz = fread(&stream_version, sizeof(int), 1, F);
|
|
|
|
assert((sz == 1 && stream_version == STREAM_VERSION) || !"unsupported data format");
|
|
|
|
assert(gdata.version >= MIN_VERSION || !"unsupported data format");
|
|
|
|
assert(gdata.version <= MAX_VERSION || !"unsupported data format");
|
|
|
|
|
|
|
|
fstream_init(&strm, F);
|
|
|
|
binstore_init(&store, &strm);
|
|
|
|
gdata.store = &store;
|
|
|
|
|
|
|
|
if (gdata.version >= BUILDNO_VERSION) {
|
|
|
|
int build;
|
|
|
|
READ_INT(&store, &build);
|
|
|
|
log_debug("data in %s created with build %d.", filename, build);
|
|
|
|
}
|
|
|
|
n = read_game(&gdata);
|
|
|
|
binstore_done(&store);
|
|
|
|
fstream_done(&strm);
|
|
|
|
}
|
2019-02-15 10:07:55 +01:00
|
|
|
else {
|
|
|
|
fclose(F);
|
|
|
|
}
|
2016-02-25 19:12:38 +01:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2016-11-11 21:41:25 +01:00
|
|
|
void write_building(gamedata *data, const building *b)
|
|
|
|
{
|
|
|
|
storage *store = data->store;
|
|
|
|
|
|
|
|
write_building_reference(b, store);
|
|
|
|
WRITE_STR(store, b->name);
|
|
|
|
WRITE_STR(store, b->display ? b->display : "");
|
2020-07-16 19:41:45 +02:00
|
|
|
assert(b->size > 0);
|
2016-11-11 21:41:25 +01:00
|
|
|
WRITE_INT(store, b->size);
|
|
|
|
WRITE_TOK(store, b->type->_name);
|
|
|
|
write_attribs(store, b->attribs, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct building *read_building(gamedata *data) {
|
|
|
|
char name[DISPLAYSIZE];
|
|
|
|
building *b;
|
|
|
|
storage * store = data->store;
|
|
|
|
|
|
|
|
b = (building *)calloc(1, sizeof(building));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (!b) abort();
|
2016-11-11 21:41:25 +01:00
|
|
|
READ_INT(store, &b->no);
|
|
|
|
bhash(b);
|
|
|
|
READ_STR(store, name, sizeof(name));
|
2016-11-11 21:46:56 +01:00
|
|
|
if (unicode_utf8_trim(name)!=0) {
|
|
|
|
log_warning("trim building %s name to '%s'", itoa36(b->no), name);
|
|
|
|
}
|
2017-12-28 18:29:40 +01:00
|
|
|
b->name = str_strdup(name);
|
2017-11-25 20:24:57 +01:00
|
|
|
READ_STR(store, name, sizeof(name));
|
|
|
|
if (unicode_utf8_trim(name)!=0) {
|
|
|
|
log_warning("trim building %s info to '%s'", itoa36(b->no), name);
|
2016-11-11 21:41:25 +01:00
|
|
|
}
|
2017-12-28 18:29:40 +01:00
|
|
|
b->display = str_strdup(name);
|
2016-11-11 21:41:25 +01:00
|
|
|
READ_INT(store, &b->size);
|
2020-07-16 19:41:45 +02:00
|
|
|
if (b->size < 1) {
|
|
|
|
log_warning("trim building %s had size %d", itoa36(b->no), b->size);
|
|
|
|
b->size = 1;
|
|
|
|
}
|
2016-11-11 21:41:25 +01:00
|
|
|
READ_STR(store, name, sizeof(name));
|
|
|
|
b->type = bt_find(name);
|
2017-02-24 14:29:14 +01:00
|
|
|
if (!b->type) {
|
|
|
|
log_error("building %d has unknown type %s", b->no, name);
|
|
|
|
b->type = bt_find("building");
|
|
|
|
assert(b->type);
|
|
|
|
}
|
2016-11-11 21:41:25 +01:00
|
|
|
read_attribs(data, &b->attribs, b);
|
|
|
|
|
2017-02-18 21:15:14 +01:00
|
|
|
/* repairs, bug 2221: */
|
2016-11-11 21:41:25 +01:00
|
|
|
if (b->type->maxsize>0 && b->size>b->type->maxsize) {
|
|
|
|
log_error("building too big: %s (%s size %d of %d), fixing.", buildingname(b), b->type->_name, b->size, b->type->maxsize);
|
|
|
|
b->size = b->type->maxsize;
|
|
|
|
}
|
2017-09-21 16:26:53 +02:00
|
|
|
resolve_building(b);
|
|
|
|
return b;
|
2016-11-11 21:41:25 +01:00
|
|
|
}
|
|
|
|
|
2016-11-11 22:25:56 +01:00
|
|
|
void write_ship(gamedata *data, const ship *sh)
|
|
|
|
{
|
|
|
|
storage *store = data->store;
|
|
|
|
write_ship_reference(sh, store);
|
|
|
|
WRITE_STR(store, (const char *)sh->name);
|
|
|
|
WRITE_STR(store, sh->display ? (const char *)sh->display : "");
|
|
|
|
WRITE_TOK(store, sh->type->_name);
|
2019-10-06 18:11:10 +02:00
|
|
|
assert(sh->number > 0);
|
2019-10-05 20:31:00 +02:00
|
|
|
WRITE_INT(store, sh->number);
|
2016-11-11 22:25:56 +01:00
|
|
|
WRITE_INT(store, sh->size);
|
|
|
|
WRITE_INT(store, sh->damage);
|
|
|
|
WRITE_INT(store, sh->flags & SFL_SAVEMASK);
|
|
|
|
assert((sh->type->flags & SFL_NOCOAST) == 0 || sh->coast == NODIRECTION);
|
|
|
|
WRITE_INT(store, sh->coast);
|
|
|
|
write_attribs(store, sh->attribs, sh);
|
|
|
|
}
|
|
|
|
|
2017-09-16 07:58:57 +02:00
|
|
|
ship *read_ship(gamedata *data)
|
2016-11-11 22:25:56 +01:00
|
|
|
{
|
2016-02-25 19:12:38 +01:00
|
|
|
char name[DISPLAYSIZE];
|
2016-11-11 22:25:56 +01:00
|
|
|
ship *sh;
|
|
|
|
int n;
|
|
|
|
storage *store = data->store;
|
|
|
|
|
|
|
|
sh = (ship *)calloc(1, sizeof(ship));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (!sh) abort();
|
2016-11-11 22:25:56 +01:00
|
|
|
READ_INT(store, &sh->no);
|
|
|
|
shash(sh);
|
|
|
|
READ_STR(store, name, sizeof(name));
|
2016-11-11 22:28:22 +01:00
|
|
|
if (unicode_utf8_trim(name)!=0) {
|
|
|
|
log_warning("trim ship %s name to '%s'", itoa36(sh->no), name);
|
|
|
|
}
|
2017-12-28 18:29:40 +01:00
|
|
|
sh->name = str_strdup(name);
|
2017-11-25 20:24:57 +01:00
|
|
|
READ_STR(store, name, sizeof(name));
|
|
|
|
if (unicode_utf8_trim(name)!=0) {
|
|
|
|
log_warning("trim ship %s info to '%s'", itoa36(sh->no), name);
|
2016-11-11 22:25:56 +01:00
|
|
|
}
|
2017-12-28 18:29:40 +01:00
|
|
|
sh->display = str_strdup(name);
|
2016-11-11 22:25:56 +01:00
|
|
|
READ_STR(store, name, sizeof(name));
|
|
|
|
sh->type = st_find(name);
|
|
|
|
if (sh->type == NULL) {
|
|
|
|
/* old datafiles */
|
|
|
|
sh->type = st_find((const char *)LOC(default_locale, name));
|
|
|
|
}
|
|
|
|
assert(sh->type || !"ship_type not registered!");
|
|
|
|
|
2020-08-03 10:31:58 +02:00
|
|
|
if (data->version < SHIP_NUMBER_VERSION) {
|
2019-10-05 20:31:00 +02:00
|
|
|
sh->number = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
READ_INT(store, &sh->number);
|
|
|
|
}
|
2016-11-11 22:25:56 +01:00
|
|
|
READ_INT(store, &sh->size);
|
|
|
|
READ_INT(store, &sh->damage);
|
|
|
|
if (data->version >= FOSS_VERSION) {
|
|
|
|
READ_INT(store, &sh->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Attribute rekursiv einlesen */
|
|
|
|
|
|
|
|
READ_INT(store, &n);
|
|
|
|
sh->coast = (direction_t)n;
|
|
|
|
if (sh->type->flags & SFL_NOCOAST) {
|
|
|
|
sh->coast = NODIRECTION;
|
|
|
|
}
|
|
|
|
read_attribs(data, &sh->attribs, sh);
|
|
|
|
return sh;
|
|
|
|
}
|
|
|
|
|
2018-11-17 21:24:59 +01:00
|
|
|
static void fix_fam_triggers(unit *u) {
|
|
|
|
attrib * a = a_find(u->attribs, &at_mage);
|
|
|
|
attrib * am = a_find(u->attribs, &at_familiarmage);
|
|
|
|
if (!am && a) {
|
|
|
|
/* not a familiar, but magical */
|
|
|
|
attrib * ae = a_find(u->attribs, &at_eventhandler);
|
|
|
|
if (ae) {
|
|
|
|
trigger **tlist;
|
|
|
|
tlist = get_triggers(ae, "destroy");
|
|
|
|
if (tlist) {
|
|
|
|
trigger *t;
|
|
|
|
unit *um = NULL;
|
|
|
|
for (t = *tlist; t; t = t->next) {
|
|
|
|
if (t->type == &tt_shock) {
|
|
|
|
um = (unit *)t->data.v;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (um) {
|
|
|
|
attrib *af = a_find(um->attribs, &at_familiar);
|
|
|
|
log_error("%s seems to be a broken familiar of %s.",
|
|
|
|
unitname(u), unitname(um));
|
|
|
|
if (af) {
|
|
|
|
unit * uf = (unit *)af->data.v;
|
|
|
|
log_error("%s already has a familiar: %s.",
|
|
|
|
unitname(um), unitname(uf));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
set_familiar(um, u);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error("%s seems to be a broken familiar with no trigger.", unitname(u));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-11-11 22:25:56 +01:00
|
|
|
|
2019-04-07 21:37:06 +02:00
|
|
|
static void fix_clone(unit *uc) {
|
|
|
|
attrib * a;
|
|
|
|
assert(uc);
|
|
|
|
assert(uc->number > 0);
|
|
|
|
ADDMSG(&uc->faction->msgs, msg_message("dissolve_units_5",
|
|
|
|
"unit region number race", uc, uc->region, uc->number, u_race(uc)));
|
|
|
|
a_removeall(&uc->attribs, &at_clonemage);
|
|
|
|
a = a_new(&at_unitdissolve);
|
|
|
|
a->data.ca[0] = 0;
|
|
|
|
a->data.ca[1] = 100;
|
|
|
|
a_add(&uc->attribs, a);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fix_clone_mage(unit *um, const item_type *itype) {
|
|
|
|
i_change(&um->items, itype, 1);
|
|
|
|
change_maxspellpoints(um, 20);
|
|
|
|
a_removeall(&um->attribs, &at_clone);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fix_clones(void) {
|
|
|
|
const race *rc_clone = rc_find("clone");
|
|
|
|
const item_type *it_potion = it_find("lifepotion");
|
|
|
|
|
|
|
|
if (rc_clone && it_potion) {
|
|
|
|
region *r;
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
unit * u;
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
if (!fval(u, UFL_MARK)) {
|
|
|
|
if (u_race(u) == rc_clone) {
|
|
|
|
attrib *a = a_find(u->attribs, &at_clonemage);
|
|
|
|
unit * um = NULL;
|
|
|
|
fset(u, UFL_MARK);
|
|
|
|
if (a) {
|
|
|
|
um = (unit *)a->data.v;
|
|
|
|
fset(um, UFL_MARK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
attrib *a = a_find(u->attribs, &at_clone);
|
|
|
|
if (a) {
|
|
|
|
unit *uc = (unit *)a->data.v;
|
|
|
|
fset(u, UFL_MARK);
|
|
|
|
fset(uc, UFL_MARK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
unit * u;
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
if (fval(u, UFL_MARK)) {
|
|
|
|
if (u_race(u) == rc_clone) {
|
|
|
|
fix_clone(u);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
fix_clone_mage(u, it_potion);
|
|
|
|
}
|
|
|
|
freset(u, UFL_MARK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-17 21:24:59 +01:00
|
|
|
static void fix_familiars(void (*callback)(unit *)) {
|
2017-08-31 21:19:25 +02:00
|
|
|
region *r;
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
unit * u;
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
if (u->_race != u->faction->race && (u->_race->flags & RCF_FAMILIAR)) {
|
|
|
|
/* unit is potentially a familiar */
|
2018-11-17 21:24:59 +01:00
|
|
|
callback(u);
|
2017-08-31 21:19:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-25 18:11:14 +02:00
|
|
|
static void read_regions(gamedata *data) {
|
2016-02-25 19:12:38 +01:00
|
|
|
storage * store = data->store;
|
2016-11-11 21:41:25 +01:00
|
|
|
const struct building_type *bt_lighthouse = bt_find("lighthouse");
|
2017-05-24 08:18:55 +02:00
|
|
|
const struct race *rc_spell = rc_find("spell");
|
2019-08-25 18:11:14 +02:00
|
|
|
region *r;
|
2019-08-25 17:55:04 +02:00
|
|
|
int nread;
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
READ_INT(store, &nread);
|
2019-08-25 17:55:04 +02:00
|
|
|
assert(nread < MAXREGIONS && nread >= 0);
|
2017-12-08 20:27:43 +01:00
|
|
|
|
|
|
|
log_debug(" - Einzulesende Regionen: %d", nread);
|
2016-11-12 01:03:07 +01:00
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
while (--nread >= 0) {
|
|
|
|
unit **up;
|
2018-12-09 03:42:08 +01:00
|
|
|
building **bp;
|
|
|
|
ship **shp;
|
2019-08-25 17:55:04 +02:00
|
|
|
int p;
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-11-12 01:03:07 +01:00
|
|
|
r = read_region(data);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
|
|
|
/* Burgen */
|
2016-02-25 19:12:38 +01:00
|
|
|
READ_INT(store, &p);
|
2017-09-16 08:42:49 +02:00
|
|
|
if (p > 0 && !r->land) {
|
2019-08-25 17:55:04 +02:00
|
|
|
log_debug("%s, uid=%d has %d %s", regionname(r, NULL), r->uid, p, (p == 1) ? "building" : "buildings");
|
2017-09-16 08:42:49 +02:00
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
bp = &r->buildings;
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2016-11-11 21:41:25 +01:00
|
|
|
building *b = *bp = read_building(data);
|
2014-07-23 11:54:51 +02:00
|
|
|
if (b->type == bt_lighthouse) {
|
|
|
|
r->flags |= RF_LIGHTHOUSE;
|
|
|
|
}
|
2016-11-11 22:25:56 +01:00
|
|
|
b->region = r;
|
2016-11-11 21:41:25 +01:00
|
|
|
bp = &b->next;
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
/* Schiffe */
|
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
READ_INT(store, &p);
|
2014-07-23 11:54:51 +02:00
|
|
|
shp = &r->ships;
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2016-11-11 22:25:56 +01:00
|
|
|
ship *sh = *shp = read_ship(data);
|
2014-07-23 11:54:51 +02:00
|
|
|
sh->region = r;
|
|
|
|
shp = &sh->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
*shp = 0;
|
|
|
|
|
|
|
|
/* Einheiten */
|
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
READ_INT(store, &p);
|
2014-07-23 11:54:51 +02:00
|
|
|
up = &r->units;
|
|
|
|
|
|
|
|
while (--p >= 0) {
|
2016-02-25 19:12:38 +01:00
|
|
|
unit *u = read_unit(data);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2017-05-24 08:52:19 +02:00
|
|
|
if (data->version < NORCSPELL_VERSION && u_race(u) == rc_spell) {
|
2017-05-24 08:18:55 +02:00
|
|
|
set_observer(r, u->faction, get_level(u, SK_PERCEPTION), u->age);
|
2017-06-04 14:12:23 +02:00
|
|
|
u_setfaction(u, NULL);
|
2017-05-24 08:18:55 +02:00
|
|
|
free_unit(u);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (data->version < JSON_REPORT_VERSION) {
|
|
|
|
if (u->_name && fval(u->faction, FFL_NPC)) {
|
|
|
|
if (!u->_name[0] || unit_name_equals_race(u)) {
|
|
|
|
unit_setname(u, NULL);
|
|
|
|
}
|
2014-12-09 07:44:26 +01:00
|
|
|
}
|
|
|
|
}
|
2017-05-24 08:18:55 +02:00
|
|
|
assert(u->region == NULL);
|
|
|
|
u->region = r;
|
|
|
|
*up = u;
|
|
|
|
up = &u->next;
|
|
|
|
update_interval(u->faction, r);
|
2014-12-09 07:44:26 +01:00
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-29 10:58:49 +02:00
|
|
|
log_debug("updating area information for lighthouses.");
|
2014-07-23 11:54:51 +02:00
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
if (r->flags & RF_LIGHTHOUSE) {
|
|
|
|
building *b;
|
2016-09-17 19:52:13 +02:00
|
|
|
for (b = r->buildings; b; b = b->next) {
|
2018-07-22 10:55:09 +02:00
|
|
|
if (is_lighthouse(b->type)) {
|
|
|
|
update_lighthouse(b);
|
|
|
|
}
|
2016-09-17 19:52:13 +02:00
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-25 18:11:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void init_factions(int data_version)
|
|
|
|
{
|
2016-05-29 10:58:49 +02:00
|
|
|
log_debug("marking factions as alive.");
|
2019-08-25 18:11:14 +02:00
|
|
|
for (faction *f = factions; f; f = f->next) {
|
2014-07-23 11:54:51 +02:00
|
|
|
if (f->flags & FFL_NPC) {
|
2016-01-11 11:54:45 +01:00
|
|
|
f->_alive = true;
|
2015-08-02 22:08:35 +02:00
|
|
|
f->magiegebiet = M_GRAY;
|
2014-07-23 11:54:51 +02:00
|
|
|
if (f->no == 0) {
|
|
|
|
int no = 666;
|
|
|
|
while (findfaction(no))
|
|
|
|
++no;
|
2016-05-29 10:58:49 +02:00
|
|
|
log_warning("renum(monsters, %d)", no);
|
2014-07-23 11:54:51 +02:00
|
|
|
renumber_faction(f, no);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2018-10-30 06:30:32 +01:00
|
|
|
assert(f->units);
|
2019-08-25 18:11:14 +02:00
|
|
|
for (unit *u = f->units; u; u = u->nextF) {
|
|
|
|
if (data_version < SPELL_LEVEL_VERSION) {
|
2018-11-18 21:47:14 +01:00
|
|
|
struct sc_mage *mage = get_mage(u);
|
2015-08-19 19:43:47 +02:00
|
|
|
if (mage) {
|
|
|
|
faction *f = u->faction;
|
2019-02-02 20:36:23 +01:00
|
|
|
int skl = effskill(u, SK_MAGIC, NULL);
|
2015-08-19 19:43:47 +02:00
|
|
|
if (f->magiegebiet == M_GRAY) {
|
2018-11-18 21:47:14 +01:00
|
|
|
f->magiegebiet = mage_get_type(mage);
|
|
|
|
log_error("faction %s had magic=gray, fixing (%s)",
|
|
|
|
factionname(f), magic_school[f->magiegebiet]);
|
2015-08-19 19:43:47 +02:00
|
|
|
}
|
|
|
|
if (f->max_spelllevel < skl) {
|
|
|
|
f->max_spelllevel = skl;
|
|
|
|
}
|
2015-08-02 22:08:35 +02:00
|
|
|
}
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
if (u->number > 0) {
|
2016-01-11 11:54:45 +01:00
|
|
|
f->_alive = true;
|
2019-08-25 18:11:14 +02:00
|
|
|
if (data_version >= SPELL_LEVEL_VERSION) {
|
2015-08-19 19:43:47 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-25 18:11:14 +02:00
|
|
|
if (data_version < SPELL_LEVEL_VERSION && f->spellbook) {
|
2015-08-19 19:43:47 +02:00
|
|
|
spellbook_foreach(f->spellbook, cb_sb_maxlevel, f);
|
|
|
|
}
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-25 18:11:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void read_factions(gamedata * data)
|
|
|
|
{
|
|
|
|
storage * store = data->store;
|
|
|
|
int nread;
|
|
|
|
faction **fp;
|
|
|
|
READ_INT(store, &nread);
|
|
|
|
log_debug(" - Einzulesende Parteien: %d\n", nread);
|
|
|
|
fp = &factions;
|
|
|
|
while (*fp) {
|
|
|
|
fp = &(*fp)->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (--nread >= 0) {
|
|
|
|
faction *f = read_faction(data);
|
|
|
|
|
|
|
|
*fp = f;
|
|
|
|
fp = &f->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int read_game(gamedata *data)
|
|
|
|
{
|
|
|
|
storage * store = data->store;
|
|
|
|
|
|
|
|
if (data->version >= SAVEGAMEID_VERSION) {
|
|
|
|
int gameid;
|
|
|
|
|
|
|
|
READ_INT(store, &gameid);
|
|
|
|
if (gameid != game_id()) {
|
|
|
|
log_warning("game mismatch: datafile contains game %d, but config is for %d", gameid, game_id());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
READ_STR(store, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->version < FIXATKEYS_VERSION) {
|
|
|
|
attrib *a = NULL;
|
|
|
|
read_attribs(data, &a, NULL);
|
|
|
|
a_removeall(&a, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
READ_INT(store, &turn);
|
|
|
|
log_debug(" - reading turn %d", turn);
|
|
|
|
rng_init(turn + config_get_int("game.seed", 0));
|
|
|
|
READ_INT(store, NULL); /* max_unique_id = ignore */
|
|
|
|
READ_INT(store, &nextborder);
|
|
|
|
|
|
|
|
read_planes(data);
|
|
|
|
read_alliances(data);
|
|
|
|
|
|
|
|
read_factions(data);
|
|
|
|
|
|
|
|
/* Regionen */
|
|
|
|
|
|
|
|
read_regions(data);
|
|
|
|
read_borders(data);
|
|
|
|
init_factions(data->version);
|
2019-04-07 21:37:06 +02:00
|
|
|
if (data->version < FIX_CLONES_VERSION) {
|
|
|
|
fix_clones();
|
|
|
|
}
|
2017-08-31 21:19:25 +02:00
|
|
|
if (data->version < FAMILIAR_FIX_VERSION) {
|
2018-11-17 21:24:59 +01:00
|
|
|
fix_familiars(fix_fam_triggers);
|
|
|
|
}
|
2018-12-09 20:05:44 +01:00
|
|
|
if (data->version < FAMILIAR_FIXSPELLBOOK_VERSION) {
|
2019-05-19 15:14:01 +02:00
|
|
|
fix_familiars(fix_fam_spells);
|
2017-08-31 21:19:25 +02:00
|
|
|
}
|
2019-05-19 15:47:13 +02:00
|
|
|
if (data->version < FIX_MIGRANT_AURA_VERSION) {
|
|
|
|
fix_familiars(fix_fam_migrant);
|
|
|
|
}
|
2017-08-31 21:19:25 +02:00
|
|
|
|
2016-05-29 10:58:49 +02:00
|
|
|
log_debug("Done loading turn %d.", turn);
|
2015-08-20 12:13:09 +02:00
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-11-11 16:53:56 +01:00
|
|
|
static void clear_npc_orders(faction *f)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
if (f) {
|
|
|
|
unit *u;
|
|
|
|
for (u = f->units; u; u = u->nextF) {
|
|
|
|
free_orders(&u->orders);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-31 10:06:28 +01:00
|
|
|
int writegame(const char *filename)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-07-23 11:54:51 +02:00
|
|
|
int n;
|
2017-12-29 11:44:14 +01:00
|
|
|
char path[PATH_MAX];
|
2014-07-23 11:54:51 +02:00
|
|
|
gamedata gdata;
|
|
|
|
storage store;
|
2014-11-03 22:29:04 +01:00
|
|
|
stream strm;
|
2014-07-23 11:54:51 +02:00
|
|
|
FILE *F;
|
|
|
|
|
2016-02-01 09:26:24 +01:00
|
|
|
create_directories();
|
2017-12-29 11:44:14 +01:00
|
|
|
path_join(datapath(), filename, path, sizeof(path));
|
2015-11-05 13:22:51 +01:00
|
|
|
/* make sure we don't overwrite an existing file (hard links) */
|
2017-12-12 21:37:44 +01:00
|
|
|
if (remove(path) != 0) {
|
|
|
|
if (errno == ENOENT) {
|
2016-09-11 17:19:27 +02:00
|
|
|
errno = 0;
|
|
|
|
}
|
|
|
|
}
|
2016-02-05 23:10:05 +01:00
|
|
|
F = fopen(path, "wb");
|
2013-12-31 10:06:28 +01:00
|
|
|
if (!F) {
|
2016-02-01 09:26:24 +01:00
|
|
|
perror(path);
|
|
|
|
return -1;
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
gdata.store = &store;
|
|
|
|
gdata.version = RELEASE_VERSION;
|
|
|
|
fwrite(&gdata.version, sizeof(int), 1, F);
|
2016-02-25 19:12:38 +01:00
|
|
|
n = STREAM_VERSION;
|
2014-07-23 11:54:51 +02:00
|
|
|
fwrite(&n, sizeof(int), 1, F);
|
|
|
|
|
2014-11-03 22:29:04 +01:00
|
|
|
fstream_init(&strm, F);
|
|
|
|
binstore_init(&store, &strm);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-09-15 20:11:38 +02:00
|
|
|
WRITE_INT(&store, version_no(eressea_version()));
|
2016-02-25 19:12:38 +01:00
|
|
|
n = write_game(&gdata);
|
|
|
|
binstore_done(&store);
|
|
|
|
fstream_done(&strm);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
int write_game(gamedata *data) {
|
|
|
|
storage * store = data->store;
|
|
|
|
region *r;
|
|
|
|
faction *f;
|
|
|
|
int n;
|
|
|
|
|
2014-07-23 11:54:51 +02:00
|
|
|
/* globale Variablen */
|
2016-02-25 19:12:38 +01:00
|
|
|
assert(data->version <= MAX_VERSION && data->version >= MIN_VERSION);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_INT(store, game_id());
|
|
|
|
WRITE_SECTION(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_INT(store, turn);
|
2016-02-26 19:01:28 +01:00
|
|
|
WRITE_INT(store, 0 /* max_unique_id */);
|
2016-03-13 09:49:04 +01:00
|
|
|
WRITE_INT(store, nextborder);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
write_planes(store);
|
|
|
|
write_alliances(data);
|
2014-07-23 11:54:51 +02:00
|
|
|
n = listlen(factions);
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_INT(store, n);
|
|
|
|
WRITE_SECTION(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-05-29 10:58:49 +02:00
|
|
|
log_debug(" - Schreibe %d Parteien...", n);
|
2014-07-23 11:54:51 +02:00
|
|
|
for (f = factions; f; f = f->next) {
|
2014-11-11 16:53:56 +01:00
|
|
|
if (fval(f, FFL_NPC)) {
|
|
|
|
clear_npc_orders(f);
|
|
|
|
}
|
2016-11-12 00:47:25 +01:00
|
|
|
write_faction(data, f);
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_SECTION(store);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Write regions */
|
|
|
|
|
|
|
|
n = listlen(regions);
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_INT(store, n);
|
|
|
|
WRITE_SECTION(store);
|
2015-09-26 22:42:22 +02:00
|
|
|
log_debug(" - Schreibe Regionen: %d", n);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
|
|
|
for (r = regions; r; r = r->next, --n) {
|
2016-02-25 19:12:38 +01:00
|
|
|
ship *sh;
|
|
|
|
building *b;
|
|
|
|
unit *u;
|
2014-07-23 11:54:51 +02:00
|
|
|
/* plus leerzeile */
|
|
|
|
if ((n % 1024) == 0) { /* das spart extrem Zeit */
|
2015-09-26 22:42:22 +02:00
|
|
|
log_debug(" - Schreibe Regionen: %d", n);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_SECTION(store);
|
2016-11-12 01:03:07 +01:00
|
|
|
write_region(data, r);
|
2014-07-23 11:54:51 +02:00
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_INT(store, listlen(r->buildings));
|
|
|
|
WRITE_SECTION(store);
|
2014-07-23 11:54:51 +02:00
|
|
|
for (b = r->buildings; b; b = b->next) {
|
2016-11-11 22:25:56 +01:00
|
|
|
assert(b->region == r);
|
2016-11-11 21:41:25 +01:00
|
|
|
write_building(data, b);
|
2016-02-25 19:12:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
WRITE_INT(store, listlen(r->ships));
|
|
|
|
WRITE_SECTION(store);
|
2014-07-23 11:54:51 +02:00
|
|
|
for (sh = r->ships; sh; sh = sh->next) {
|
|
|
|
assert(sh->region == r);
|
2016-11-11 22:25:56 +01:00
|
|
|
write_ship(data, sh);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_INT(store, listlen(r->units));
|
|
|
|
WRITE_SECTION(store);
|
2014-07-23 11:54:51 +02:00
|
|
|
for (u = r->units; u; u = u->next) {
|
2016-11-11 22:25:56 +01:00
|
|
|
assert(u->region == r);
|
2016-02-25 19:12:38 +01:00
|
|
|
write_unit(data, u);
|
2014-07-23 11:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
2016-02-25 19:12:38 +01:00
|
|
|
WRITE_SECTION(store);
|
|
|
|
write_borders(store);
|
|
|
|
WRITE_SECTION(store);
|
2014-07-23 11:54:51 +02:00
|
|
|
return 0;
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|