server/src/common/kernel/pathfinder.c

203 lines
4.6 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
*
* 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)
* Enno Rehling (enno@eressea.de)
2001-01-25 10:37:55 +01:00
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* based on:
*
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
* This program may not be sold or used commercially without prior written
* permission from the authors.
*/
#include <config.h>
#include <kernel/eressea.h>
2001-01-25 10:37:55 +01:00
#include "pathfinder.h"
#include "region.h"
#include "terrain.h"
2001-01-25 10:37:55 +01:00
#include <limits.h>
#include <stdlib.h>
boolean
allowed_swim(const region * src, const region * r)
2001-01-25 10:37:55 +01:00
{
if (fval(r->terrain, SWIM_INTO)) return true;
2001-01-25 10:37:55 +01:00
return false;
}
boolean
allowed_walk(const region * src, const region * r)
2001-01-25 10:37:55 +01:00
{
if (fval(r->terrain, WALK_INTO)) return true;
2001-01-25 10:37:55 +01:00
return false;
}
boolean
allowed_fly(const region * src, const region * r)
2001-01-25 10:37:55 +01:00
{
if (fval(r->terrain, FLY_INTO)) return true;
2001-01-25 10:37:55 +01:00
return false;
}
typedef struct node {
struct node * next;
region * r;
struct node * prev;
int distance;
} node;
static node * node_garbage;
2005-06-11 10:09:55 +02:00
void
pathfinder_cleanup(void)
{
while (node_garbage) {
node * n = node_garbage;
node_garbage = n->next;
free(n);
}
}
2001-01-25 10:37:55 +01:00
static node *
new_node(region * r, int distance, node * prev)
{
node * n;
if (node_garbage!=NULL) {
n = node_garbage;
node_garbage = n->next;
}
else n = malloc(sizeof(node));
n->next = NULL;
n->prev = prev;
n->r= r;
n->distance= distance;
return n;
}
static node *
free_node(node * n)
{
node * s = n->next;
n->next = node_garbage;
node_garbage = n;
return s;
}
static void
free_nodes(node * root)
{
while (root!=NULL) {
region * r = root->r;
2007-06-20 02:34:02 +02:00
freset(r, RF_MARK);
root = free_node(root);
}
}
struct region_list *
regions_in_range(struct region * start, int maxdist, boolean (*allowed)(const struct region*, const struct region*))
{
region_list * rlist = NULL;
node * root = new_node(start, 0, NULL);
node ** end = &root->next;
node * n = root;
while (n!=NULL) {
region * r = n->r;
int depth = n->distance+1;
direction_t d;
if (n->distance >= maxdist) break;
for (d=0;d!=MAXDIRECTIONS; ++d) {
region * rn = rconnect(r, d);
if (rn==NULL) continue;
2007-06-20 02:34:02 +02:00
if (fval(rn, RF_MARK)) continue; /* already been there */
if (allowed && !allowed(r, rn)) continue; /* can't go there */
/* add the region to the list of available ones. */
add_regionlist(&rlist, rn);
/* make sure we don't go here again, and put the region into the set for
further BFS'ing */
2007-06-20 02:34:02 +02:00
fset(rn, RF_MARK);
*end = new_node(rn, depth, n);
end = &(*end)->next;
}
n = n->next;
}
free_nodes(root);
return rlist;
}
2001-01-25 10:37:55 +01:00
static region **
internal_path_find(region *start, const region *target, int maxlen, boolean (*allowed)(const region*, const region*))
{
static region * path[MAXDEPTH+2]; /* STATIC_RETURN: used for return, not across calls */
2001-01-25 10:37:55 +01:00
direction_t d;
node * root = new_node(start, 0, NULL);
node ** end = &root->next;
node * n = root;
boolean found = false;
assert(maxlen<=MAXDEPTH);
2007-06-20 02:34:02 +02:00
fset(start, RF_MARK);
2001-01-25 10:37:55 +01:00
while (n!=NULL) {
region * r = n->r;
int depth = n->distance+1;
if (n->distance >= maxlen) break;
for (d=0;d!=MAXDIRECTIONS; ++d) {
region * rn = rconnect(r, d);
if (rn==NULL) continue;
2007-06-20 02:34:02 +02:00
if (fval(rn, RF_MARK)) continue; /* already been there */
2001-01-25 10:37:55 +01:00
if (!allowed(r, rn)) continue; /* can't go there */
if (rn==target) {
int i = depth;
path[i+1] = NULL;
path[i] = rn;
while (n) {
path[--i] = n->r;
n = n->prev;
}
found = true;
break;
} else {
2007-06-20 02:34:02 +02:00
fset(rn, RF_MARK);
2001-01-25 10:37:55 +01:00
*end = new_node(rn, depth, n);
end = &(*end)->next;
}
}
if (found) break;
n = n->next;
}
free_nodes(root);
2001-01-25 10:37:55 +01:00
if (found) return path;
return NULL;
}
boolean
path_exists(region *start, const region *target, int maxlen, boolean (*allowed)(const region*, const region*))
{
2007-06-20 02:34:02 +02:00
assert((!fval(start, RF_MARK) && !fval(target, RF_MARK)) || !"Some Algorithm did not clear its RF_MARKs!");
2001-01-25 10:37:55 +01:00
if (start==target) return true;
if (internal_path_find(start, target, maxlen, allowed)!=NULL) return true;
return false;
}
region **
path_find(region *start, const region *target, int maxlen, boolean (*allowed)(const region*, const region*))
{
2007-06-20 02:34:02 +02:00
assert((!fval(start, RF_MARK) && !fval(target, RF_MARK)) || !"Did you call path_init()?");
2001-01-25 10:37:55 +01:00
return internal_path_find(start, target, maxlen, allowed);
}