/* $OpenBSD: main.c,v 1.30 2015/12/06 12:00:16 tobias Exp $ */ /* * Copyright (c) 1994 * The Regents of the University of California. All rights reserved. * * This code is derived from software contributed to Berkeley by * Ralph Campbell. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include #include #include #include #include #include #include #include "gomoku.h" #define USER 0 /* get input from standard input */ #define PROGRAM 1 /* get input from program */ #define INPUTF 2 /* get input from a file */ extern char *__progname; /* name of program */ int interactive = 1; /* true if interactive */ int debug; /* true if debugging */ int test; /* both moves come from 1: input, 2: computer */ FILE *debugfp; /* file for debug output */ FILE *inputfp; /* file for debug input */ char pdir[4] = "-\\|/"; char fmtbuf[128]; struct spotstr board[BAREA]; /* info for board */ struct combostr frames[FAREA]; /* storage for all frames */ struct combostr *sortframes[2]; /* sorted list of non-empty frames */ u_char overlap[FAREA * FAREA]; /* true if frame [a][b] overlap */ short intersect[FAREA * FAREA]; /* frame [a][b] intersection */ int movelog[BSZ * BSZ]; /* log of all the moves */ int movenum; /* current move number */ char *plyr[2]; /* who's who */ static char you[LOGIN_NAME_MAX]; /* username */ int main(argc, argv) int argc; char **argv; { char buf[128]; char fname[PATH_MAX]; int color = BLACK, curmove = 0, i, ch; int input[2]; static char *fmt[2] = { "%3d %-6s", "%3d %-6s" }; char *tmpname; if (pledge("stdio rpath wpath cpath tty", NULL) == -1) err(1, "pledge"); if ((tmpname = getlogin()) != NULL) strlcpy(you, tmpname, sizeof(you)); else strlcpy(you, "you", sizeof(you)); while ((ch = getopt(argc, argv, "bcdD:hu")) != -1) { switch (ch) { case 'b': /* background */ interactive = 0; break; case 'd': /* debugging */ debug++; break; case 'D': /* log debug output to file */ if ((debugfp = fopen(optarg, "w")) == NULL) err(1, "%s", optarg); break; case 'u': /* testing: user versus user */ test = 1; break; case 'c': /* testing: computer versus computer */ test = 2; break; case 'h': default: fprintf(stderr, "usage: %s [-bcdu] [-D debugfile] [inputfile]\n", __progname); exit(1); } } argc -= optind; argv += optind; if (argc) { if ((inputfp = fopen(*argv, "r")) == NULL) err(1, "%s", *argv); } if (interactive) cursinit(); /* initialize curses */ again: bdinit(board); /* initialize board contents */ if (interactive) { plyr[BLACK] = plyr[WHITE] = "???"; bdisp_init(); /* initialize display of board */ #ifdef DEBUG signal(SIGINT, whatsup); #else signal(SIGINT, quit); #endif if (inputfp == NULL && test == 0) { ask("black or white? "); while (((ch = getchar()) != 'b') && (ch != 'B') && (ch != 'w') && (ch != 'W')) { move(BSZ3, 0); printw("Black moves first. Please enter `black' or `white'\n"); refresh(); } if (ch == 'b' || ch == 'B') color = BLACK; else color = WHITE; move(BSZ3, 0); clrtoeol(); } } else { setvbuf(stdout, NULL, _IONBF, 0); get_line(buf, sizeof(buf)); if (strcmp(buf, "black") == 0) color = BLACK; else if (strcmp(buf, "white") == 0) color = WHITE; else { snprintf(fmtbuf, sizeof fmtbuf, "Huh? Expected `black' or `white', got `%s'\n", buf); panic(fmtbuf); } } if (inputfp) { input[BLACK] = INPUTF; input[WHITE] = INPUTF; } else { switch (test) { case 0: /* user versus program */ input[color] = USER; input[!color] = PROGRAM; break; case 1: /* user versus user */ input[BLACK] = USER; input[WHITE] = USER; break; case 2: /* program versus program */ input[BLACK] = PROGRAM; input[WHITE] = PROGRAM; break; } } if (interactive) { plyr[BLACK] = input[BLACK] == USER ? you : __progname; plyr[WHITE] = input[WHITE] == USER ? you : __progname; bdwho(1); } for (color = BLACK; ; color = !color) { top: switch (input[color]) { case INPUTF: /* input comes from a file */ curmove = readinput(inputfp); if (curmove != ILLEGAL) break; switch (test) { case 0: /* user versus program */ input[color] = USER; input[!color] = PROGRAM; break; case 1: /* user versus user */ input[BLACK] = USER; input[WHITE] = USER; break; case 2: /* program versus program */ input[BLACK] = PROGRAM; input[WHITE] = PROGRAM; break; } plyr[BLACK] = input[BLACK] == USER ? you : __progname; plyr[WHITE] = input[WHITE] == USER ? you : __progname; bdwho(1); goto top; case USER: /* input comes from standard input */ getinput: if (interactive) { ask("Enter move (hjklyubn/S/Q)"); curmove = getcoord(); if (curmove == SAVE) { FILE *fp; ask("save file name? "); (void)get_line(fname, sizeof(fname)); if ((fp = fopen(fname, "w")) == NULL) { logit("cannot create save file"); goto getinput; } for (i = 0; i < movenum - 1; i++) fprintf(fp, "%s\n", stoc(movelog[i])); fclose(fp); goto getinput; } if (curmove != RESIGN && board[curmove].s_occ != EMPTY) { /* logit("Illegal move"); */ beep(); goto getinput; } } else { if (!get_line(buf, sizeof(buf))) { curmove = RESIGN; break; } if (buf[0] == '\0') goto getinput; curmove = ctos(buf); } break; case PROGRAM: /* input comes from the program */ if (interactive) ask("Thinking..."); curmove = pickmove(color); break; } if (interactive) { snprintf(fmtbuf, sizeof fmtbuf, fmt[color], movenum, stoc(curmove)); logit(fmtbuf); } if ((i = makemove(color, curmove)) != MOVEOK) break; if (interactive) bdisp(); } if (interactive) { move(BSZ3, 0); switch (i) { case WIN: if (input[color] == PROGRAM) addstr("Ha ha, I won"); else if (input[0] == USER && input[1] == USER) addstr("Well, you won (and lost)."); else addstr("Rats! You won"); break; case TIE: addstr("Wow! its a tie"); break; case ILLEGAL: addstr("Illegal move"); break; } clrtoeol(); bdisp(); if (i != RESIGN) { replay: ask("replay? "); if (get_line(buf, sizeof(buf)) && (buf[0] == 'y' || buf[0] == 'Y')) goto again; if (strcmp(buf, "save") == 0) { FILE *fp; ask("save file name? "); (void)get_line(buf, sizeof(buf)); if ((fp = fopen(buf, "w")) == NULL) { logit("cannot create save file"); goto replay; } for (i = 0; i < movenum - 1; i++) fprintf(fp, "%s\n", stoc(movelog[i])); fclose(fp); goto replay; } } } quit(0); /* NOTREACHED */ } int readinput(fp) FILE *fp; { char *cp; int c; cp = fmtbuf; while ((c = getc(fp)) != EOF && c != '\n') *cp++ = c; *cp = '\0'; return (ctos(fmtbuf)); } #ifdef DEBUG /* * Handle strange situations. */ void whatsup(signum) int signum; { int i, pnum, n, s1, s2, d1, d2; struct spotstr *sp; FILE *fp; char *str; struct elist *ep; struct combostr *cbp; if (!interactive) quit(0); top: ask("cmd? "); if (!get_line(fmtbuf, sizeof(fmtbuf))) quit(0); switch (*fmtbuf) { case '\0': goto top; case 'q': /* conservative quit */ quit(0); case 'd': /* set debug level */ debug = fmtbuf[1] - '0'; snprintf(fmtbuf, sizeof fmtbuf, "Debug set to %d", debug); dlog(fmtbuf); sleep(1); case 'c': break; case 'b': /* back up a move */ if (movenum > 1) { movenum--; board[movelog[movenum - 1]].s_occ = EMPTY; bdisp(); } goto top; case 's': /* suggest a move */ i = fmtbuf[1] == 'b' ? BLACK : WHITE; snprintf(fmtbuf, sizeof fmtbuf, "suggest %c %s", i == BLACK ? 'B' : 'W', stoc(pickmove(i))); dlog(fmtbuf); goto top; case 'f': /* go forward a move */ board[movelog[movenum - 1]].s_occ = movenum & 1 ? BLACK : WHITE; movenum++; bdisp(); goto top; case 'l': /* print move history */ if (fmtbuf[1] == '\0') { for (i = 0; i < movenum - 1; i++) dlog(stoc(movelog[i])); goto top; } if ((fp = fopen(fmtbuf + 1, "w")) == NULL) goto top; for (i = 0; i < movenum - 1; i++) { fprintf(fp, "%s", stoc(movelog[i])); if (++i < movenum - 1) fprintf(fp, " %s\n", stoc(movelog[i])); else fputc('\n', fp); } bdump(fp); fclose(fp); goto top; case 'o': n = 0; for (str = fmtbuf + 1; *str; str++) if (*str == ',') { for (d1 = 0; d1 < 4; d1++) if (str[-1] == pdir[d1]) break; str[-1] = '\0'; sp = &board[s1 = ctos(fmtbuf + 1)]; n = (sp->s_frame[d1] - frames) * FAREA; *str++ = '\0'; break; } sp = &board[s2 = ctos(str)]; while (*str) str++; for (d2 = 0; d2 < 4; d2++) if (str[-1] == pdir[d2]) break; n += sp->s_frame[d2] - frames; str = fmtbuf; snprintf(str, fmtbuf + sizeof fmtbuf - str, "overlap %s%c,", stoc(s1), pdir[d1]); str += strlen(str); snprintf(str, fmtbuf + sizeof fmtbuf - str, "%s%c = %x", stoc(s2), pdir[d2], overlap[n]); dlog(fmtbuf); goto top; case 'p': sp = &board[i = ctos(fmtbuf + 1)]; snprintf(fmtbuf, sizeof fmtbuf, "V %s %x/%d %d %x/%d %d %d %x", stoc(i), sp->s_combo[BLACK].s, sp->s_level[BLACK], sp->s_nforce[BLACK], sp->s_combo[WHITE].s, sp->s_level[WHITE], sp->s_nforce[WHITE], sp->s_wval, sp->s_flg); dlog(fmtbuf); snprintf(fmtbuf, sizeof fmtbuf, "FB %s %x %x %x %x", stoc(i), sp->s_fval[BLACK][0].s, sp->s_fval[BLACK][1].s, sp->s_fval[BLACK][2].s, sp->s_fval[BLACK][3].s); dlog(fmtbuf); snprintf(fmtbuf, sizeof fmtbuf, "FW %s %x %x %x %x", stoc(i), sp->s_fval[WHITE][0].s, sp->s_fval[WHITE][1].s, sp->s_fval[WHITE][2].s, sp->s_fval[WHITE][3].s); dlog(fmtbuf); goto top; case 'e': /* e {b|w} [0-9] spot */ str = fmtbuf + 1; if (*str >= '0' && *str <= '9') n = *str++ - '0'; else n = 0; sp = &board[i = ctos(str)]; for (ep = sp->s_empty; ep; ep = ep->e_next) { cbp = ep->e_combo; if (n) { if (cbp->c_nframes > n) continue; if (cbp->c_nframes != n) break; } printcombo(cbp, fmtbuf, sizeof fmtbuf); dlog(fmtbuf); } goto top; default: syntax: dlog("Options are:"); dlog("q - quit"); dlog("c - continue"); dlog("d# - set debug level to #"); dlog("p# - print values at #"); goto top; } } #endif /* DEBUG */ /* * Display debug info. */ void dlog(str) char *str; { if (debugfp) fprintf(debugfp, "%s\n", str); if (interactive) dislog(str); else fprintf(stderr, "%s\n", str); } void logit(str) char *str; { if (debugfp) fprintf(debugfp, "%s\n", str); if (interactive) dislog(str); else printf("%s\n", str); } /* * Deal with a fatal error. */ void qlog(str) char *str; { dlog(str); if (interactive) beep(); sleep(5); quit(0); } /* ARGSUSED */ void quit(sig) int sig; { if (interactive) { bdisp(); /* show final board */ cursfini(); } exit(0); } /* * Die gracefully. */ void panic(str) char *str; { fprintf(stderr, "%s: %s\n", __progname, str); fputs("resign\n", stdout); quit(0); }