2010-08-08 10:06:34 +02:00
|
|
|
/*
|
2015-01-30 22:10:29 +01:00
|
|
|
Copyright (c) 1998-2015, Enno Rehling <enno@eressea.de>
|
2014-10-31 15:13:05 +01:00
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
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 "group.h"
|
|
|
|
|
|
|
|
/* kernel includes */
|
2012-06-30 20:07:28 +02:00
|
|
|
#include "ally.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
#include "faction.h"
|
2012-06-30 20:07:28 +02:00
|
|
|
#include "unit.h"
|
2017-09-22 17:19:55 +02:00
|
|
|
#include "objtypes.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
/* attrib includes */
|
|
|
|
#include <attributes/raceprefix.h>
|
|
|
|
|
|
|
|
/* util includes */
|
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 13:21:46 +02:00
|
|
|
#include <kernel/gamedata.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/resolve.h>
|
2017-12-28 18:29:40 +01:00
|
|
|
#include <util/strings.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/unicode.h>
|
|
|
|
|
2013-12-31 10:06:28 +01:00
|
|
|
#include <storage.h>
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
/* libc includes */
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <wctype.h>
|
|
|
|
|
|
|
|
#define GMAXHASH 2039
|
2011-03-07 08:02:35 +01:00
|
|
|
static group *ghash[GMAXHASH];
|
2010-08-08 10:06:34 +02:00
|
|
|
static int maxgid;
|
|
|
|
|
2014-11-03 18:10:05 +01:00
|
|
|
group *new_group(faction * f, const char *name, int gid)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-10-31 15:13:05 +01:00
|
|
|
group **gp = &f->groups;
|
|
|
|
int index = gid % GMAXHASH;
|
|
|
|
group *g = calloc(sizeof(group), 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-10-31 15:13:05 +01:00
|
|
|
while (*gp)
|
|
|
|
gp = &(*gp)->next;
|
|
|
|
*gp = g;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-01-10 16:31:05 +01:00
|
|
|
maxgid = MAX(gid, maxgid);
|
2017-12-28 18:29:40 +01:00
|
|
|
g->name = str_strdup(name);
|
2014-10-31 15:13:05 +01:00
|
|
|
g->gid = gid;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2014-10-31 15:13:05 +01:00
|
|
|
g->nexthash = ghash[index];
|
|
|
|
return ghash[index] = g;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static void init_group(faction * f, group * g)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-10-31 15:13:05 +01:00
|
|
|
ally *a, **an;
|
|
|
|
|
|
|
|
an = &g->allies;
|
|
|
|
for (a = f->allies; a; a = a->next)
|
|
|
|
if (a->faction) {
|
|
|
|
ally *ga = ally_add(an, a->faction);
|
|
|
|
ga->status = a->status;
|
|
|
|
an = &ga->next;
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static group *find_groupbyname(group * g, const char *name)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-10-31 15:13:05 +01:00
|
|
|
while (g && unicode_utf8_strcasecmp(name, g->name) != 0)
|
|
|
|
g = g->next;
|
|
|
|
return g;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static group *find_group(int gid)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-10-31 15:13:05 +01:00
|
|
|
int index = gid % GMAXHASH;
|
|
|
|
group *g = ghash[index];
|
|
|
|
while (g && g->gid != gid)
|
|
|
|
g = g->nexthash;
|
|
|
|
return g;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2018-02-09 21:20:43 +01:00
|
|
|
static int read_group(variant *var, void *owner, gamedata *data)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2016-02-13 13:42:02 +01:00
|
|
|
struct storage *store = data->store;
|
2014-10-31 15:13:05 +01:00
|
|
|
group *g;
|
|
|
|
int gid;
|
|
|
|
|
|
|
|
READ_INT(store, &gid);
|
2018-02-09 21:20:43 +01:00
|
|
|
var->v = g = find_group(gid);
|
2014-10-31 15:13:05 +01:00
|
|
|
if (g != 0) {
|
|
|
|
g->members++;
|
|
|
|
return AT_READ_OK;
|
|
|
|
}
|
|
|
|
return AT_READ_FAIL;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-02-09 21:20:43 +01:00
|
|
|
write_group(const variant *var, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-02-09 21:20:43 +01:00
|
|
|
group *g = (group *)var->v;
|
2014-10-31 15:13:05 +01:00
|
|
|
WRITE_INT(store, g->gid);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
attrib_type at_group = { /* attribute for units assigned to a group */
|
2014-10-31 15:13:05 +01:00
|
|
|
"grp",
|
2011-03-07 08:02:35 +01:00
|
|
|
DEFAULT_INIT,
|
2016-02-09 06:43:19 +01:00
|
|
|
DEFAULT_FINALIZE, DEFAULT_AGE, write_group, read_group, NULL, ATF_UNIQUE };
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
void free_group(group * g)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2016-09-08 17:40:17 +02:00
|
|
|
int index;
|
|
|
|
group **g_ptr;
|
|
|
|
assert(g);
|
|
|
|
index = g->gid % GMAXHASH;
|
|
|
|
g_ptr = ghash + index;
|
|
|
|
while (*g_ptr && (*g_ptr)->gid != g->gid) {
|
2014-10-31 15:13:05 +01:00
|
|
|
g_ptr = &(*g_ptr)->nexthash;
|
2016-09-08 17:40:17 +02:00
|
|
|
}
|
2014-10-31 15:13:05 +01:00
|
|
|
assert(*g_ptr == g);
|
|
|
|
*g_ptr = g->nexthash;
|
|
|
|
|
2016-03-10 22:30:19 +01:00
|
|
|
if (g->attribs) {
|
|
|
|
a_removeall(&g->attribs, NULL);
|
|
|
|
}
|
2014-10-31 15:13:05 +01:00
|
|
|
while (g->allies) {
|
|
|
|
ally *a = g->allies;
|
|
|
|
g->allies = a->next;
|
|
|
|
free(a);
|
|
|
|
}
|
|
|
|
free(g->name);
|
|
|
|
free(g);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2012-06-07 18:47:02 +02:00
|
|
|
group * get_group(const struct unit *u)
|
|
|
|
{
|
2014-10-31 15:13:05 +01:00
|
|
|
if (fval(u, UFL_GROUP)) {
|
|
|
|
attrib * a = a_find(u->attribs, &at_group);
|
|
|
|
if (a) {
|
|
|
|
return (group *)a->data.v;
|
|
|
|
}
|
2012-06-07 18:47:02 +02:00
|
|
|
}
|
2014-10-31 15:13:05 +01:00
|
|
|
return 0;
|
2012-06-07 18:47:02 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void set_group(struct unit *u, struct group *g)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-10-31 15:13:05 +01:00
|
|
|
attrib *a = NULL;
|
|
|
|
|
|
|
|
if (fval(u, UFL_GROUP)) {
|
|
|
|
a = a_find(u->attribs, &at_group);
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
a->data.v = g;
|
|
|
|
g->members++;
|
|
|
|
}
|
|
|
|
else if (a) {
|
|
|
|
a_remove(&u->attribs, a);
|
|
|
|
freset(u, UFL_GROUP);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-16 07:30:26 +01:00
|
|
|
group *join_group(unit * u, const char *name)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-10-31 15:13:05 +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);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-10-31 15:13:05 +01:00
|
|
|
set_group(u, g);
|
2016-02-16 07:30:26 +01:00
|
|
|
return g;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-11-03 18:10:05 +01:00
|
|
|
void write_groups(struct storage *store, const faction * f)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-11-03 18:10:05 +01:00
|
|
|
group *g;
|
2015-01-30 20:37:14 +01:00
|
|
|
for (g = f->groups; g; g = g->next) {
|
2014-10-31 15:13:05 +01:00
|
|
|
ally *a;
|
|
|
|
WRITE_INT(store, g->gid);
|
|
|
|
WRITE_STR(store, g->name);
|
|
|
|
for (a = g->allies; a; a = a->next) {
|
2016-02-16 07:30:26 +01:00
|
|
|
if (a->faction && a->faction->_alive) {
|
2014-10-31 15:13:05 +01:00
|
|
|
write_faction_reference(a->faction, store);
|
|
|
|
WRITE_INT(store, a->status);
|
|
|
|
}
|
|
|
|
}
|
2016-02-16 07:30:26 +01:00
|
|
|
write_faction_reference(NULL, store);
|
2014-10-31 15:13:05 +01:00
|
|
|
a_write(store, g->attribs, g);
|
|
|
|
WRITE_SECTION(store);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2013-12-31 10:06:28 +01:00
|
|
|
WRITE_INT(store, 0);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2016-02-13 13:42:02 +01:00
|
|
|
void read_groups(gamedata *data, faction * f)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2016-02-13 13:42:02 +01:00
|
|
|
struct storage *store = data->store;
|
2010-08-08 10:06:34 +02:00
|
|
|
for (;;) {
|
2014-10-31 15:13:05 +01:00
|
|
|
ally **pa;
|
|
|
|
group *g;
|
|
|
|
int gid;
|
|
|
|
char buf[1024];
|
|
|
|
|
|
|
|
READ_INT(store, &gid);
|
|
|
|
if (gid == 0)
|
|
|
|
break;
|
|
|
|
READ_STR(store, buf, sizeof(buf));
|
|
|
|
g = new_group(f, buf, gid);
|
|
|
|
pa = &g->allies;
|
|
|
|
for (;;) {
|
2017-09-21 16:26:53 +02:00
|
|
|
ally *al;
|
2017-09-22 17:19:55 +02:00
|
|
|
int id;
|
|
|
|
READ_INT(store, &id);
|
|
|
|
if (id == 0) break;
|
2017-09-21 16:26:53 +02:00
|
|
|
al = ally_add(pa, NULL);
|
2017-09-22 17:19:55 +02:00
|
|
|
al->faction = findfaction(id);
|
|
|
|
if (!al->faction) {
|
|
|
|
ur_add(RESOLVE_FACTION | id, (void **)&al->faction, NULL);
|
2017-09-21 16:26:53 +02:00
|
|
|
}
|
|
|
|
READ_INT(store, &al->status);
|
2014-10-31 15:13:05 +01:00
|
|
|
}
|
2016-02-13 13:42:02 +01:00
|
|
|
read_attribs(data, &g->attribs, g);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|