home *** CD-ROM | disk | FTP | other *** search
- /*Program 8_4 - The Maze Problem
- by Stephen R. Davis, 1987
-
- Notice how small this reiterative solution to the maze problem
- is. The routine evaluate() checks the current location for
- cheese (success) or wall (failure). If the answer is neither,
- it then calls itself in each of the 3 directions open to it
- (we forbid it from looking in the direction from which it came)
- until success or failure is attained. The successful path is
- marked with '.'s.
-
- Create mazes with any ASCII file editor. Rules for mazes are:
- - no loops
- - halls marked with spaces
- - cheese marked with '&' (need not be present)
- - halls must be single space wide
- - must have a single opening at the top; search starts here
- */
-
- #include <stdio.h>
- #include <dos.h>
- /*use 0xb0000000 for monochrome and Hercules screens*/
- #define screenaddr ((unsigned far *)0xb8000000L)
- #define screen(y,x) screenaddr [(y * 80) + x]
-
- #define space ' '
- #define cheese '&'
- #define path '.'
- #define VISIBLE 1
-
- /*define prototypes for routines used*/
-
- int main (int, char **);
- void readmaze (FILE *);
- int evaluate (unsigned, unsigned, unsigned);
- int value (unsigned, unsigned);
- void clrscrn (void);
-
- /*Main - if one argument provided, read it up onto the screen and
- evaluate it for a maze solution*/
- main (argc, argv)
- int argc;
- char *argv[];
- {
- FILE *fp;
- unsigned x;
-
- /*clear the screen - first line of maze must be at top of screen*/
- clrscrn ();
-
- if (argc == 2) {
- if (fp = fopen(argv[1], "r")) {
- readmaze (fp);
-
- /*solve maze by finding a space in the top and
- searching from there*/
- for (x = 0; (char)screen (0, x) != space; x++);
- if (evaluate (0, x, 2))
- printf ("solution!\n");
- else
- printf ("no solution found\n");
- } else
- printf ("File not found\n");
- } else
- printf ("Enter 'Prg8_4 <filename>'\n"
- " where filename contains the maze to be solved\n");
- }
-
- /*Readmaze - read a maze from a file onto the screen. Transfer
- only the character part to the screen.*/
- void readmaze (fptr)
- FILE *fptr;
- {
- char buffer [81];
-
- while (fgets (buffer, 80, fptr))
- printf (buffer);
- }
-
- /*Evaluate - solve the maze*/
-
- int deltax [] = {0, 1, 0, -1}; /*define the directions*/
- int deltay [] = {-1, 0, 1, 0};
- int noallow [] = {2, 3, 0, 1};
-
- int evaluate(yloc, xloc, prevmove)
- unsigned yloc, xloc, prevmove;
- {
- int i, val;
- int value();
-
- if ((val = value(xloc, yloc)) == -1) /*wall*/
- return 0;
- if (val == 1) /*cheese!*/
- return 1;
-
- for (i = 0; i < 4; i++) /*4 possible moves*/
- if (i != noallow[prevmove]) /*don't go backwards*/
- if (evaluate(yloc+deltay[i], xloc+deltax[i], i)) {
- (char)screen (yloc, xloc) = path; /*found it!*/
- return 1;
- }
- return 0; /*nothing down this path*/
- }
-
- /*Value - evaluate the current location*/
- int value (xloc, yloc)
- unsigned xloc, yloc;
- {
- char curr;
- int i;
-
- curr = (char)screen (yloc, xloc);
- #if VISIBLE /*make the search visible*/
- (char)screen (yloc, xloc) = '#';
- for (i = 0; i < 10000; i++) ;
- (char)screen (yloc, xloc) = curr;
- #endif
- switch (curr) {
- case cheese: return 1; /*cheese -> success*/
- case space : return 0; /*space -> keep looking*/
- default : return -1; /*else -> can't go that way*/
- }
- }
-
- /*Clrscrn - clear the screen*/
- struct REGS regs;
- void clrscrn (void)
- {
- regs.h.ah = 6;
- regs.h.al = 0;
- regs.h.bh = 7;
- regs.x.cx = 0;
- regs.x.dx = 0x1950;
- int86 (0x10, ®s, ®s);
-
- regs.h.ah = 2;
- regs.h.bh = 0;
- regs.x.dx = 0;
- int86 (0x10, ®s, ®s);
- }