X-Git-Url: https://git.martlubbers.net/?a=blobdiff_plain;f=modelchecker%2Fsokoban.c;h=fd268c03fc5549503d2436d2cbb042b37010010e;hb=b3851097adbafa74d01e05b25ba550a814a5767d;hp=d5e967fe6d5c2ef81e111c9c087303adfb9f4f94;hpb=476b70d5c7cb978fec97c99fe462402eb38405de;p=mc1516pa.git diff --git a/modelchecker/sokoban.c b/modelchecker/sokoban.c index d5e967f..fd268c0 100644 --- a/modelchecker/sokoban.c +++ b/modelchecker/sokoban.c @@ -1,6 +1,5 @@ #include #include -#include "uthash.h" #include "sokoban.h" sokoban_screen *add_coord(int x, int y, sokoban_tile tile, sokoban_screen *screen) @@ -50,7 +49,24 @@ void sokoban_print(sokoban_screen *screen) } } -void sokoban_clear(sokoban_screen *screen) +sokoban_screen *sokoban_shrink(int x, int y, sokoban_screen *screen, sokoban_screen *newscreen) +{ + sokoban_screen *c, *nc = NULL; + c = get_coord(x, y, screen); + if (c) { + nc = get_coord(c->coord.x, c->coord.y, newscreen); + if (c->tile != WALL && nc == NULL) { + newscreen = add_coord(x, y, c->tile, newscreen); + newscreen = sokoban_shrink(x-1, y, screen, newscreen); + newscreen = sokoban_shrink(x+1, y, screen, newscreen); + newscreen = sokoban_shrink(x, y-1, screen, newscreen); + newscreen = sokoban_shrink(x, y+1, screen, newscreen); + } + } + return newscreen; +} + +void sokoban_free(sokoban_screen *screen) { sokoban_screen *r, *tmp = NULL; HASH_ITER(hh, screen, r, tmp) { @@ -59,11 +75,16 @@ void sokoban_clear(sokoban_screen *screen) } } -sokoban_screen *parse_screen(FILE *stream) +sokoban_screen *parse_screen(FILE *stream, bool safe) { - int buffer, x, y; + int buffer, x, y, agent_x, agent_y, boxes, targets, agents; x = 0; y = 0; + agent_x = 0; + agent_y = 0; + boxes = 0; + targets = 0; + agents = 0; sokoban_screen *screen = NULL; while((buffer = fgetc(stream)) != EOF){ if (buffer == '\n'){ @@ -73,19 +94,52 @@ sokoban_screen *parse_screen(FILE *stream) else { switch(buffer) { case ' ': screen = add_coord(x, y, FREE, screen); break; - case '@': screen = add_coord(x, y, AGENT, screen); break; - case '.': screen = add_coord(x, y, TARGET, screen); break; + case '@': + screen = add_coord(x, y, AGENT, screen); + agent_x = x; + agent_y = y; + agents++; + break; + case '.': + screen = add_coord(x, y, TARGET, screen); + targets++; + break; case '#': screen = add_coord(x, y, WALL, screen); break; - case '$': screen = add_coord(x, y, BOX, screen); break; - case '*': screen = add_coord(x, y, TARGBOX, screen); break; - case '+': screen = add_coord(x, y, TARGAGENT, screen); break; + case '$': + screen = add_coord(x, y, BOX, screen); + boxes++; + break; + case '*': + screen = add_coord(x, y, TARGBOX, screen); + boxes++; + targets++; + break; + case '+': + screen = add_coord(x, y, TARGAGENT, screen); + agent_x = x; + agent_y = y; + agents++; + targets++; + break; default: return NULL; } x++; } } - sokoban_screen *test = NULL; - test = get_coord(0,0,screen); - if(test->tile == TARGAGENT) printf("Yuppi\n"); - return screen; + if(safe == true && boxes != targets){ + fprintf(stderr, + "Invalid screen. Boxes: %d, Targets: %d\n", boxes, targets); + exit(1); + } + if(safe == true && agents != 1){ + fprintf(stderr, + "Invalid screen. There has to be exactly one agent. Found: %d\n", + agents); + exit(1); + } + + sokoban_screen *newscreen = NULL; + newscreen = sokoban_shrink(agent_x, agent_y, screen, newscreen); + sokoban_free(screen); + return newscreen; }