2 Copyright (c) 1993 Richard V. Nash.
3 Copyright (c) 2000 Dan Papasian
4 Copyright (C) Andrew Tridgell 2002
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software
18 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
23 static int check_kings(struct game_state_t *gs);
24 static int get_empty_slot(void);
25 static int ReadGameAttrs_common(FILE *fp, int g, int version);
26 static long OldestHistGame(char *login);
27 static void RemoveHistGame(char *file, int maxlines);
28 static void write_g_out(int g, char *file, int maxlines, int isDraw, char *EndSymbol, char *name, time_t *now);
29 static int game_zero(int g);
31 const char *TypeStrings[NUM_GAMETYPES] = {"untimed", "blitz", "standard",
32 "nonstandard", "wild", "lightning",
33 "bughouse", "gothic", "knightmate",
36 /* this method is awful! how about allocation as we need it and freeing
38 static int get_empty_slot(void)
42 for (i = 0; i < game_globals.g_num; i++) {
43 if (game_globals.garray[i].status == GAME_EMPTY)
47 game_globals.garray = (struct game *)realloc(game_globals.garray, sizeof(struct game) * game_globals.g_num);
48 /* yeah great, bet this causes lag! - DAV*/
49 /* I have serious doubt of the truth to the above client- bugg */
50 game_globals.garray[game_globals.g_num - 1].status = GAME_EMPTY;
51 return game_globals.g_num - 1;
56 int new = get_empty_slot();
61 static int game_zero(int g)
63 ZERO_STRUCT(game_globals.garray[g]);
65 game_globals.garray[g].white = -1;
66 game_globals.garray[g].black = -1;
68 game_globals.garray[g].status = GAME_NEW;
69 game_globals.garray[g].link = -1;
70 game_globals.garray[g].result = END_NOTENDED;
71 game_globals.garray[g].type = TYPE_UNTIMED;
72 game_globals.garray[g].game_state.gameNum = g;
73 game_globals.garray[g].wInitTime = 300; /* 5 minutes */
74 game_globals.garray[g].wIncrement = 0;
75 game_globals.garray[g].bInitTime = 300; /* 5 minutes */
76 game_globals.garray[g].bIncrement = 0;
77 game_globals.garray[g].flag_pending = FLAG_NONE;
78 strcpy(game_globals.garray[g].FENstartPos,INITIAL_FEN);
84 FREE(game_globals.garray[g].moveList);
85 FREE(game_globals.garray[g].examMoveList);
86 game_globals.garray[g].moveList = NULL;
87 game_globals.garray[g].examMoveList = NULL;
88 game_globals.garray[g].moveListSize = 0;
89 game_globals.garray[g].examMoveListSize = 0;
93 int game_remove(int g)
95 /* Should remove game from players observation list */
98 game_globals.garray[g].status = GAME_EMPTY;
102 /* old moves not stored now - uses smoves */
103 int game_finish(int g)
105 player_game_ended(g); /* Alert playerdb that game ended */
110 void MakeFENpos (int g, char *FEN)
112 strcpy(FEN, boardToFEN(g));
115 static char *game_time_str(int wt, int winc, int bt, int binc)
117 static char tstr[50];
119 if ((!wt) && (!winc)) { /* Untimed */
123 if ((wt == bt) && (winc == binc)) {
124 sprintf(tstr, " %d %d", wt, winc);
126 sprintf(tstr, " %d %d : %d %d", wt, winc, bt, binc);
131 const char *bstr[] = {"untimed", "blitz", "standard", "non-standard", "wild", "lightning", "Bughouse", "Gothic", "Knightmate", "Capablanca"};
133 const char *rstr[] = {"unrated", "rated"};
135 char *game_str(int rated, int wt, int winc, int bt, int binc,
136 char *cat, char *board)
138 static char tstr[200];
140 if (cat && cat[0] && board && board[0] &&
141 (strcmp(cat, "standard") || strcmp(board, "standard"))) {
142 sprintf(tstr, "%s %s%s Loaded from %s/%s",
144 bstr[game_isblitz(wt / 60, winc, bt / 60, binc, cat, board)],
145 game_time_str(wt / 60, winc, bt / 60, binc),
148 sprintf(tstr, "%s %s%s",
150 bstr[game_isblitz(wt / 60, winc, bt / 60, binc, cat, board)],
151 game_time_str(wt / 60, winc, bt / 60, binc));
156 int game_isblitz(int wt, int winc, int bt, int binc,
157 char *cat, char *board)
162 if (!strcmp(cat, "bughouse"))
163 return TYPE_BUGHOUSE;
164 if (!strcmp(cat, "gothic"))
166 if (!strcmp(cat, "knightmate"))
167 return TYPE_KNIGHTMATE;
168 if (!strcmp(cat, "capablanca"))
169 return TYPE_CAPABLANCA;
170 if (board && board[0]) {
171 if (!strcmp(cat, "wild"))
173 if (strcmp(cat, "standard") || strcmp(board, "standard"))
174 return TYPE_NONSTANDARD;
178 if ((wt == 0) || (bt == 0))
179 /* nonsense if one is timed and one is not */
182 if ((wt != bt) || (winc != binc))
183 return TYPE_NONSTANDARD;
184 total = wt * 60 + winc * 40;
185 if (total < 180) /* 3 minute */
187 if (total >= 900) /* 15 minutes */
193 void send_board_to(int g, int p)
195 struct player *pp = &player_globals.parray[p];
200 /* since we know g and p, figure out our relationship to this game */
203 if ((game_globals.garray[g].status == GAME_EXAMINE) || (game_globals.garray[g].status == GAME_SETUP)) {
212 relation = ((side == game_globals.garray[g].game_state.onMove) ? 1 : -1);
218 if (CheckPFlag(p, PFLAG_FLIP)) { /* flip board? */
225 b = board_to_string(game_globals.garray[g].white_name,
226 game_globals.garray[g].black_name,
227 game_globals.garray[g].wTime,
228 game_globals.garray[g].bTime,
229 &game_globals.garray[g].game_state,
230 (game_globals.garray[g].status == GAME_EXAMINE || game_globals.garray[g].status == GAME_SETUP) ?
231 game_globals.garray[g].examMoveList : game_globals.garray[g].moveList,
236 if (pp->game == g && net_globals.con[pp->socket]->timeseal) {
237 pprintf_noformat(p, "\n%s\n[G]\n", b);
239 pprintf_noformat(p, "\n%s", b);
242 if (p != command_globals.commanding_player) {
247 void send_boards(int g)
250 struct simul_info_t *simInfo = player_globals.parray[game_globals.garray[g].white].simul_info;
251 int which_board = -1;
253 /* Begin code added 3/28/96 - Figbert */
255 if ((game_globals.garray[g].status != GAME_SETUP) &&
256 (check_kings(&game_globals.garray[g].game_state))) {
257 d_printf( "Game has invalid number of kings. Aborting...\n");
259 d_printf("files = %d, ranks = %d\n", game_globals.garray[g].game_state.files, game_globals.garray[g].game_state.ranks);
260 for(r=0; r<game_globals.garray[g].game_state.ranks; r++) {
261 for(f=0; f<game_globals.garray[g].game_state.files; f++)
262 d_printf("b[%d][%d]=%d\n",f,r,game_globals.garray[g].game_state.board[f][r]);
267 for (p = 0; p < player_globals.p_num; p++) {
268 struct player *pp = &player_globals.parray[p];
269 if (pp->status == PLAYER_EMPTY)
275 if (p1 >= 0 && player_globals.parray[p1].game == g) {
276 pprintf (p1, "Disconnecting you from game number %d.\n", g+1);
277 player_globals.parray[p1].game = -1;
280 pprintf (p, "Disconnecting you from game number %d.\n", g+1);
287 /* End code added 3/28/96 - Figbert */
290 which_board = simInfo->boards[simInfo->onBoard];
292 if ((simInfo == NULL) || (which_board == g)) {
293 for (p = 0; p < player_globals.p_num; p++) {
294 struct player *pp = &player_globals.parray[p];
295 if (pp->status == PLAYER_EMPTY)
297 if (player_is_observe(p, g) || (pp->game == g))
303 void game_update_time(int g)
306 unsigned now, timesince;
312 gg = &game_globals.garray[g];
314 /* no update on first move */
315 if (gg->game_state.moveNum == 1)
317 if (gg->clockStopped)
319 if (gg->type == TYPE_UNTIMED)
322 timesince = now - gg->lastDecTime;
323 if (gg->game_state.onMove == WHITE) {
324 gg->wTime -= timesince;
326 gg->bTime -= timesince;
328 gg->lastDecTime = now;
332 static void game_update_times(void)
336 for (g = 0; g < game_globals.g_num; g++) {
337 if (game_globals.garray[g].status != GAME_ACTIVE)
339 if (game_globals.garray[g].clockStopped)
346 char *EndString(int g, int personal)
348 static char endstr[200];
349 char *blackguy, *whiteguy;
350 static char blackstr[] = "Black";
351 static char whitestr[] = "White";
353 blackguy = (personal ? game_globals.garray[g].black_name : blackstr);
354 whiteguy = (personal ? game_globals.garray[g].white_name : whitestr);
356 switch (game_globals.garray[g].result) {
358 sprintf(endstr, "%s checkmated",
359 game_globals.garray[g].winner == WHITE ? blackguy : whiteguy);
362 sprintf(endstr, "%s bared",
363 game_globals.garray[g].winner == WHITE ? blackguy : whiteguy);
366 sprintf(endstr, "%s perpetually checking",
367 game_globals.garray[g].winner == WHITE ? blackguy : whiteguy);
370 sprintf(endstr, "%s resigned",
371 game_globals.garray[g].winner == WHITE ? blackguy : whiteguy);
374 sprintf(endstr, "%s ran out of time",
375 game_globals.garray[g].winner == WHITE ? blackguy : whiteguy);
378 sprintf(endstr, "Game drawn by mutual agreement");
381 sprintf(endstr, "Game drawn because both players ran out of time");
384 sprintf(endstr, "Game drawn by repetition");
387 sprintf(endstr, "Draw by the 50 move rule");
390 sprintf(endstr, "Game adjourned by mutual agreement");
392 case END_LOSTCONNECTION:
393 sprintf(endstr, "%s lost connection, game adjourned",
394 game_globals.garray[g].winner == WHITE ? whiteguy : blackguy);
397 sprintf(endstr, "Game aborted by mutual agreement");
400 sprintf(endstr, "Stalemate.");
403 sprintf(endstr, "Still in progress");
406 sprintf(endstr, "Game courtesyaborted by %s",
407 game_globals.garray[g].winner == WHITE ? whiteguy : blackguy);
409 case END_COURTESYADJOURN:
410 sprintf(endstr, "Game courtesyadjourned by %s",
411 game_globals.garray[g].winner == WHITE ? whiteguy : blackguy);
414 sprintf(endstr, "Game drawn because neither player has mating material");
416 case END_FLAGNOMATERIAL:
417 sprintf(endstr, "%s ran out of time and %s has no material to mate",
418 game_globals.garray[g].winner == WHITE ? blackguy : whiteguy,
419 game_globals.garray[g].winner == WHITE ? whiteguy : blackguy);
422 sprintf(endstr, "Game drawn by adjudication");
425 sprintf(endstr, "%s wins by adjudication",
426 game_globals.garray[g].winner == WHITE ? whiteguy : blackguy);
429 sprintf(endstr, "Game aborted by adjudication");
432 sprintf(endstr, "???????");
439 const char *EndSym(int g)
441 static const char *symbols[] = {"1-0", "0-1", "1/2-1/2", "*"};
443 switch (game_globals.garray[g].result) {
450 return ((game_globals.garray[g].winner == WHITE) ? symbols[0] : symbols[1]);
458 case END_FLAGNOMATERIAL:
469 /* This should be enough to hold any game up to at least 8000 moves
470 * If we overwrite this, the server will crash :-).
472 /* 8000? who you trying to kid? this is awful - enough for 600 halfs :) -DAV*/
473 #define GAME_STRING_LEN 19000
474 static char gameString[GAME_STRING_LEN];
475 char *movesToString(int g, int pgn)
481 struct move_t *moves;
483 if (game_globals.garray[g].status == GAME_EXAMINE || game_globals.garray[g].status == GAME_SETUP)
484 moves = game_globals.garray[g].examMoveList;
485 else moves = game_globals.garray[g].moveList;
487 wr = game_globals.garray[g].white_rating;
488 br = game_globals.garray[g].black_rating;
491 curTime = untenths(game_globals.garray[g].timeOfStart);
495 "\n[Event \"%s %s %s game\"]\n"
496 "[Site \"%s, %s\"]\n",
497 config_get_tmp("SERVER_NAME"),
498 rstr[game_globals.garray[g].rated], /*bstr[game_globals.garray[g].type],*/
499 game_globals.garray[g].variant, // [HGM] allow more variation in game_types
500 config_get_tmp("SERVER_NAME"),
501 config_get_tmp("SERVER_LOCATION"));
502 strftime(tmp, sizeof(tmp),
503 "[Date \"%Y.%m.%d\"]\n"
504 "[Time \"%H:%M:%S\"]\n",
505 localtime((time_t *) &curTime));
506 strcat(gameString, tmp);
511 "[WhiteElo \"%d\"]\n"
512 "[BlackElo \"%d\"]\n",
513 game_globals.garray[g].white_name, game_globals.garray[g].black_name, wr, br);
514 strcat(gameString, tmp);
515 if(game_globals.garray[g].game_state.variant[0]) { // [HGM] variant: print variant tag
517 "[Variant \"%s\"]\n",
518 game_globals.garray[g].game_state.variant);
519 strcat(gameString, tmp);
521 if(game_globals.garray[g].game_state.setup) { // [HGM] setup: print the FEN
525 game_globals.garray[g].FENstartPos);
526 strcat(gameString, tmp);
529 "[TimeControl \"%d+%d\"]\n"
531 "[Result \"%s\"]\n\n",
532 game_globals.garray[g].wInitTime / 10, game_globals.garray[g].wIncrement / 10, EndSym(g));
533 strcat(gameString, tmp);
536 for (i = 0; i < game_globals.garray[g].numHalfMoves; i++) {
537 if (moves[i].color == WHITE) {
538 if ((col += sprintf(tmp, "%d. ", (i+1) / 2 + 1)) > 70) {
539 strcat(gameString, "\n");
542 strcat(gameString, tmp);
544 strcat (tmp, "1. ... ");
547 if ((col += sprintf(tmp, "%s ", moves[i].algString)) > 70) {
548 strcat(gameString, "\n");
551 strcat(gameString, tmp);
552 if(moves[i].depth > 0) { // [HGM] computer game, add {score/depth} comment
553 if ((col += sprintf(tmp, "{%s%.2f/%d} ", moves[i].score > 0 ? "+" : "",
554 moves[i].score, moves[i].depth)) > 70) {
555 strcat(gameString, "\n");
558 strcat(gameString, tmp);
561 strcat(gameString, "\n");
565 sprintf(gameString, "\n%s ", game_globals.garray[g].white_name);
567 sprintf(tmp, "(%d) ", wr);
569 sprintf(tmp, "(UNR) ");
571 strcat(gameString, tmp);
572 sprintf(tmp, "vs. %s ", game_globals.garray[g].black_name);
573 strcat(gameString, tmp);
575 sprintf(tmp, "(%d) ", br);
577 sprintf(tmp, "(UNR) ");
579 strcat(gameString, tmp);
580 strcat(gameString, "--- ");
581 strcat(gameString, strltime(&curTime));
582 if (game_globals.garray[g].rated) {
583 strcat(gameString, "\nRated ");
585 strcat(gameString, "\nUnrated ");
587 // strcat (gameString, TypeStrings[game_globals.garray[g].type]);
588 strcat (gameString, game_globals.garray[g].variant);
589 strcat(gameString, " match, initial time: ");
590 if ((game_globals.garray[g].bInitTime != game_globals.garray[g].wInitTime) || (game_globals.garray[g].wIncrement != game_globals.garray[g].bIncrement)) { /* different starting times */
591 sprintf(tmp, "%d minutes, increment: %d seconds AND %d minutes, increment: %d seconds.\n\n", game_globals.garray[g].wInitTime / 600, game_globals.garray[g].wIncrement / 10, game_globals.garray[g].bInitTime / 600, game_globals.garray[g].bIncrement / 10);
593 sprintf(tmp, "%d minutes, increment: %d seconds.\n\n", game_globals.garray[g].wInitTime / 600, game_globals.garray[g].wIncrement / 10);
595 strcat(gameString, tmp);
597 if(game_globals.garray[g].game_state.setup) { // [HGM] setup: print the initial position board
598 char *q; struct game_state_t initial_gs; struct move_t ml[600]; int r, f;
600 initial_gs.gameNum = g;
601 initial_gs.wkrmoved = 0; // [HGM] for some reason calling reset_board_vars() does not work here
602 initial_gs.wqrmoved = 0; // so I just duplicated the code and pasted it here...
603 initial_gs.wkmoved = 0;
604 initial_gs.bkrmoved = 0;
605 initial_gs.bqrmoved = 0;
606 initial_gs.bkmoved = 0;
607 initial_gs.onMove = WHITE;
608 initial_gs.lastIrreversable = -1;
609 initial_gs.files = game_globals.garray[g].game_state.files;
610 initial_gs.ranks = game_globals.garray[g].game_state.ranks;
611 initial_gs.holdings = game_globals.garray[g].game_state.holdings;
612 strcpy(initial_gs.variant, game_globals.garray[g].game_state.variant);
613 for (f = 0; f < 2; f++) {
614 for (r = 0; r < initial_gs.files; r++)
615 initial_gs.ep_possible[f][r] = 0;
616 for (r = PAWN; r <= PIECES-1; r++)
617 initial_gs.holding[f][r-PAWN] = 0;
619 FEN_to_board(game_globals.garray[g].FENstartPos ,&initial_gs);
621 kludgeFlag = 1; // [HGM] setup: this is not thread safe. Must it be???
623 game_globals.garray[g].white_name,
624 game_globals.garray[g].black_name,
625 game_globals.garray[g].wTime,
626 game_globals.garray[g].bTime,
633 strcat(gameString, q);
634 strcat(gameString, "\n");
636 sprintf(tmp, "Move %-19s%-19s\n", game_globals.garray[g].white_name, game_globals.garray[g].black_name);
637 strcat(gameString, tmp);
638 strcat(gameString, "---- ---------------- ----------------\n");
640 for (i = 0; i < game_globals.garray[g].numHalfMoves; i += 2) {
641 if (i==0 && (moves[i].color == BLACK)) {
642 sprintf(tmp, "%3d. %-16s %-16s\n", (i+1)/2 + 1, "...",
643 move_and_time(&moves[i]));
645 } else if (i + 1 < game_globals.garray[g].numHalfMoves) {
646 sprintf(tmp, "%3d. %-16s ", (i+1)/2 + 1, move_and_time(&moves[i]));
647 strcat(gameString, tmp);
648 sprintf(tmp, "%-16s\n", move_and_time(&moves[i+1]));
650 sprintf(tmp, "%3d. %-16s\n", (i+1)/2 + 1, move_and_time(&moves[i]));
652 strcat(gameString, tmp);
653 if (strlen(gameString) > GAME_STRING_LEN - 100) { /* Bug out if getting
654 close to filling this
660 strcat(gameString, " ");
663 sprintf(tmp, "{%s} %s\n", EndString(g, 0), EndSym(g));
664 strcat(gameString, tmp);
669 void game_disconnect(int g, int p)
671 game_ended(g, (game_globals.garray[g].white == p) ? WHITE : BLACK, END_LOSTCONNECTION);
674 int CharToPiece(char c, char *variant)
678 if(!strcmp(variant, "shogi")) {
681 return W_HONORABLEHORSE;
683 return B_HONORABLEHORSE;
697 } else if(!strcmp(variant, "xiangqi")) {
712 } else if(!strcmp(variant, "super")) {
731 } else if(!strcmp(variant, "spartan")) {
780 if(!strcmp(variant, "seirawan")) return W_ELEPHANT;
783 if(!strcmp(variant, "seirawan")) return B_ELEPHANT;
798 char PieceToChar(int piece)
801 case W_PAWN:return 'P';
805 case W_HONORABLEHORSE:
808 case B_HONORABLEHORSE:
919 write a new style generic game file
921 static void WriteGameFile_v100(FILE * fp, int g)
923 struct game gg = game_globals.garray[g];
926 /* zero any elements we don't want to save */
927 memset(&gg, 0, offsetof(struct game, not_saved_marker));
928 gg.game_state.gameNum = 0;
930 /* marshall it into a string */
931 s = marshall_game(&gg);
933 d_printf("Unable to marshall game structure!\n");
938 fprintf(fp, "v 100\n%s\n", s);
944 read a game file using the new generic and extensible format
946 static int ReadGameAttrs_v100(FILE *fp, int g)
949 struct game *gg = &game_globals.garray[g];
952 s = fd_load(fileno(fp), NULL);
954 d_printf("Error reading game file!\n");
958 /* skip first line */
959 s2 = strchr(s, '\n');
961 /* remember the game state for later */
964 /* the marshaller doesn't save zero elements, but some elements don't
965 default to zero. Make sure they get the right value */
966 memset(&gg->not_saved_marker, 0,
967 sizeof(struct game) - offsetof(struct game, not_saved_marker));
968 gg->game_state.gameNum = g;
970 if (!s2 || unmarshall_game(gg, s2) != 0) {
971 d_printf("Error unmarshalling game data!\n");
977 /* when examining we are not supposed to restore the game
978 state, so put it back here */
979 if (g1.status == GAME_EXAMINE || g1.status == GAME_SETUP) {
980 gg->game_state = g1.game_state;
982 gg->status = g1.status;
984 /* cope with continuing a game with timeseal that was started without it */
985 gg->wRealTime = gg->wTime * 100;
986 gg->bRealTime = gg->bTime * 100;
992 static int ReadGameAttrs_common(FILE * fp, int g,int version)
994 if (version == 9 || version == 100) {
995 return ReadGameAttrs_v100(fp, g);
998 return ReadGameAttrs_old(fp, g, version);
1001 int ReadGameAttrs_exam(FILE * fp, int g)
1004 char line[MAX_GLINE_SIZE];
1006 fgets(line, MAX_GLINE_SIZE - 1, fp);
1008 if (line[0] == 'v') {
1009 sscanf(line, "%*c %d", &version);
1014 return ReadGameAttrs_common(fp,g,version);
1020 int ReadGameAttrs(FILE * fp, int g)
1023 char line[MAX_GLINE_SIZE];
1025 fgets(line, MAX_GLINE_SIZE - 1, fp);
1027 if (line[0] == 'v') {
1028 sscanf(line, "%*c %d", &version);
1030 return ReadGameAttrs_common(fp,g,version);
1033 int game_read(int g, int wp, int bp)
1038 game_globals.garray[g].white = wp;
1039 game_globals.garray[g].black = bp;
1040 /* game_globals.garray[g].old_white = -1;
1041 game_globals.garray[g].old_black = -1;
1043 game_globals.garray[g].moveListSize = 0;
1044 game_globals.garray[g].game_state.gameNum = g;
1045 strcpy(game_globals.garray[g].white_name, player_globals.parray[wp].name);
1046 strcpy(game_globals.garray[g].black_name, player_globals.parray[bp].name);
1047 if (game_globals.garray[g].type == TYPE_BLITZ) {
1048 game_globals.garray[g].white_rating = player_globals.parray[wp].b_stats.rating;
1049 game_globals.garray[g].black_rating = player_globals.parray[bp].b_stats.rating;
1050 } else if (game_globals.garray[g].type == TYPE_WILD ||
1051 game_globals.garray[g].type == TYPE_KNIGHTMATE ||
1052 game_globals.garray[g].type == TYPE_CAPABLANCA ||
1053 game_globals.garray[g].type == TYPE_GOTHIC) {
1054 game_globals.garray[g].white_rating = player_globals.parray[wp].w_stats.rating;
1055 game_globals.garray[g].black_rating = player_globals.parray[bp].w_stats.rating;
1056 } else if (game_globals.garray[g].type == TYPE_LIGHT) {
1057 game_globals.garray[g].white_rating = player_globals.parray[wp].l_stats.rating;
1058 game_globals.garray[g].black_rating = player_globals.parray[bp].l_stats.rating;
1059 } else if (game_globals.garray[g].type == TYPE_BUGHOUSE) {
1060 game_globals.garray[g].white_rating = player_globals.parray[wp].bug_stats.rating;
1061 game_globals.garray[g].black_rating = player_globals.parray[bp].bug_stats.rating;
1063 game_globals.garray[g].white_rating = player_globals.parray[wp].s_stats.rating;
1064 game_globals.garray[g].black_rating = player_globals.parray[bp].s_stats.rating;
1066 fp = fopen_p("%s/%c/%s-%s", "r", ADJOURNED_DIR, player_globals.parray[wp].login[0],
1067 player_globals.parray[wp].login, player_globals.parray[bp].login);
1071 for (piece=PAWN; piece < KING; piece++) {
1072 game_globals.garray[g].game_state.holding[0][piece-PAWN]
1073 = game_globals.garray[g].game_state.holding[1][piece-PAWN] = 0;
1075 if (ReadGameAttrs(fp, g) < 0) {
1081 if (game_globals.garray[g].result == END_ADJOURN
1082 || game_globals.garray[g].result == END_COURTESYADJOURN)
1083 game_globals.garray[g].result = END_NOTENDED;
1084 game_globals.garray[g].status = GAME_ACTIVE;
1085 game_globals.garray[g].startTime = tenth_secs();
1086 game_globals.garray[g].lastMoveTime = game_globals.garray[g].startTime;
1087 game_globals.garray[g].lastDecTime = game_globals.garray[g].startTime;
1089 /* cope with continuing a game with timeseal that was started without it */
1090 game_globals.garray[g].wRealTime = game_globals.garray[g].wTime * 100;
1091 game_globals.garray[g].bRealTime = game_globals.garray[g].bTime * 100;
1093 /* Need to do notification and pending cleanup */
1097 int game_delete(int wp, int bp)
1099 char fname[MAX_FILENAME_SIZE];
1100 char lname[MAX_FILENAME_SIZE];
1102 sprintf(fname, "%s/%c/%s-%s", ADJOURNED_DIR, player_globals.parray[wp].login[0],
1103 player_globals.parray[wp].login, player_globals.parray[bp].login);
1104 sprintf(lname, "%s/%c/%s-%s", ADJOURNED_DIR, player_globals.parray[bp].login[0],
1105 player_globals.parray[wp].login, player_globals.parray[bp].login);
1111 int game_save(int g)
1114 struct player *wp, *bp;
1115 struct game *gg = &game_globals.garray[g];
1118 wp = &player_globals.parray[gg->white];
1119 bp = &player_globals.parray[gg->black];
1120 asprintf(&fname, "%s/%c/%s-%s", ADJOURNED_DIR, wp->login[0],
1121 wp->login, bp->login);
1122 fp = fopen_p("%s", "w", fname);
1124 d_printf("CHESSD: Problem opening adjourn file %s-%s for write\n",
1125 wp->login, bp->login);
1129 WriteGameFile_v100(fp, g);
1131 /* Create link for easier stored game finding */
1132 if (bp->login[0] != wp->login[0]) {
1134 asprintf(&lname, "%s/%c/%s-%s", ADJOURNED_DIR, bp->login[0],
1135 wp->login, bp->login);
1143 static long OldestHistGame(char *login)
1148 fp = fopen_p("%s/player_data/%c/%s.%s", "r", STATS_DIR,
1149 login[0], login, STATS_GAMES);
1151 fp = fopen_p("%s/player_data/%c/.rem.%s.%s", "r", STATS_DIR,
1152 login[0], login, STATS_GAMES);
1156 fscanf(fp, "%*d %*c %*d %*c %*d %*s %*s %*d %*d %*d %*d %*s %*s %ld",
1162 static void RemoveHistGame(char *file, int maxlines)
1165 char GameFile[MAX_FILENAME_SIZE];
1166 char Opponent[MAX_LOGIN_NAME];
1167 char line[MAX_LINE_SIZE];
1171 fp = fopen_p("%s", "r", file);
1175 fgets(line, MAX_LINE_SIZE - 1, fp);
1176 sscanf(line, "%*d %*c %*d %*c %*d %s %*s %*d %*d %*d %*d %*s %*s %ld",
1181 fgets(line, MAX_LINE_SIZE - 1, fp);
1188 if (count > maxlines) {
1189 truncate_file(file, maxlines);
1191 oppWhen = OldestHistGame(Opponent);
1192 if (oppWhen > When || oppWhen <= 0L) {
1193 sprintf(GameFile, "%s/%ld/%ld", HISTORY_DIR, When % 100, When);
1199 void RemHist(char *who)
1202 char Opp[MAX_LOGIN_NAME];
1205 fp = fopen_p("%s/player_data/%c/%s.%s", "r",
1207 who[0], who, STATS_GAMES);
1212 fscanf(fp, "%*d %*c %*d %*c %*d %s %*s %*d %*d %*d %*d %*s %*s %ld",
1215 oppWhen = OldestHistGame(Opp);
1216 if (oppWhen > When || oppWhen <= 0L) {
1218 asprintf(&fName, "%s/%ld/%ld", HISTORY_DIR, When % 100, When);
1226 static void write_g_out(int g, char *file, int maxlines, int isDraw,
1227 char *EndSymbol, char *name, time_t *now)
1239 wp = game_globals.garray[g].white;
1240 bp = game_globals.garray[g].black;
1242 if (game_globals.garray[g].private) {
1247 if (game_globals.garray[g].type == TYPE_BLITZ) {
1248 wr = player_globals.parray[wp].b_stats.rating;
1249 br = player_globals.parray[bp].b_stats.rating;
1251 } else if (game_globals.garray[g].type == TYPE_WILD ||
1252 game_globals.garray[g].type == TYPE_KNIGHTMATE ||
1253 game_globals.garray[g].type == TYPE_CAPABLANCA ||
1254 game_globals.garray[g].type == TYPE_GOTHIC) {
1255 wr = player_globals.parray[wp].w_stats.rating;
1256 br = player_globals.parray[bp].w_stats.rating;
1258 } else if (game_globals.garray[g].type == TYPE_STAND) {
1259 wr = player_globals.parray[wp].s_stats.rating;
1260 br = player_globals.parray[bp].s_stats.rating;
1262 } else if (game_globals.garray[g].type == TYPE_LIGHT) {
1263 wr = player_globals.parray[wp].l_stats.rating;
1264 br = player_globals.parray[bp].l_stats.rating;
1266 } else if (game_globals.garray[g].type == TYPE_BUGHOUSE) {
1267 wr = player_globals.parray[wp].bug_stats.rating;
1268 br = player_globals.parray[bp].bug_stats.rating;
1273 if (game_globals.garray[g].type == TYPE_NONSTANDARD)
1278 if (game_globals.garray[g].rated) {
1285 fp = fopen_s(file, "r");
1288 fgets(tmp, 1024, fp);
1289 sscanf(ptmp, "%d", &count);
1292 count = (count + 1) % 100;
1294 fp = fopen_s(file, "a");
1300 /* Counter Result MyRating MyColor OppRating OppName [pbr 2 12 2 12] ECO End Date */
1301 if (name == player_globals.parray[wp].name) {
1304 else if (game_globals.garray[g].winner == WHITE)
1309 fprintf(fp, "%d %c %d W %d %s %s %d %d %d %d %s %s %ld\n",
1310 count, cResult, wr, br, player_globals.parray[bp].name, type,
1311 game_globals.garray[g].wInitTime, game_globals.garray[g].wIncrement,
1312 game_globals.garray[g].bInitTime, game_globals.garray[g].bIncrement,
1319 else if (game_globals.garray[g].winner == BLACK)
1324 fprintf(fp, "%d %c %d B %d %s %s %d %d %d %d %s %s %ld\n",
1325 count, cResult, br, wr, player_globals.parray[wp].name, type,
1326 game_globals.garray[g].wInitTime, game_globals.garray[g].wIncrement,
1327 game_globals.garray[g].bInitTime, game_globals.garray[g].bIncrement,
1334 RemoveHistGame(file, maxlines);
1337 /* Test if entry is present - 1 yes 0 no */
1338 /* returns -1 if an error */
1340 char get_journalgame_type(int p,char* fname,char slot)
1343 struct player *pp = &player_globals.parray[p];
1347 FILE* fp = fopen_s(fname,"r");
1349 d_printf("Corrupt journal file! %s\n",fname);
1350 pprintf (p, "The journal file is corrupt! See an admin.\n");
1355 if (fscanf(fp, "%c %*s %*d %*s %*d %s %*d %*d %*s %*s %*s\n",
1356 &cur_slot,type) != 2) {
1357 d_printf( "CHESSD: Error in journal info format for player %s.\n",
1359 pprintf(p, "The journal file is corrupt! Error in internal format.\n");
1362 if (cur_slot == slot) {
1375 /* Returns 1 if successful */
1377 int removejournalitem(int p, char slot,FILE* fp,char* fname, int* empty)
1385 fp_new = fopen_s(fname, "w");
1387 d_printf("Can't write to journal %s!\n",fname);
1388 pprintf (p, "Was unable to write to the file! See an admin.\n");
1391 j = (struct journal*) malloc(sizeof(struct journal));
1393 if (fscanf(fp, "%c %s %d %s %d %s %d %d %s %s %s\n",
1404 d_printf( "CHESSD: Error in journal info format. %s\n", fname);
1405 pprintf(p, "The journal file is corrupt! Error in internal format.\n");
1410 if (slot != j->slot) {
1412 fprintf(fp_new, "%c %s %d %s %d %s %d %d %s %s %s\n",
1431 /* Find from_spot in journal list - return 0 if corrupted */
1432 int journal_get_info(int p, char from_spot,struct journal* j, char *fname)
1436 fp = fopen_s(fname, "r");
1438 d_printf("Corrupt journal file! %s\n",fname);
1439 pprintf (p, "The journal file is corrupt! See an admin.\n");
1443 if (fscanf(fp, "%c %s %d %s %d %s %d %d %s %s %s\n",
1454 d_printf( "CHESSD: Error in journal info format. %s\n", fname);
1455 pprintf(p, "The journal file is corrupt! Error in internal format.\n");
1459 if (tolower(j->slot) == from_spot) {
1468 void addjournalitem(int p,struct journal* j_add, char* fname)
1470 struct journal* j_cur;
1472 char fname2[MAX_FILENAME_SIZE];
1476 strcpy (fname2,fname);
1477 strcat (fname2,".w");
1478 fp2 = fopen_s(fname2, "w");
1480 d_printf( "CHESSD: Problem opening file %s for write\n", fname);
1481 pprintf (p, "Couldn't update journal! Report this to an admin.\n");
1484 fp = fopen_s(fname, "r");
1485 if (!fp) { /* Empty? */
1486 fprintf(fp2, "%c %s %d %s %d %s %d %d %s %s %s\n",
1487 j_add->slot, j_add->WhiteName, j_add->WhiteRating,
1488 j_add->BlackName, j_add->BlackRating,
1489 j_add->type, j_add->t, j_add->i, j_add->eco, j_add->ending,
1492 rename (fname2, fname);
1495 j_cur = (struct journal*) malloc(sizeof(struct journal));
1497 if (fscanf(fp, "%c %s %d %s %d %s %d %d %s %s %s\n",
1500 &j_cur->WhiteRating,
1502 &j_cur->BlackRating,
1504 &j_cur->t, &j_cur->i,
1507 j_cur->result) != 11) {
1508 d_printf( "CHESSD: Error in journal info format - aborting. %s\n", fname);
1514 if ((j_cur->slot >= j_add->slot) && (!have_output)) {
1516 fprintf(fp2, "%c %s %d %s %d %s %d %d %s %s %s\n",
1530 if (j_cur->slot != j_add->slot) {
1532 fprintf(fp2, "%c %s %d %s %d %s %d %d %s %s %s\n",
1546 if (!have_output) { /* Haven't written yet */
1547 fprintf(fp2, "%c %s %d %s %d %s %d %d %s %s %s\n",
1562 rename(fname2, fname);
1567 int pjournal(int p, int p1, char *fname)
1572 fp = fopen_s(fname, "r");
1574 pprintf(p, "Sorry, no journal information available.\n");
1578 j = (struct journal*) malloc(sizeof(struct journal));
1579 pprintf(p, "Journal for %s:\n", player_globals.parray[p1].name);
1580 pprintf(p, " White Rating Black Rating Type ECO End Result\n");
1582 if (fscanf(fp, "%c %s %d %s %d %s %d %d %s %s %s\n",
1593 d_printf( "CHESSD: Error in journal info format. %s\n", fname);
1598 j->WhiteName[13] = '\0'; /* only first 13 chars in name */
1599 j->BlackName[13] = '\0';
1600 pprintf(p, "%c: %-13s %4d %-13s %4d [%3s%3d%4d] %s %3s %-7s\n",
1601 j->slot, j->WhiteName, j->WhiteRating,
1602 j->BlackName, j->BlackRating,
1603 j->type, j->t / 600, j->i / 10, j->eco, j->ending,
1611 int pgames(int p, int p1, char *fname)
1615 int MyRating, OppRating;
1617 char OppName[MAX_LOGIN_NAME + 1];
1625 fp = fopen_s(fname, "r");
1627 pprintf(p, "Sorry, no game information available.\n");
1630 pprintf(p, "History for %s:\n", player_globals.parray[p1].name);
1631 pprintf(p, " Opponent Type ECO End Date\n");
1633 if (fscanf(fp, "%d %s %d %s %d %s %s %d %d %d %d %s %s %ld\n",
1645 (long *) &t) != 14) {
1646 d_printf( "CHESSD: Error in games info format. %s\n", fname);
1650 OppName[13] = '\0'; /* only first 13 chars in name */
1651 pprintf(p, "%2d: %s %4d %s %4d %-13s [%3s%3d%4d] %s %3s %s",
1652 count, result, MyRating, MyColor,
1654 type, wt / 600, wi / 10, eco, ending,
1661 void game_write_complete(int g, int isDraw, char *EndSymbol)
1663 char fname[MAX_FILENAME_SIZE];
1664 int wp = game_globals.garray[g].white, bp = game_globals.garray[g].black;
1665 time_t now = time(NULL);
1674 sprintf(fname, "%s/%ld/%ld", HISTORY_DIR, (long) now % 100, (long) now);
1675 fp = fopen_s(fname, "r");
1676 } while (fp); /* terminates when the file doesn't exist */
1678 fp = fopen_s(fname, "w");
1681 WriteGameFile_v100(fp, g);
1684 d_printf( "Trouble writing history file %s", fname);
1687 sprintf(fname, "%s/player_data/%c/%s.%s", STATS_DIR,
1688 player_globals.parray[wp].login[0], player_globals.parray[wp].login, STATS_GAMES);
1689 write_g_out(g, fname, 40, isDraw, EndSymbol, player_globals.parray[wp].name, &now);
1690 sprintf(fname, "%s/player_data/%c/%s.%s", STATS_DIR,
1691 player_globals.parray[bp].login[0], player_globals.parray[bp].login, STATS_GAMES);
1692 write_g_out(g, fname, 40, isDraw, EndSymbol, player_globals.parray[bp].name, &now);
1696 else if (game_globals.garray[g].winner == WHITE)
1700 sprintf(fname,"%s/player_data/%c/%s.gstats", STATS_DIR, player_globals.parray[game_globals.garray[g].white].login[0], player_globals.parray[game_globals.garray[g].white].login);
1701 if ((CheckPFlag(bp, PFLAG_REG)) && (CheckPFlag(wp, PFLAG_REG)) && (game_globals.garray[g].type != TYPE_WILD))
1702 game_save_playerratio(fname,player_globals.parray[game_globals.garray[g].black].name,Result,game_globals.garray[g].rated);
1705 else if (game_globals.garray[g].winner == BLACK)
1709 sprintf(fname,"%s/player_data/%c/%s.gstats", STATS_DIR, player_globals.parray[game_globals.garray[g].black].login[0], player_globals.parray[game_globals.garray[g].black].login);
1710 if ((CheckPFlag(bp, PFLAG_REG)) && (CheckPFlag(wp, PFLAG_REG)) && (game_globals.garray[g].type != TYPE_WILD))
1711 game_save_playerratio(fname,player_globals.parray[game_globals.garray[g].white].name,Result,game_globals.garray[g].rated);
1714 int game_count(void)
1718 for (g = 0; g < game_globals.g_num; g++) {
1719 if ((game_globals.garray[g].status == GAME_ACTIVE) || (game_globals.garray[g].status == GAME_EXAMINE)
1720 || (game_globals.garray[g].status == GAME_SETUP))
1723 if (count > command_globals.game_high)
1724 command_globals.game_high = count;
1728 static int check_kings(struct game_state_t *gs)
1730 /* Function written 3/28/96 by Figbert to check for 1 king each side only! */
1731 int blackking = 0, whiteking = 0;
1735 for (f = 0; f < gs->files; f++) {
1736 for (r = 0; r < gs->ranks; r++) {
1737 if (gs->board[f][r] == B_KING) blackking++;
1738 if (gs->board[f][r] == W_KING) whiteking++;
1742 if ((blackking == 1 || blackking == 2 && !strcmp(gs->variant, "spartan")) && whiteking == 1) return 0; /* Perfect! */