2001-01-25 10:37:55 +01:00
|
|
|
/* vi: set ts=2:
|
|
|
|
*
|
2001-04-14 13:39:14 +02:00
|
|
|
*
|
2003-07-29 11:48:03 +02:00
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
2001-01-25 10:37:55 +01:00
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
2007-09-02 20:11:17 +02:00
|
|
|
* Enno Rehling (enno@eressea.de)
|
2001-01-25 10:37:55 +01:00
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
*
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
2008-04-20 16:48:15 +02:00
|
|
|
#include <kernel/eressea.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
#include "group.h"
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include "unit.h"
|
|
|
|
#include "faction.h"
|
|
|
|
#include "save.h"
|
2007-06-20 02:34:02 +02:00
|
|
|
#include "version.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
/* attrib includes */
|
|
|
|
#include <attributes/raceprefix.h>
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
/* util includes */
|
2007-06-20 02:34:02 +02:00
|
|
|
#include <util/attrib.h>
|
|
|
|
#include <util/base36.h>
|
|
|
|
#include <util/resolve.h>
|
2008-04-26 16:55:22 +02:00
|
|
|
#include <util/storage.h>
|
2007-08-09 08:42:37 +02:00
|
|
|
#include <util/unicode.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
/* libc includes */
|
2008-04-27 13:58:14 +02:00
|
|
|
#include <assert.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2007-08-09 08:42:37 +02:00
|
|
|
#include <wctype.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
2004-05-26 08:42:58 +02:00
|
|
|
#define GMAXHASH 2039
|
2001-01-25 10:37:55 +01:00
|
|
|
static group * ghash[GMAXHASH];
|
|
|
|
static int maxgid;
|
|
|
|
|
|
|
|
static group *
|
2007-08-10 09:03:23 +02:00
|
|
|
new_group(faction * f, const char * name, int gid)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
|
|
|
group ** gp = &f->groups;
|
|
|
|
int index = gid % GMAXHASH;
|
|
|
|
group * g = calloc(sizeof(group), 1);
|
|
|
|
|
|
|
|
while (*gp) gp = &(*gp)->next;
|
|
|
|
*gp = g;
|
|
|
|
|
|
|
|
maxgid = max(gid, maxgid);
|
2007-08-10 09:03:23 +02:00
|
|
|
g->name = strdup(name);
|
2001-01-25 10:37:55 +01:00
|
|
|
g->gid = gid;
|
|
|
|
|
|
|
|
g->nexthash = ghash[index];
|
|
|
|
return ghash[index] = g;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
init_group(faction * f, group * g)
|
|
|
|
{
|
|
|
|
ally * a, ** an;
|
|
|
|
|
|
|
|
an = &g->allies;
|
|
|
|
for (a=f->allies;a;a=a->next) if (a->faction) {
|
|
|
|
ally * ga = calloc(sizeof(ally), 1);
|
|
|
|
*ga = *a;
|
|
|
|
*an = ga;
|
|
|
|
an = &ga->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static group *
|
2007-08-10 09:03:23 +02:00
|
|
|
find_groupbyname(group * g, const char * name)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
2007-08-09 08:42:37 +02:00
|
|
|
while (g && unicode_utf8_strcasecmp(name, g->name)!=0) g = g->next;
|
2001-01-25 10:37:55 +01:00
|
|
|
return g;
|
|
|
|
}
|
|
|
|
|
|
|
|
static group *
|
|
|
|
find_group(int gid)
|
|
|
|
{
|
|
|
|
int index = gid % GMAXHASH;
|
|
|
|
group * g = ghash[index];
|
|
|
|
while (g && g->gid!=gid) g = g->nexthash;
|
|
|
|
return g;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2008-04-26 16:55:22 +02:00
|
|
|
read_group(attrib * a, struct storage * store)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
group * g;
|
|
|
|
int gid = store->r_int(store);
|
|
|
|
a->data.v = g = find_group(gid);
|
|
|
|
if (g!=0) {
|
|
|
|
g->members++;
|
|
|
|
return AT_READ_OK;
|
|
|
|
}
|
|
|
|
return AT_READ_FAIL;
|
2001-01-25 10:37:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2008-04-26 16:55:22 +02:00
|
|
|
write_group(const attrib * a, struct storage * store)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
group * g = (group*)a->data.v;
|
|
|
|
store->w_int(store, g->gid);
|
2001-01-25 10:37:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
attrib_type
|
|
|
|
at_group = { /* attribute for units assigned to a group */
|
|
|
|
"grp",
|
|
|
|
DEFAULT_INIT,
|
|
|
|
DEFAULT_FINALIZE,
|
|
|
|
DEFAULT_AGE,
|
|
|
|
write_group,
|
|
|
|
read_group,
|
|
|
|
ATF_UNIQUE
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
free_group(group * g)
|
|
|
|
{
|
|
|
|
while (g->allies) {
|
|
|
|
ally * a = g->allies;
|
|
|
|
g->allies = a->next;
|
|
|
|
free(a);
|
|
|
|
}
|
|
|
|
free(g->name);
|
|
|
|
free(g);
|
|
|
|
}
|
|
|
|
|
2007-08-26 22:33:50 +02:00
|
|
|
void
|
|
|
|
set_group(struct unit * u, struct group * g)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
2007-08-26 22:33:50 +02:00
|
|
|
attrib * a = NULL;
|
|
|
|
|
2005-06-12 19:57:14 +02:00
|
|
|
if (fval(u, UFL_GROUP)) {
|
|
|
|
a = a_find(u->attribs, &at_group);
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
2007-08-26 22:33:50 +02:00
|
|
|
if (a) {
|
|
|
|
group * og = (group *)a->data.v;
|
|
|
|
if (og==g) return;
|
|
|
|
--og->members;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g) {
|
|
|
|
if (!a) {
|
|
|
|
a = a_add(&u->attribs, a_new(&at_group));
|
|
|
|
fset(u, UFL_GROUP);
|
2005-06-12 19:57:14 +02:00
|
|
|
}
|
2007-08-26 22:33:50 +02:00
|
|
|
a->data.v = g;
|
|
|
|
g->members++;
|
|
|
|
} else if (a) {
|
|
|
|
a_remove(&u->attribs, a);
|
|
|
|
freset(u, UFL_GROUP);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean
|
|
|
|
join_group(unit * u, const char * name)
|
|
|
|
{
|
2008-02-03 16:48:38 +01:00
|
|
|
group * g = NULL;
|
|
|
|
|
|
|
|
if (name && name[0]) {
|
|
|
|
g = find_groupbyname(u->faction->groups, name);
|
|
|
|
if (g==NULL) {
|
|
|
|
g = new_group(u->faction, name, ++maxgid);
|
|
|
|
init_group(u->faction, g);
|
|
|
|
}
|
2005-06-12 19:57:14 +02:00
|
|
|
}
|
2007-08-26 22:33:50 +02:00
|
|
|
|
|
|
|
set_group(u, g);
|
2008-02-03 16:48:38 +01:00
|
|
|
return true;
|
2001-01-25 10:37:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
write_groups(struct storage * store, group * g)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
while (g) {
|
|
|
|
ally * a;
|
|
|
|
store->w_int(store, g->gid);
|
|
|
|
store->w_str(store, g->name);
|
|
|
|
for (a=g->allies;a;a=a->next) {
|
|
|
|
if (a->faction) {
|
|
|
|
write_faction_reference(a->faction, store);
|
|
|
|
store->w_int(store, a->status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
store->w_id(store, -1);
|
|
|
|
a_write(store, g->attribs);
|
|
|
|
store->w_brk(store);
|
|
|
|
g=g->next;
|
|
|
|
}
|
|
|
|
store->w_int(store, 0);
|
2001-01-25 10:37:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
read_groups(struct storage * store, faction * f)
|
2001-01-25 10:37:55 +01:00
|
|
|
{
|
2008-04-26 16:55:22 +02:00
|
|
|
for(;;) {
|
|
|
|
ally ** pa;
|
|
|
|
group * g;
|
|
|
|
int gid;
|
2007-06-27 02:08:46 +02:00
|
|
|
char buf[1024];
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
gid = store->r_int(store);
|
|
|
|
if (gid==0) break;
|
|
|
|
store->r_str_buf(store, buf, sizeof(buf));
|
|
|
|
g = new_group(f, buf, gid);
|
|
|
|
pa = &g->allies;
|
|
|
|
for (;;) {
|
|
|
|
ally * a;
|
|
|
|
variant fid;
|
|
|
|
fid.i = store->r_id(store);
|
2008-04-27 13:58:14 +02:00
|
|
|
if (fid.i<=0) break;
|
2008-04-26 16:55:22 +02:00
|
|
|
if (store->version<STORAGE_VERSION && fid.i==0) break;
|
|
|
|
a = malloc(sizeof(ally));
|
|
|
|
*pa = a;
|
|
|
|
pa = &a->next;
|
|
|
|
a->status = store->r_int(store);
|
|
|
|
a->faction = findfaction(fid.i);
|
|
|
|
if (!a->faction) ur_add(fid, (void**)&a->faction, resolve_faction);
|
|
|
|
}
|
2007-08-15 08:27:40 +02:00
|
|
|
*pa = 0;
|
2008-04-26 16:55:22 +02:00
|
|
|
a_read(store, &g->attribs);
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
}
|