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 "faction.h"
|
|
|
|
|
|
|
|
|
|
#include "alliance.h"
|
|
|
|
|
#include "equipment.h"
|
|
|
|
|
#include "group.h"
|
|
|
|
|
#include "item.h"
|
|
|
|
|
#include "message.h"
|
|
|
|
|
#include "plane.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "terrain.h"
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
#include "version.h"
|
|
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
#include <util/base36.h>
|
|
|
|
|
#include <util/event.h>
|
|
|
|
|
#include <util/goodies.h>
|
|
|
|
|
#include <util/lists.h>
|
|
|
|
|
#include <util/language.h>
|
|
|
|
|
#include <util/log.h>
|
|
|
|
|
#include <util/resolve.h>
|
|
|
|
|
#include <util/rng.h>
|
|
|
|
|
#include <util/storage.h>
|
|
|
|
|
#include <util/sql.h>
|
|
|
|
|
#include <util/variant.h>
|
|
|
|
|
#include <util/unicode.h>
|
|
|
|
|
#include <attributes/otherfaction.h>
|
|
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
faction *factions;
|
|
|
|
|
|
|
|
|
|
/** remove the faction from memory.
|
|
|
|
|
* this frees all memory that's only accessible through the faction,
|
|
|
|
|
* but you should still call funhash and remove the faction from the
|
|
|
|
|
* global list.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
free_faction (faction * f)
|
|
|
|
|
{
|
|
|
|
|
if (f->msgs) free_messagelist(f->msgs);
|
|
|
|
|
while (f->battles) {
|
|
|
|
|
struct bmsg * bm = f->battles;
|
|
|
|
|
f->battles = bm->next;
|
|
|
|
|
if (bm->msgs) free_messagelist(bm->msgs);
|
|
|
|
|
free(bm);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while (f->groups) {
|
|
|
|
|
group * g = f->groups;
|
|
|
|
|
f->groups = g->next;
|
|
|
|
|
free_group(g);
|
|
|
|
|
}
|
|
|
|
|
freelist(f->allies);
|
|
|
|
|
|
|
|
|
|
free(f->email);
|
|
|
|
|
free(f->banner);
|
|
|
|
|
free(f->passw);
|
|
|
|
|
free(f->override);
|
|
|
|
|
free(f->name);
|
|
|
|
|
|
|
|
|
|
while (f->attribs) {
|
|
|
|
|
a_remove (&f->attribs, f->attribs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
i_freeall(&f->items);
|
|
|
|
|
|
|
|
|
|
freelist(f->ursprung);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
faction *
|
|
|
|
|
get_monsters(void)
|
|
|
|
|
{
|
|
|
|
|
static faction * monsters;
|
|
|
|
|
static int gamecookie = -1;
|
|
|
|
|
|
|
|
|
|
if (gamecookie!=global.cookie) {
|
|
|
|
|
monsters = NULL;
|
|
|
|
|
gamecookie = global.cookie;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!monsters) {
|
|
|
|
|
faction * f;
|
|
|
|
|
for (f=factions;f;f=f->next) {
|
|
|
|
|
if (f->flags&FFL_NPC) {
|
|
|
|
|
return monsters=f;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!monsters) {
|
|
|
|
|
/* shit! */
|
|
|
|
|
monsters = findfaction(666);
|
|
|
|
|
}
|
|
|
|
|
if (monsters) {
|
|
|
|
|
fset(monsters, FFL_NPC|FFL_NOIDLEOUT);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return monsters;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const unit *
|
|
|
|
|
random_unit_in_faction(const faction *f)
|
|
|
|
|
{
|
|
|
|
|
unit *u;
|
|
|
|
|
int c = 0, u_nr;
|
|
|
|
|
|
|
|
|
|
for(u = f->units; u; u=u->next) c++;
|
|
|
|
|
|
|
|
|
|
u_nr = rng_int()%c;
|
|
|
|
|
c = 0;
|
|
|
|
|
|
|
|
|
|
for(u = f->units; u; u=u->next)
|
|
|
|
|
if(u_nr == c) return u;
|
|
|
|
|
|
|
|
|
|
/* Hier sollte er nie ankommen */
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
|
factionname(const faction * f)
|
|
|
|
|
{
|
|
|
|
|
typedef char name[OBJECTIDSIZE+1];
|
|
|
|
|
static name idbuf[8];
|
|
|
|
|
static int nextbuf = 0;
|
|
|
|
|
|
|
|
|
|
char *ibuf = idbuf[(++nextbuf) % 8];
|
|
|
|
|
|
|
|
|
|
if (f && f->name) {
|
|
|
|
|
snprintf(ibuf, sizeof(name), "%s (%s)", f->name, itoa36(f->no));
|
|
|
|
|
ibuf[sizeof(name)-1] = 0;
|
|
|
|
|
} else {
|
|
|
|
|
strcpy(ibuf, "Unbekannte Partei (?)");
|
|
|
|
|
}
|
|
|
|
|
return ibuf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
resolve_faction(variant id, void * address) {
|
|
|
|
|
int result = 0;
|
|
|
|
|
faction * f = NULL;
|
|
|
|
|
if (id.i!=0) {
|
|
|
|
|
f = findfaction(id.i);
|
|
|
|
|
if (f==NULL) {
|
|
|
|
|
result = -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*(faction**)address = f;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define MAX_FACTION_ID (36*36*36*36)
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
unused_faction_id(void)
|
|
|
|
|
{
|
|
|
|
|
int id = rng_int()%MAX_FACTION_ID;
|
|
|
|
|
|
|
|
|
|
while (!faction_id_is_unused(id)) {
|
|
|
|
|
id++; if(id == MAX_FACTION_ID) id = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
faction *
|
|
|
|
|
addfaction(const char *email, const char * password,
|
|
|
|
|
const struct race * frace, const struct locale *loc,
|
|
|
|
|
int subscription)
|
|
|
|
|
{
|
|
|
|
|
faction * f = calloc(sizeof(faction), 1);
|
|
|
|
|
char buf[128];
|
|
|
|
|
|
|
|
|
|
assert(frace);
|
|
|
|
|
|
|
|
|
|
if (set_email(&f->email, email)!=0) {
|
|
|
|
|
log_error(("Invalid email address for faction %s: %s\n", itoa36(f->no), email));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f->override = strdup(itoa36(rng_int()));
|
|
|
|
|
faction_setpassword(f, password);
|
|
|
|
|
|
|
|
|
|
f->alliance_joindate = turn;
|
|
|
|
|
f->lastorders = turn;
|
|
|
|
|
f->alive = 1;
|
|
|
|
|
f->age = 0;
|
|
|
|
|
f->race = frace;
|
|
|
|
|
f->magiegebiet = 0;
|
|
|
|
|
f->locale = loc;
|
|
|
|
|
f->subscription = subscription;
|
|
|
|
|
|
|
|
|
|
f->options = want(O_REPORT) | want(O_ZUGVORLAGE) | want(O_COMPUTER) | want(O_COMPRESS) | want(O_ADRESSEN) | want(O_STATISTICS);
|
|
|
|
|
|
|
|
|
|
f->no = unused_faction_id();
|
|
|
|
|
if (rule_region_owners()) {
|
|
|
|
|
alliance * al = makealliance(f->no, NULL);
|
|
|
|
|
setalliance(f, al);
|
|
|
|
|
}
|
|
|
|
|
addlist(&factions, f);
|
|
|
|
|
fhash(f);
|
|
|
|
|
|
|
|
|
|
snprintf(buf, sizeof(buf), "%s %s", LOC(loc, "factiondefault"), factionid(f));
|
|
|
|
|
f->name = strdup(buf);
|
|
|
|
|
|
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unit *
|
|
|
|
|
addplayer(region *r, faction * f)
|
|
|
|
|
{
|
|
|
|
|
unit *u;
|
|
|
|
|
char buffer[32];
|
|
|
|
|
|
|
|
|
|
assert(f->units==NULL);
|
|
|
|
|
set_ursprung(f, 0, r->x, r->y);
|
|
|
|
|
u = createunit(r, f, 1, f->race);
|
|
|
|
|
equip_items(&u->faction->items, get_equipment("new_faction"));
|
|
|
|
|
equip_unit(u, get_equipment("first_unit"));
|
|
|
|
|
sprintf(buffer, "first_%s", u->race->_name[0]);
|
|
|
|
|
equip_unit(u, get_equipment(buffer));
|
|
|
|
|
u->hp = unit_max_hp(u) * u->number;
|
|
|
|
|
fset(u, UFL_ISNEW);
|
|
|
|
|
if (f->race == new_race[RC_DAEMON]) {
|
|
|
|
|
race_t urc;
|
|
|
|
|
do {
|
|
|
|
|
urc = (race_t)(rng_int() % MAXRACES);
|
|
|
|
|
} while (new_race[urc]==NULL || urc == RC_DAEMON || !playerrace(new_race[urc]));
|
|
|
|
|
u->irace = new_race[urc];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return u;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
checkpasswd(const faction * f, const char * passwd, boolean shortp)
|
|
|
|
|
{
|
|
|
|
|
if (unicode_utf8_strcasecmp(f->passw, passwd)==0) return true;
|
|
|
|
|
if (unicode_utf8_strcasecmp(f->override, passwd)==0) return true;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
variant
|
|
|
|
|
read_faction_reference(struct storage * store)
|
|
|
|
|
{
|
|
|
|
|
variant id;
|
|
|
|
|
id.i = store->r_id(store);
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
write_faction_reference(const faction * f, struct storage * store)
|
|
|
|
|
{
|
|
|
|
|
store->w_id(store, f?f->no:0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
destroyfaction(faction * f)
|
|
|
|
|
{
|
|
|
|
|
unit *u = f->units;
|
|
|
|
|
faction *ff;
|
|
|
|
|
|
|
|
|
|
if (!f->alive) return;
|
|
|
|
|
fset(f, FFL_QUIT);
|
|
|
|
|
|
|
|
|
|
freelist(f->spellbook);
|
|
|
|
|
f->spellbook = NULL;
|
|
|
|
|
|
|
|
|
|
while (f->battles) {
|
|
|
|
|
struct bmsg * bm = f->battles;
|
|
|
|
|
f->battles = bm->next;
|
|
|
|
|
if (bm->msgs) free_messagelist(bm->msgs);
|
|
|
|
|
free(bm);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while (u) {
|
|
|
|
|
/* give away your stuff, make zombies if you cannot (quest items) */
|
|
|
|
|
int result = gift_items(u, GIFT_FRIENDS|GIFT_PEASANTS);
|
|
|
|
|
if (result!=0) {
|
|
|
|
|
unit * zombie = u;
|
|
|
|
|
u = u->nextF;
|
|
|
|
|
make_zombie(zombie);
|
|
|
|
|
} else {
|
|
|
|
|
region * r = u->region;
|
|
|
|
|
|
|
|
|
|
if (!fval(r->terrain, SEA_REGION) && !!playerrace(u->race)) {
|
|
|
|
|
const race * rc = u->race;
|
|
|
|
|
int m = rmoney(r);
|
|
|
|
|
|
|
|
|
|
if ((rc->ec_flags & ECF_REC_ETHEREAL)==0) {
|
|
|
|
|
int p = rpeasants(u->region);
|
|
|
|
|
int h = rhorses(u->region);
|
|
|
|
|
item * itm;
|
|
|
|
|
|
|
|
|
|
/* Personen gehen nur an die Bauern, wenn sie auch von dort
|
|
|
|
|
* stammen */
|
|
|
|
|
if (rc->ec_flags & ECF_REC_HORSES) { /* Zentauren an die Pferde */
|
|
|
|
|
h += u->number;
|
|
|
|
|
} else { /* Orks z<>hlen nur zur H<>lfte */
|
|
|
|
|
p += (int)(u->number * rc->recruit_multi);
|
|
|
|
|
}
|
|
|
|
|
for (itm=u->items;itm;itm=itm->next) {
|
|
|
|
|
if (itm->type->flags&ITF_ANIMAL) {
|
|
|
|
|
h += itm->number;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
rsetpeasants(r, p);
|
|
|
|
|
rsethorses(r, h);
|
|
|
|
|
}
|
|
|
|
|
m += get_money(u);
|
|
|
|
|
rsetmoney(r, m);
|
|
|
|
|
}
|
|
|
|
|
set_number(u, 0);
|
|
|
|
|
u=u->nextF;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
f->alive = 0;
|
|
|
|
|
/* no way! f->units = NULL; */
|
|
|
|
|
handle_event(f->attribs, "destroy", f);
|
|
|
|
|
for (ff = factions; ff; ff = ff->next) {
|
|
|
|
|
group *g;
|
|
|
|
|
ally *sf, *sfn;
|
|
|
|
|
|
|
|
|
|
/* Alle HELFE f<>r die Partei l<>schen */
|
|
|
|
|
for (sf = ff->allies; sf; sf = sf->next) {
|
|
|
|
|
if (sf->faction == f) {
|
|
|
|
|
removelist(&ff->allies, sf);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for(g=ff->groups; g; g=g->next) {
|
|
|
|
|
for (sf = g->allies; sf;) {
|
|
|
|
|
sfn = sf->next;
|
|
|
|
|
if (sf->faction == f) {
|
|
|
|
|
removelist(&g->allies, sf);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
sf = sfn;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* units of other factions that were disguised as this faction
|
|
|
|
|
* have their disguise replaced by ordinary faction hiding. */
|
|
|
|
|
if (rule_stealth_faction()) {
|
|
|
|
|
region * rc;
|
|
|
|
|
for (rc=regions; rc; rc=rc->next) {
|
|
|
|
|
for(u=rc->units; u; u=u->next) {
|
|
|
|
|
attrib *a = a_find(u->attribs, &at_otherfaction);
|
|
|
|
|
if(!a) continue;
|
|
|
|
|
if (get_otherfaction(a) == f) {
|
|
|
|
|
a_removeall(&u->attribs, &at_otherfaction);
|
|
|
|
|
fset(u, UFL_ANON_FACTION);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
get_alliance(const faction * a, const faction * b)
|
|
|
|
|
{
|
|
|
|
|
const ally * sf = a->allies;
|
|
|
|
|
for (;sf!=NULL;sf=sf->next) {
|
|
|
|
|
if (sf->faction==b) {
|
|
|
|
|
return sf->status;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
set_alliance(faction * a, faction * b, int status)
|
|
|
|
|
{
|
|
|
|
|
ally ** sfp;
|
|
|
|
|
sfp = &a->allies;
|
|
|
|
|
while (*sfp) {
|
|
|
|
|
ally * sf = *sfp;
|
|
|
|
|
if (sf->faction==b) break;
|
|
|
|
|
sfp = &sf->next;
|
|
|
|
|
}
|
|
|
|
|
if (*sfp==NULL) {
|
|
|
|
|
ally * sf = *sfp = malloc(sizeof(ally));
|
|
|
|
|
sf->next = NULL;
|
|
|
|
|
sf->status = status;
|
|
|
|
|
sf->faction = b;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
(*sfp)->status |= status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
renumber_faction(faction * f, int no)
|
|
|
|
|
{
|
|
|
|
|
if (f->subscription) {
|
|
|
|
|
sql_print(("UPDATE subscriptions set faction='%s' where id=%u;\n",
|
|
|
|
|
itoa36(no), f->subscription));
|
|
|
|
|
}
|
|
|
|
|
funhash(f);
|
|
|
|
|
f->no = no;
|
|
|
|
|
fhash(f);
|
|
|
|
|
fset(f, FFL_NEWID);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef SMART_INTERVALS
|
|
|
|
|
void
|
|
|
|
|
update_interval(struct faction * f, struct region * r)
|
|
|
|
|
{
|
|
|
|
|
if (r==NULL || f==NULL) return;
|
|
|
|
|
if (f->first==NULL || f->first->index>r->index) {
|
|
|
|
|
f->first = r;
|
|
|
|
|
}
|
|
|
|
|
if (f->last==NULL || f->last->index<=r->index) {
|
|
|
|
|
f->last = r;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
const char * faction_getname(const faction * self)
|
|
|
|
|
{
|
|
|
|
|
return self->name?self->name:"";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void faction_setname(faction * self, const char * name)
|
|
|
|
|
{
|
|
|
|
|
free(self->name);
|
|
|
|
|
if (name) self->name = strdup(name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char * faction_getemail(const faction * self)
|
|
|
|
|
{
|
|
|
|
|
return self->email?self->email:"";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void faction_setemail(faction * self, const char * email)
|
|
|
|
|
{
|
|
|
|
|
free(self->email);
|
|
|
|
|
if (email) self->email = strdup(email);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char * faction_getbanner(const faction * self)
|
|
|
|
|
{
|
|
|
|
|
return self->banner?self->banner:"";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void faction_setbanner(faction * self, const char * banner)
|
|
|
|
|
{
|
|
|
|
|
free(self->banner);
|
|
|
|
|
if (banner) self->banner = strdup(banner);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
faction_setpassword(faction * f, const char * passw)
|
|
|
|
|
{
|
|
|
|
|
free(f->passw);
|
|
|
|
|
if (passw) f->passw = strdup(passw);
|
|
|
|
|
else f->passw = strdup(itoa36(rng_int()));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean valid_race(const struct faction * f, const struct race * rc)
|
|
|
|
|
{
|
|
|
|
|
if (f->race==rc) return true;
|
|
|
|
|
else {
|
|
|
|
|
const char * str = get_param(f->race->parameters, "other_race");
|
|
|
|
|
if (str) return (boolean)(rc_find(str)==rc);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
|
faction_getpassword(const faction * f)
|
|
|
|
|
{
|
|
|
|
|
return f->passw;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct alliance *
|
|
|
|
|
f_get_alliance(const struct faction * f)
|
|
|
|
|
{
|
|
|
|
|
if (f->alliance && !(f->alliance->flags&ALF_NON_ALLIED)) {
|
|
|
|
|
return f->alliance;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2010-10-13 08:17:35 +02:00
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
flist_add(faction_list ** flist, faction * sf)
|
|
|
|
|
{
|
|
|
|
|
faction_list ** fnew = flist;
|
|
|
|
|
while (*fnew && (*fnew)->data->no < sf->no) {
|
|
|
|
|
fnew =&(*fnew)->next;
|
|
|
|
|
}
|
|
|
|
|
if ((*fnew==NULL) || (*fnew)->data!=sf) {
|
|
|
|
|
faction_list * finsert = malloc(sizeof(faction_list));
|
|
|
|
|
finsert->next = *fnew;
|
|
|
|
|
*fnew = finsert;
|
|
|
|
|
finsert->data = sf;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|