refactor looping through the hashtable of seen_regions into a map function.

This commit is contained in:
Enno Rehling 2015-09-09 09:38:57 +02:00
parent 3eabf082ac
commit 42e51a0ec4
3 changed files with 56 additions and 17 deletions

View file

@ -96,30 +96,45 @@ seen_region *find_seen(struct seen_region *seehash[], const region * r)
return NULL; return NULL;
} }
void get_seen_interval(struct seen_region *seen[], struct region **firstp, struct region **lastp) void seenhash_map(struct seen_region *seen[], void(*cb)(seen_region *, void *), void *cbdata) {
{
/* this is required to find the neighbour regions of the ones we are in,
* which may well be outside of [firstregion, lastregion) */
int i; int i;
region *first, *last;
assert(seen && firstp && lastp);
first = *firstp;
last = *lastp;
for (i = 0; i != MAXSEEHASH; ++i) { for (i = 0; i != MAXSEEHASH; ++i) {
seen_region *sr = seen[i]; seen_region *sr = seen[i];
while (sr != NULL) { while (sr != NULL) {
if (first == NULL || sr->r->index < first->index) { cb(sr, cbdata);
first = sr->r;
}
if (last != NULL && sr->r->index >= last->index) {
last = sr->r->next;
}
sr = sr->nextHash; sr = sr->nextHash;
} }
} }
*firstp = first; }
*lastp = last;
typedef struct cb_interval {
region *first;
region *last;
} cb_interval;
static void cb_get_interval(seen_region *sr, void *cbdata) {
cb_interval *iv = (cb_interval *)cbdata;
region *r = sr->r;
if (iv->first == NULL || r->index < iv->first->index) {
iv->first = r;
}
if (iv->last != NULL && r->index >= iv->last->index) {
iv->last = r->next;
}
}
/* this function adds the neighbour regions of the ones we have seen
* to the interval, which may be outside of [faction.first, faction.last)
*/
void get_seen_interval(struct seen_region *seen[], struct region **firstp, struct region **lastp)
{
cb_interval interval;
interval.first = *firstp;
interval.last = *lastp;
seenhash_map(seen, cb_get_interval, &interval);
*firstp = interval.first;
*lastp = interval.last;
} }
bool add_seen(struct seen_region *seehash[], struct region *r, unsigned char mode, bool dis) bool add_seen(struct seen_region *seehash[], struct region *r, unsigned char mode, bool dis)

View file

@ -53,6 +53,7 @@ struct seen_region *find_seen(struct seen_region *seehash[], const struct region
void get_seen_interval(struct seen_region *seen[], struct region **firstp, struct region **lastp); void get_seen_interval(struct seen_region *seen[], struct region **firstp, struct region **lastp);
bool add_seen(struct seen_region *seehash[], struct region *r, unsigned char mode, bool dis); bool add_seen(struct seen_region *seehash[], struct region *r, unsigned char mode, bool dis);
void link_seen(struct seen_region *seehash[], const struct region *first, const struct region *last); void link_seen(struct seen_region *seehash[], const struct region *first, const struct region *last);
void seenhash_map(struct seen_region *seen[], void(*cb)(struct seen_region *, void *), void *cbdata);
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif #endif

View file

@ -121,6 +121,28 @@ static void test_seen_interval_forward(CuTest *tc) {
test_cleanup(); test_cleanup();
} }
static void cb_testmap(seen_region *sr, void *cbdata) {
int *ip = (int *)cbdata;
*ip += sr->r->y;
}
static void test_seenhash_map(CuTest *tc) {
region *r;
seen_region **seen;
int i = 0;
test_cleanup();
seen = seen_init();
r = test_create_region(1, 1, 0);
add_seen(seen, r, see_unit, false);
r = test_create_region(2, 2, 0);
add_seen(seen, r, see_unit, false);
seenhash_map(seen, cb_testmap, &i);
CuAssertIntEquals(tc, 3, i);
seen_done(seen);
test_cleanup();
}
CuSuite *get_seen_suite(void) CuSuite *get_seen_suite(void)
{ {
CuSuite *suite = CuSuiteNew(); CuSuite *suite = CuSuiteNew();
@ -129,5 +151,6 @@ CuSuite *get_seen_suite(void)
SUITE_ADD_TEST(suite, test_seen_region); SUITE_ADD_TEST(suite, test_seen_region);
SUITE_ADD_TEST(suite, test_seen_interval_backward); SUITE_ADD_TEST(suite, test_seen_interval_backward);
SUITE_ADD_TEST(suite, test_seen_interval_forward); SUITE_ADD_TEST(suite, test_seen_interval_forward);
SUITE_ADD_TEST(suite, test_seenhash_map);
return suite; return suite;
} }