2 * backend.c -- Common back end for X and Windows NT versions of
4 * Copyright 1991 by Digital Equipment Corporation, Maynard,
7 * Enhancements Copyright 1992-2001, 2002, 2003, 2004, 2005, 2006,
8 * 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015 Free Software Foundation, Inc.
10 * Enhancements Copyright 2005 Alessandro Scotti
12 * The following terms apply to Digital Equipment Corporation's copyright
14 * ------------------------------------------------------------------------
17 * Permission to use, copy, modify, and distribute this software and its
18 * documentation for any purpose and without fee is hereby granted,
19 * provided that the above copyright notice appear in all copies and that
20 * both that copyright notice and this permission notice appear in
21 * supporting documentation, and that the name of Digital not be
22 * used in advertising or publicity pertaining to distribution of the
23 * software without specific, written prior permission.
25 * DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
26 * ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
27 * DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
28 * ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
29 * WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
30 * ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
32 * ------------------------------------------------------------------------
34 * The following terms apply to the enhanced version of XBoard
35 * distributed by the Free Software Foundation:
36 * ------------------------------------------------------------------------
38 * GNU XBoard is free software: you can redistribute it and/or modify
39 * it under the terms of the GNU General Public License as published by
40 * the Free Software Foundation, either version 3 of the License, or (at
41 * your option) any later version.
43 * GNU XBoard is distributed in the hope that it will be useful, but
44 * WITHOUT ANY WARRANTY; without even the implied warranty of
45 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
46 * General Public License for more details.
48 * You should have received a copy of the GNU General Public License
49 * along with this program. If not, see http://www.gnu.org/licenses/. *
51 *------------------------------------------------------------------------
52 ** See the file ChangeLog for a revision history. */
54 /* [AS] Also useful here for debugging */
58 int flock(int f, int code);
63 # define EGBB_NAME "egbbdll64.dll"
65 # define EGBB_NAME "egbbdll.dll"
70 # include <sys/file.h>
75 # define EGBB_NAME "egbbso64.so"
77 # define EGBB_NAME "egbbso.so"
79 // kludge to allow Windows code in back-end by converting it to corresponding Linux code
81 # define HMODULE void *
82 # define LoadLibrary(x) dlopen(x, RTLD_LAZY)
83 # define GetProcAddress dlsym
93 #include <sys/types.h>
102 #else /* not STDC_HEADERS */
105 # else /* not HAVE_STRING_H */
106 # include <strings.h>
107 # endif /* not HAVE_STRING_H */
108 #endif /* not STDC_HEADERS */
111 # include <sys/fcntl.h>
112 #else /* not HAVE_SYS_FCNTL_H */
115 # endif /* HAVE_FCNTL_H */
116 #endif /* not HAVE_SYS_FCNTL_H */
118 #if TIME_WITH_SYS_TIME
119 # include <sys/time.h>
123 # include <sys/time.h>
129 #if defined(_amigados) && !defined(__GNUC__)
134 extern int gettimeofday(struct timeval *, struct timezone *);
142 #include "frontend.h"
149 #include "backendz.h"
150 #include "evalgraph.h"
151 #include "engineoutput.h"
155 # define _(s) gettext (s)
156 # define N_(s) gettext_noop (s)
157 # define T_(s) gettext(s)
170 int establish P((void));
171 void read_from_player P((InputSourceRef isr, VOIDSTAR closure,
172 char *buf, int count, int error));
173 void read_from_ics P((InputSourceRef isr, VOIDSTAR closure,
174 char *buf, int count, int error));
175 void SendToICS P((char *s));
176 void SendToICSDelayed P((char *s, long msdelay));
177 void SendMoveToICS P((ChessMove moveType, int fromX, int fromY, int toX, int toY, char promoChar));
178 void HandleMachineMove P((char *message, ChessProgramState *cps));
179 int AutoPlayOneMove P((void));
180 int LoadGameOneMove P((ChessMove readAhead));
181 int LoadGameFromFile P((char *filename, int n, char *title, int useList));
182 int LoadPositionFromFile P((char *filename, int n, char *title));
183 int SavePositionToFile P((char *filename));
184 void MakeMove P((int fromX, int fromY, int toX, int toY, int promoChar));
185 void ShowMove P((int fromX, int fromY, int toX, int toY));
186 int FinishMove P((ChessMove moveType, int fromX, int fromY, int toX, int toY,
187 /*char*/int promoChar));
188 void BackwardInner P((int target));
189 void ForwardInner P((int target));
190 int Adjudicate P((ChessProgramState *cps));
191 void GameEnds P((ChessMove result, char *resultDetails, int whosays));
192 void EditPositionDone P((Boolean fakeRights));
193 void PrintOpponents P((FILE *fp));
194 void PrintPosition P((FILE *fp, int move));
195 void SendToProgram P((char *message, ChessProgramState *cps));
196 void SendMoveToProgram P((int moveNum, ChessProgramState *cps));
197 void ReceiveFromProgram P((InputSourceRef isr, VOIDSTAR closure,
198 char *buf, int count, int error));
199 void SendTimeControl P((ChessProgramState *cps,
200 int mps, long tc, int inc, int sd, int st));
201 char *TimeControlTagValue P((void));
202 void Attention P((ChessProgramState *cps));
203 void FeedMovesToProgram P((ChessProgramState *cps, int upto));
204 int ResurrectChessProgram P((void));
205 void DisplayComment P((int moveNumber, char *text));
206 void DisplayMove P((int moveNumber));
208 void ParseGameHistory P((char *game));
209 void ParseBoard12 P((char *string));
210 void KeepAlive P((void));
211 void StartClocks P((void));
212 void SwitchClocks P((int nr));
213 void StopClocks P((void));
214 void ResetClocks P((void));
215 char *PGNDate P((void));
216 void SetGameInfo P((void));
217 int RegisterMove P((void));
218 void MakeRegisteredMove P((void));
219 void TruncateGame P((void));
220 int looking_at P((char *, int *, char *));
221 void CopyPlayerNameIntoFileName P((char **, char *));
222 char *SavePart P((char *));
223 int SaveGameOldStyle P((FILE *));
224 int SaveGamePGN P((FILE *));
225 int CheckFlags P((void));
226 long NextTickLength P((long));
227 void CheckTimeControl P((void));
228 void show_bytes P((FILE *, char *, int));
229 int string_to_rating P((char *str));
230 void ParseFeatures P((char* args, ChessProgramState *cps));
231 void InitBackEnd3 P((void));
232 void FeatureDone P((ChessProgramState* cps, int val));
233 void InitChessProgram P((ChessProgramState *cps, int setup));
234 void OutputKibitz(int window, char *text);
235 int PerpetualChase(int first, int last);
236 int EngineOutputIsUp();
237 void InitDrawingSizes(int x, int y);
238 void NextMatchGame P((void));
239 int NextTourneyGame P((int nr, int *swap));
240 int Pairing P((int nr, int nPlayers, int *w, int *b, int *sync));
241 FILE *WriteTourneyFile P((char *results, FILE *f));
242 void DisplayTwoMachinesTitle P(());
243 static void ExcludeClick P((int index));
244 void ToggleSecond P((void));
245 void PauseEngine P((ChessProgramState *cps));
246 static int NonStandardBoardSize P((VariantClass v, int w, int h, int s));
249 extern void ConsoleCreate();
252 ChessProgramState *WhitePlayer();
253 int VerifyDisplayMode P(());
255 char *GetInfoFromComment( int, char * ); // [HGM] PV time: returns stripped comment
256 void InitEngineUCI( const char * iniDir, ChessProgramState * cps ); // [HGM] moved here from winboard.c
257 char *ProbeBook P((int moveNr, char *book)); // [HGM] book: returns a book move
258 char *SendMoveToBookUser P((int nr, ChessProgramState *cps, int initial)); // [HGM] book
259 void ics_update_width P((int new_width));
260 extern char installDir[MSG_SIZ];
261 VariantClass startVariant; /* [HGM] nicks: initial variant */
264 extern int tinyLayout, smallLayout;
265 ChessProgramStats programStats;
266 char lastPV[2][2*MSG_SIZ]; /* [HGM] pv: last PV in thinking output of each engine */
268 static int exiting = 0; /* [HGM] moved to top */
269 static int setboardSpoiledMachineBlack = 0 /*, errorExitFlag = 0*/;
270 int startedFromPositionFile = FALSE; Board filePosition; /* [HGM] loadPos */
271 Board partnerBoard; /* [HGM] bughouse: for peeking at partner game */
272 int partnerHighlight[2];
273 Boolean partnerBoardValid = 0;
274 char partnerStatus[MSG_SIZ];
276 Boolean originalFlip;
277 Boolean twoBoards = 0;
278 char endingGame = 0; /* [HGM] crash: flag to prevent recursion of GameEnds() */
279 int whiteNPS, blackNPS; /* [HGM] nps: for easily making clocks aware of NPS */
280 VariantClass currentlyInitializedVariant; /* [HGM] variantswitch */
281 int lastIndex = 0; /* [HGM] autoinc: last game/position used in match mode */
282 Boolean connectionAlive;/* [HGM] alive: ICS connection status from probing */
283 int opponentKibitzes;
284 int lastSavedGame; /* [HGM] save: ID of game */
285 char chatPartner[MAX_CHAT][MSG_SIZ]; /* [HGM] chat: list of chatting partners */
286 extern int chatCount;
288 char marker[BOARD_RANKS][BOARD_FILES]; /* [HGM] marks for target squares */
289 char legal[BOARD_RANKS][BOARD_FILES]; /* [HGM] legal target squares */
290 char lastMsg[MSG_SIZ];
291 char lastTalker[MSG_SIZ];
292 ChessSquare pieceSweep = EmptySquare;
293 ChessSquare promoSweep = EmptySquare, defaultPromoChoice;
294 int promoDefaultAltered;
295 int keepInfo = 0; /* [HGM] to protect PGN tags in auto-step game analysis */
296 static int initPing = -1;
297 int border; /* [HGM] width of board rim, needed to size seek graph */
298 char bestMove[MSG_SIZ];
299 int solvingTime, totalTime;
301 /* States for ics_getting_history */
303 #define H_REQUESTED 1
304 #define H_GOT_REQ_HEADER 2
305 #define H_GOT_UNREQ_HEADER 3
306 #define H_GETTING_MOVES 4
307 #define H_GOT_UNWANTED_HEADER 5
309 /* whosays values for GameEnds */
318 /* Maximum number of games in a cmail message */
319 #define CMAIL_MAX_GAMES 20
321 /* Different types of move when calling RegisterMove */
323 #define CMAIL_RESIGN 1
325 #define CMAIL_ACCEPT 3
327 /* Different types of result to remember for each game */
328 #define CMAIL_NOT_RESULT 0
329 #define CMAIL_OLD_RESULT 1
330 #define CMAIL_NEW_RESULT 2
332 /* Telnet protocol constants */
343 safeStrCpy (char *dst, const char *src, size_t count)
346 assert( dst != NULL );
347 assert( src != NULL );
350 for(i=0; i<count; i++) if((dst[i] = src[i]) == NULLCHAR) break;
351 if( i == count && dst[count-1] != NULLCHAR)
353 dst[ count-1 ] = '\0'; // make sure incomplete copy still null-terminated
354 if(appData.debugMode)
355 fprintf(debugFP, "safeStrCpy: copying %s into %s didn't work, not enough space %d\n",src,dst, (int)count);
361 /* Some compiler can't cast u64 to double
362 * This function do the job for us:
364 * We use the highest bit for cast, this only
365 * works if the highest bit is not
366 * in use (This should not happen)
368 * We used this for all compiler
371 u64ToDouble (u64 value)
374 u64 tmp = value & u64Const(0x7fffffffffffffff);
375 r = (double)(s64)tmp;
376 if (value & u64Const(0x8000000000000000))
377 r += 9.2233720368547758080e18; /* 2^63 */
381 /* Fake up flags for now, as we aren't keeping track of castling
382 availability yet. [HGM] Change of logic: the flag now only
383 indicates the type of castlings allowed by the rule of the game.
384 The actual rights themselves are maintained in the array
385 castlingRights, as part of the game history, and are not probed
391 int flags = F_ALL_CASTLE_OK;
392 if ((index % 2) == 0) flags |= F_WHITE_ON_MOVE;
393 switch (gameInfo.variant) {
395 flags &= ~F_ALL_CASTLE_OK;
396 case VariantGiveaway: // [HGM] moved this case label one down: seems Giveaway does have castling on ICC!
397 flags |= F_IGNORE_CHECK;
399 flags |= F_MANDATORY_CAPTURE; //[HGM] losers: sets flag so TestLegality rejects non-capts if capts exist
402 flags |= F_IGNORE_CHECK | F_ATOMIC_CAPTURE;
404 case VariantKriegspiel:
405 flags |= F_KRIEGSPIEL_CAPTURE;
407 case VariantCapaRandom:
408 case VariantFischeRandom:
409 flags |= F_FRC_TYPE_CASTLING; /* [HGM] enable this through flag */
410 case VariantNoCastle:
411 case VariantShatranj:
416 flags &= ~F_ALL_CASTLE_OK;
419 case VariantChuChess:
421 flags |= F_NULL_MOVE;
426 if(appData.fischerCastling) flags |= F_FRC_TYPE_CASTLING, flags &= ~F_ALL_CASTLE_OK; // [HGM] fischer
430 FILE *gameFileFP, *debugFP, *serverFP;
431 char *currentDebugFile; // [HGM] debug split: to remember name
434 [AS] Note: sometimes, the sscanf() function is used to parse the input
435 into a fixed-size buffer. Because of this, we must be prepared to
436 receive strings as long as the size of the input buffer, which is currently
437 set to 4K for Windows and 8K for the rest.
438 So, we must either allocate sufficiently large buffers here, or
439 reduce the size of the input buffer in the input reading part.
442 char cmailMove[CMAIL_MAX_GAMES][MOVE_LEN], cmailMsg[MSG_SIZ];
443 char bookOutput[MSG_SIZ*10], thinkOutput[MSG_SIZ*10], lastHint[MSG_SIZ];
444 char thinkOutput1[MSG_SIZ*10];
445 char promoRestrict[MSG_SIZ];
447 ChessProgramState first, second, pairing;
449 /* premove variables */
452 int premoveFromX = 0;
453 int premoveFromY = 0;
454 int premovePromoChar = 0;
456 Boolean alarmSounded;
457 /* end premove variables */
459 char *ics_prefix = "$";
460 enum ICS_TYPE ics_type = ICS_GENERIC;
462 int currentMove = 0, forwardMostMove = 0, backwardMostMove = 0;
463 int pauseExamForwardMostMove = 0;
464 int nCmailGames = 0, nCmailResults = 0, nCmailMovesRegistered = 0;
465 int cmailMoveRegistered[CMAIL_MAX_GAMES], cmailResult[CMAIL_MAX_GAMES];
466 int cmailMsgLoaded = FALSE, cmailMailedMove = FALSE;
467 int cmailOldMove = -1, firstMove = TRUE, flipView = FALSE;
468 int blackPlaysFirst = FALSE, startedFromSetupPosition = FALSE;
469 int searchTime = 0, pausing = FALSE, pauseExamInvalid = FALSE;
470 int whiteFlag = FALSE, blackFlag = FALSE;
471 int userOfferedDraw = FALSE;
472 int ics_user_moved = 0, ics_gamenum = -1, ics_getting_history = H_FALSE;
473 int matchMode = FALSE, hintRequested = FALSE, bookRequested = FALSE;
474 int cmailMoveType[CMAIL_MAX_GAMES];
475 long ics_clock_paused = 0;
476 ProcRef icsPR = NoProc, cmailPR = NoProc;
477 InputSourceRef telnetISR = NULL, fromUserISR = NULL, cmailISR = NULL;
478 GameMode gameMode = BeginningOfGame;
479 char moveList[MAX_MOVES][MOVE_LEN], parseList[MAX_MOVES][MOVE_LEN * 2];
480 char *commentList[MAX_MOVES], *cmailCommentList[CMAIL_MAX_GAMES];
481 ChessProgramStats_Move pvInfoList[MAX_MOVES]; /* [AS] Info about engine thinking */
482 int hiddenThinkOutputState = 0; /* [AS] */
483 int adjudicateLossThreshold = 0; /* [AS] Automatic adjudication */
484 int adjudicateLossPlies = 6;
485 char white_holding[64], black_holding[64];
486 TimeMark lastNodeCountTime;
487 long lastNodeCount=0;
488 int shiftKey, controlKey; // [HGM] set by mouse handler
490 int have_sent_ICS_logon = 0;
492 int suddenDeath, whiteStartMove, blackStartMove; /* [HGM] for implementation of 'any per time' sessions, as in first part of byoyomi TC */
493 long whiteTimeRemaining, blackTimeRemaining, timeControl, timeIncrement, lastWhite, lastBlack, activePartnerTime;
494 Boolean adjustedClock;
495 long timeControl_2; /* [AS] Allow separate time controls */
496 char *fullTimeControlString = NULL, *nextSession, *whiteTC, *blackTC, activePartner; /* [HGM] secondary TC: merge of MPS, TC and inc */
497 long timeRemaining[2][MAX_MOVES];
498 int matchGame = 0, nextGame = 0, roundNr = 0;
499 Boolean waitingForGame = FALSE, startingEngine = FALSE;
500 TimeMark programStartTime, pauseStart;
501 char ics_handle[MSG_SIZ];
502 int have_set_title = 0;
504 /* animateTraining preserves the state of appData.animate
505 * when Training mode is activated. This allows the
506 * response to be animated when appData.animate == TRUE and
507 * appData.animateDragging == TRUE.
509 Boolean animateTraining;
515 Board boards[MAX_MOVES];
516 /* [HGM] Following 7 needed for accurate legality tests: */
517 signed char castlingRank[BOARD_FILES]; // and corresponding ranks
518 unsigned char initialRights[BOARD_FILES];
519 int nrCastlingRights; // For TwoKings, or to implement castling-unknown status
520 int initialRulePlies, FENrulePlies;
521 FILE *serverMoves = NULL; // next two for broadcasting (/serverMoves option)
523 Boolean shuffleOpenings;
524 int mute; // mute all sounds
526 // [HGM] vari: next 12 to save and restore variations
527 #define MAX_VARIATIONS 10
528 int framePtr = MAX_MOVES-1; // points to free stack entry
530 int savedFirst[MAX_VARIATIONS];
531 int savedLast[MAX_VARIATIONS];
532 int savedFramePtr[MAX_VARIATIONS];
533 char *savedDetails[MAX_VARIATIONS];
534 ChessMove savedResult[MAX_VARIATIONS];
536 void PushTail P((int firstMove, int lastMove));
537 Boolean PopTail P((Boolean annotate));
538 void PushInner P((int firstMove, int lastMove));
539 void PopInner P((Boolean annotate));
540 void CleanupTail P((void));
542 ChessSquare FIDEArray[2][BOARD_FILES] = {
543 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
544 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
545 { BlackRook, BlackKnight, BlackBishop, BlackQueen,
546 BlackKing, BlackBishop, BlackKnight, BlackRook }
549 ChessSquare twoKingsArray[2][BOARD_FILES] = {
550 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
551 WhiteKing, WhiteKing, WhiteKnight, WhiteRook },
552 { BlackRook, BlackKnight, BlackBishop, BlackQueen,
553 BlackKing, BlackKing, BlackKnight, BlackRook }
556 ChessSquare KnightmateArray[2][BOARD_FILES] = {
557 { WhiteRook, WhiteMan, WhiteBishop, WhiteQueen,
558 WhiteUnicorn, WhiteBishop, WhiteMan, WhiteRook },
559 { BlackRook, BlackMan, BlackBishop, BlackQueen,
560 BlackUnicorn, BlackBishop, BlackMan, BlackRook }
563 ChessSquare SpartanArray[2][BOARD_FILES] = {
564 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
565 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
566 { BlackAlfil, BlackMarshall, BlackKing, BlackDragon,
567 BlackDragon, BlackKing, BlackAngel, BlackAlfil }
570 ChessSquare fairyArray[2][BOARD_FILES] = { /* [HGM] Queen side differs from King side */
571 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
572 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
573 { BlackCardinal, BlackAlfil, BlackMarshall, BlackAngel,
574 BlackKing, BlackMarshall, BlackAlfil, BlackCardinal }
577 ChessSquare ShatranjArray[2][BOARD_FILES] = { /* [HGM] (movGen knows about Shatranj Q and P) */
578 { WhiteRook, WhiteKnight, WhiteAlfil, WhiteKing,
579 WhiteFerz, WhiteAlfil, WhiteKnight, WhiteRook },
580 { BlackRook, BlackKnight, BlackAlfil, BlackKing,
581 BlackFerz, BlackAlfil, BlackKnight, BlackRook }
584 ChessSquare makrukArray[2][BOARD_FILES] = { /* [HGM] (movGen knows about Shatranj Q and P) */
585 { WhiteRook, WhiteKnight, WhiteMan, WhiteKing,
586 WhiteFerz, WhiteMan, WhiteKnight, WhiteRook },
587 { BlackRook, BlackKnight, BlackMan, BlackFerz,
588 BlackKing, BlackMan, BlackKnight, BlackRook }
591 ChessSquare aseanArray[2][BOARD_FILES] = { /* [HGM] (movGen knows about Shatranj Q and P) */
592 { WhiteRook, WhiteKnight, WhiteMan, WhiteFerz,
593 WhiteKing, WhiteMan, WhiteKnight, WhiteRook },
594 { BlackRook, BlackKnight, BlackMan, BlackFerz,
595 BlackKing, BlackMan, BlackKnight, BlackRook }
598 ChessSquare lionArray[2][BOARD_FILES] = {
599 { WhiteRook, WhiteLion, WhiteBishop, WhiteQueen,
600 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
601 { BlackRook, BlackLion, BlackBishop, BlackQueen,
602 BlackKing, BlackBishop, BlackKnight, BlackRook }
606 #if (BOARD_FILES>=10)
607 ChessSquare ShogiArray[2][BOARD_FILES] = {
608 { WhiteQueen, WhiteKnight, WhiteFerz, WhiteWazir,
609 WhiteKing, WhiteWazir, WhiteFerz, WhiteKnight, WhiteQueen },
610 { BlackQueen, BlackKnight, BlackFerz, BlackWazir,
611 BlackKing, BlackWazir, BlackFerz, BlackKnight, BlackQueen }
614 ChessSquare XiangqiArray[2][BOARD_FILES] = {
615 { WhiteRook, WhiteKnight, WhiteAlfil, WhiteFerz,
616 WhiteWazir, WhiteFerz, WhiteAlfil, WhiteKnight, WhiteRook },
617 { BlackRook, BlackKnight, BlackAlfil, BlackFerz,
618 BlackWazir, BlackFerz, BlackAlfil, BlackKnight, BlackRook }
621 ChessSquare CapablancaArray[2][BOARD_FILES] = {
622 { WhiteRook, WhiteKnight, WhiteAngel, WhiteBishop, WhiteQueen,
623 WhiteKing, WhiteBishop, WhiteMarshall, WhiteKnight, WhiteRook },
624 { BlackRook, BlackKnight, BlackAngel, BlackBishop, BlackQueen,
625 BlackKing, BlackBishop, BlackMarshall, BlackKnight, BlackRook }
628 ChessSquare GreatArray[2][BOARD_FILES] = {
629 { WhiteDragon, WhiteKnight, WhiteAlfil, WhiteGrasshopper, WhiteKing,
630 WhiteSilver, WhiteCardinal, WhiteAlfil, WhiteKnight, WhiteDragon },
631 { BlackDragon, BlackKnight, BlackAlfil, BlackGrasshopper, BlackKing,
632 BlackSilver, BlackCardinal, BlackAlfil, BlackKnight, BlackDragon },
635 ChessSquare JanusArray[2][BOARD_FILES] = {
636 { WhiteRook, WhiteAngel, WhiteKnight, WhiteBishop, WhiteKing,
637 WhiteQueen, WhiteBishop, WhiteKnight, WhiteAngel, WhiteRook },
638 { BlackRook, BlackAngel, BlackKnight, BlackBishop, BlackKing,
639 BlackQueen, BlackBishop, BlackKnight, BlackAngel, BlackRook }
642 ChessSquare GrandArray[2][BOARD_FILES] = {
643 { EmptySquare, WhiteKnight, WhiteBishop, WhiteQueen, WhiteKing,
644 WhiteMarshall, WhiteAngel, WhiteBishop, WhiteKnight, EmptySquare },
645 { EmptySquare, BlackKnight, BlackBishop, BlackQueen, BlackKing,
646 BlackMarshall, BlackAngel, BlackBishop, BlackKnight, EmptySquare }
649 ChessSquare ChuChessArray[2][BOARD_FILES] = {
650 { WhiteMan, WhiteKnight, WhiteBishop, WhiteCardinal, WhiteLion,
651 WhiteQueen, WhiteDragon, WhiteBishop, WhiteKnight, WhiteMan },
652 { BlackMan, BlackKnight, BlackBishop, BlackDragon, BlackQueen,
653 BlackLion, BlackCardinal, BlackBishop, BlackKnight, BlackMan }
657 ChessSquare GothicArray[2][BOARD_FILES] = {
658 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen, WhiteMarshall,
659 WhiteKing, WhiteAngel, WhiteBishop, WhiteKnight, WhiteRook },
660 { BlackRook, BlackKnight, BlackBishop, BlackQueen, BlackMarshall,
661 BlackKing, BlackAngel, BlackBishop, BlackKnight, BlackRook }
664 #define GothicArray CapablancaArray
668 ChessSquare FalconArray[2][BOARD_FILES] = {
669 { WhiteRook, WhiteKnight, WhiteBishop, WhiteFalcon, WhiteQueen,
670 WhiteKing, WhiteFalcon, WhiteBishop, WhiteKnight, WhiteRook },
671 { BlackRook, BlackKnight, BlackBishop, BlackFalcon, BlackQueen,
672 BlackKing, BlackFalcon, BlackBishop, BlackKnight, BlackRook }
675 #define FalconArray CapablancaArray
678 #else // !(BOARD_FILES>=10)
679 #define XiangqiPosition FIDEArray
680 #define CapablancaArray FIDEArray
681 #define GothicArray FIDEArray
682 #define GreatArray FIDEArray
683 #endif // !(BOARD_FILES>=10)
685 #if (BOARD_FILES>=12)
686 ChessSquare CourierArray[2][BOARD_FILES] = {
687 { WhiteRook, WhiteKnight, WhiteAlfil, WhiteBishop, WhiteMan, WhiteKing,
688 WhiteFerz, WhiteWazir, WhiteBishop, WhiteAlfil, WhiteKnight, WhiteRook },
689 { BlackRook, BlackKnight, BlackAlfil, BlackBishop, BlackMan, BlackKing,
690 BlackFerz, BlackWazir, BlackBishop, BlackAlfil, BlackKnight, BlackRook }
692 ChessSquare ChuArray[6][BOARD_FILES] = {
693 { WhiteLance, WhiteUnicorn, WhiteMan, WhiteFerz, WhiteWazir, WhiteKing,
694 WhiteAlfil, WhiteWazir, WhiteFerz, WhiteMan, WhiteUnicorn, WhiteLance },
695 { BlackLance, BlackUnicorn, BlackMan, BlackFerz, BlackWazir, BlackAlfil,
696 BlackKing, BlackWazir, BlackFerz, BlackMan, BlackUnicorn, BlackLance },
697 { WhiteCannon, EmptySquare, WhiteBishop, EmptySquare, WhiteNightrider, WhiteMarshall,
698 WhiteAngel, WhiteNightrider, EmptySquare, WhiteBishop, EmptySquare, WhiteCannon },
699 { BlackCannon, EmptySquare, BlackBishop, EmptySquare, BlackNightrider, BlackAngel,
700 BlackMarshall, BlackNightrider, EmptySquare, BlackBishop, EmptySquare, BlackCannon },
701 { WhiteFalcon, WhiteSilver, WhiteRook, WhiteCardinal, WhiteDragon, WhiteLion,
702 WhiteQueen, WhiteDragon, WhiteCardinal, WhiteRook, WhiteSilver, WhiteFalcon },
703 { BlackFalcon, BlackSilver, BlackRook, BlackCardinal, BlackDragon, BlackQueen,
704 BlackLion, BlackDragon, BlackCardinal, BlackRook, BlackSilver, BlackFalcon }
706 #else // !(BOARD_FILES>=12)
707 #define CourierArray CapablancaArray
708 #define ChuArray CapablancaArray
709 #endif // !(BOARD_FILES>=12)
712 Board initialPosition;
715 /* Convert str to a rating. Checks for special cases of "----",
717 "++++", etc. Also strips ()'s */
719 string_to_rating (char *str)
721 while(*str && !isdigit(*str)) ++str;
723 return 0; /* One of the special "no rating" cases */
731 /* Init programStats */
732 programStats.movelist[0] = 0;
733 programStats.depth = 0;
734 programStats.nr_moves = 0;
735 programStats.moves_left = 0;
736 programStats.nodes = 0;
737 programStats.time = -1; // [HGM] PGNtime: make invalid to recognize engine output
738 programStats.score = 0;
739 programStats.got_only_move = 0;
740 programStats.got_fail = 0;
741 programStats.line_is_book = 0;
746 { // [HGM] moved some code here from InitBackend1 that has to be done after both engines have contributed their settings
747 if (appData.firstPlaysBlack) {
748 first.twoMachinesColor = "black\n";
749 second.twoMachinesColor = "white\n";
751 first.twoMachinesColor = "white\n";
752 second.twoMachinesColor = "black\n";
755 first.other = &second;
756 second.other = &first;
759 if(appData.timeOddsMode) {
760 norm = appData.timeOdds[0];
761 if(norm > appData.timeOdds[1]) norm = appData.timeOdds[1];
763 first.timeOdds = appData.timeOdds[0]/norm;
764 second.timeOdds = appData.timeOdds[1]/norm;
767 if(programVersion) free(programVersion);
768 if (appData.noChessProgram) {
769 programVersion = (char*) malloc(5 + strlen(PACKAGE_STRING));
770 sprintf(programVersion, "%s", PACKAGE_STRING);
772 /* [HGM] tidy: use tidy name, in stead of full pathname (which was probably a bug due to / vs \ ) */
773 programVersion = (char*) malloc(8 + strlen(PACKAGE_STRING) + strlen(first.tidy));
774 sprintf(programVersion, "%s + %s", PACKAGE_STRING, first.tidy);
779 UnloadEngine (ChessProgramState *cps)
781 /* Kill off first chess program */
782 if (cps->isr != NULL)
783 RemoveInputSource(cps->isr);
786 if (cps->pr != NoProc) {
788 DoSleep( appData.delayBeforeQuit );
789 SendToProgram("quit\n", cps);
790 DestroyChildProcess(cps->pr, 4 + cps->useSigterm);
793 if(appData.debugMode) fprintf(debugFP, "Unload %s\n", cps->which);
797 ClearOptions (ChessProgramState *cps)
800 cps->nrOptions = cps->comboCnt = 0;
801 for(i=0; i<MAX_OPTIONS; i++) {
802 cps->option[i].min = cps->option[i].max = cps->option[i].value = 0;
803 cps->option[i].textValue = 0;
807 char *engineNames[] = {
808 /* TRANSLATORS: "first" is the first of possible two chess engines. It is inserted into strings
809 such as "%s engine" / "%s chess program" / "%s machine" - all meaning the same thing */
811 /* TRANSLATORS: "second" is the second of possible two chess engines. It is inserted into strings
812 such as "%s engine" / "%s chess program" / "%s machine" - all meaning the same thing */
817 InitEngine (ChessProgramState *cps, int n)
818 { // [HGM] all engine initialiation put in a function that does one engine
822 cps->which = engineNames[n];
823 cps->maybeThinking = FALSE;
827 cps->sendDrawOffers = 1;
829 cps->program = appData.chessProgram[n];
830 cps->host = appData.host[n];
831 cps->dir = appData.directory[n];
832 cps->initString = appData.engInitString[n];
833 cps->computerString = appData.computerString[n];
834 cps->useSigint = TRUE;
835 cps->useSigterm = TRUE;
836 cps->reuse = appData.reuse[n];
837 cps->nps = appData.NPS[n]; // [HGM] nps: copy nodes per second
838 cps->useSetboard = FALSE;
840 cps->usePing = FALSE;
843 cps->usePlayother = FALSE;
844 cps->useColors = TRUE;
845 cps->useUsermove = FALSE;
846 cps->sendICS = FALSE;
847 cps->sendName = appData.icsActive;
848 cps->sdKludge = FALSE;
849 cps->stKludge = FALSE;
850 if(cps->tidy == NULL) cps->tidy = (char*) malloc(MSG_SIZ);
851 TidyProgramName(cps->program, cps->host, cps->tidy);
853 ASSIGN(cps->variants, appData.noChessProgram ? "" : appData.variant);
854 cps->analysisSupport = 2; /* detect */
855 cps->analyzing = FALSE;
856 cps->initDone = FALSE;
858 cps->pseudo = appData.pseudo[n];
860 /* New features added by Tord: */
861 cps->useFEN960 = FALSE;
862 cps->useOOCastle = TRUE;
863 /* End of new features added by Tord. */
864 cps->fenOverride = appData.fenOverride[n];
866 /* [HGM] time odds: set factor for each machine */
867 cps->timeOdds = appData.timeOdds[n];
869 /* [HGM] secondary TC: how to handle sessions that do not fit in 'level'*/
870 cps->accumulateTC = appData.accumulateTC[n];
871 cps->maxNrOfSessions = 1;
876 cps->drawDepth = appData.drawDepth[n];
877 cps->supportsNPS = UNKNOWN;
878 cps->memSize = FALSE;
879 cps->maxCores = FALSE;
880 ASSIGN(cps->egtFormats, "");
883 cps->optionSettings = appData.engOptions[n];
885 cps->scoreIsAbsolute = appData.scoreIsAbsolute[n]; /* [AS] */
886 cps->isUCI = appData.isUCI[n]; /* [AS] */
887 cps->hasOwnBookUCI = appData.hasOwnBookUCI[n]; /* [AS] */
890 if (appData.protocolVersion[n] > PROTOVER
891 || appData.protocolVersion[n] < 1)
896 len = snprintf(buf, MSG_SIZ, _("protocol version %d not supported"),
897 appData.protocolVersion[n]);
898 if( (len >= MSG_SIZ) && appData.debugMode )
899 fprintf(debugFP, "InitBackEnd1: buffer truncated.\n");
901 DisplayFatalError(buf, 0, 2);
905 cps->protocolVersion = appData.protocolVersion[n];
908 InitEngineUCI( installDir, cps ); // [HGM] moved here from winboard.c, to make available in xboard
909 ParseFeatures(appData.featureDefaults, cps);
912 ChessProgramState *savCps;
920 if(WaitForEngine(savCps, LoadEngine)) return;
921 CommonEngineInit(); // recalculate time odds
922 if(gameInfo.variant != StringToVariant(appData.variant)) {
923 // we changed variant when loading the engine; this forces us to reset
924 Reset(TRUE, savCps != &first);
925 oldMode = BeginningOfGame; // to prevent restoring old mode
927 InitChessProgram(savCps, FALSE);
928 if(gameMode == EditGame) SendToProgram("force\n", savCps); // in EditGame mode engine must be in force mode
929 DisplayMessage("", "");
930 if (startedFromSetupPosition) SendBoard(savCps, backwardMostMove);
931 for (i = backwardMostMove; i < currentMove; i++) SendMoveToProgram(i, savCps);
934 if(oldMode == AnalyzeMode) AnalyzeModeEvent();
938 ReplaceEngine (ChessProgramState *cps, int n)
940 oldMode = gameMode; // remember mode, so it can be restored after loading sequence is complete
942 if(oldMode != BeginningOfGame) EditGameEvent();
945 appData.noChessProgram = FALSE;
946 appData.clockMode = TRUE;
949 if(n) return; // only startup first engine immediately; second can wait
950 savCps = cps; // parameter to LoadEngine passed as globals, to allow scheduled calling :-(
954 extern char *engineName, *engineDir, *engineChoice, *engineLine, *nickName, *params;
955 extern Boolean isUCI, hasBook, storeVariant, v1, addToList, useNick;
957 static char resetOptions[] =
958 "-reuse -firstIsUCI false -firstHasOwnBookUCI true -firstTimeOdds 1 "
959 "-firstInitString \"" INIT_STRING "\" -firstComputerString \"" COMPUTER_STRING "\" "
960 "-firstFeatures \"\" -firstLogo \"\" -firstAccumulateTC 1 -fd \".\" "
961 "-firstOptions \"\" -firstNPS -1 -fn \"\" -firstScoreAbs false";
964 FloatToFront(char **list, char *engineLine)
966 char buf[MSG_SIZ], tidy[MSG_SIZ], *p = buf, *q, *r = buf;
968 if(appData.recentEngines <= 0) return;
969 TidyProgramName(engineLine, "localhost", tidy+1);
970 tidy[0] = buf[0] = '\n'; strcat(tidy, "\n");
971 strncpy(buf+1, *list, MSG_SIZ-50);
972 if(p = strstr(buf, tidy)) { // tidy name appears in list
973 q = strchr(++p, '\n'); if(q == NULL) return; // malformed, don't touch
974 while(*p++ = *++q); // squeeze out
976 strcat(tidy, buf+1); // put list behind tidy name
977 p = tidy + 1; while(q = strchr(p, '\n')) i++, r = p, p = q + 1; // count entries in new list
978 if(i > appData.recentEngines) *r = NULLCHAR; // if maximum rached, strip off last
979 ASSIGN(*list, tidy+1);
982 char *insert, *wbOptions; // point in ChessProgramNames were we should insert new engine
985 Load (ChessProgramState *cps, int i)
987 char *p, *q, buf[MSG_SIZ], command[MSG_SIZ], buf2[MSG_SIZ], buf3[MSG_SIZ], jar;
988 if(engineLine && engineLine[0]) { // an engine was selected from the combo box
989 snprintf(buf, MSG_SIZ, "-fcp %s", engineLine);
990 SwapEngines(i); // kludge to parse -f* / -first* like it is -s* / -second*
991 ParseArgsFromString(resetOptions); appData.pvSAN[0] = FALSE;
992 FREE(appData.fenOverride[0]); appData.fenOverride[0] = NULL;
993 appData.firstProtocolVersion = PROTOVER;
994 ParseArgsFromString(buf);
996 ReplaceEngine(cps, i);
997 FloatToFront(&appData.recentEngineList, engineLine);
1001 while(q = strchr(p, SLASH)) p = q+1;
1002 if(*p== NULLCHAR) { DisplayError(_("You did not specify the engine executable"), 0); return; }
1003 if(engineDir[0] != NULLCHAR) {
1004 ASSIGN(appData.directory[i], engineDir); p = engineName;
1005 } else if(p != engineName) { // derive directory from engine path, when not given
1007 ASSIGN(appData.directory[i], engineName);
1009 if(SLASH == '/' && p - engineName > 1) *(p -= 2) = '.'; // for XBoard use ./exeName as command after split!
1010 } else { ASSIGN(appData.directory[i], "."); }
1011 jar = (strstr(p, ".jar") == p + strlen(p) - 4);
1013 if(strchr(p, ' ') && !strchr(p, '"')) snprintf(buf2, MSG_SIZ, "\"%s\"", p), p = buf2; // quote if it contains spaces
1014 snprintf(command, MSG_SIZ, "%s %s", p, params);
1017 if(jar) { snprintf(buf3, MSG_SIZ, "java -jar %s", p); p = buf3; }
1018 ASSIGN(appData.chessProgram[i], p);
1019 appData.isUCI[i] = isUCI;
1020 appData.protocolVersion[i] = v1 ? 1 : PROTOVER;
1021 appData.hasOwnBookUCI[i] = hasBook;
1022 if(!nickName[0]) useNick = FALSE;
1023 if(useNick) ASSIGN(appData.pgnName[i], nickName);
1027 q = firstChessProgramNames;
1028 if(nickName[0]) snprintf(buf, MSG_SIZ, "\"%s\" -fcp ", nickName); else buf[0] = NULLCHAR;
1029 quote = strchr(p, '"') ? '\'' : '"'; // use single quotes around engine command if it contains double quotes
1030 snprintf(buf+strlen(buf), MSG_SIZ-strlen(buf), "%c%s%c -fd \"%s\"%s%s%s%s%s%s%s%s\n",
1031 quote, p, quote, appData.directory[i],
1032 useNick ? " -fn \"" : "",
1033 useNick ? nickName : "",
1034 useNick ? "\"" : "",
1035 v1 ? " -firstProtocolVersion 1" : "",
1036 hasBook ? "" : " -fNoOwnBookUCI",
1037 isUCI ? (isUCI == TRUE ? " -fUCI" : gameInfo.variant == VariantShogi ? " -fUSI" : " -fUCCI") : "",
1038 storeVariant ? " -variant " : "",
1039 storeVariant ? VariantName(gameInfo.variant) : "");
1040 if(wbOptions && wbOptions[0]) snprintf(buf+strlen(buf)-1, MSG_SIZ-strlen(buf), " %s\n", wbOptions);
1041 firstChessProgramNames = malloc(len = strlen(q) + strlen(buf) + 1);
1042 if(insert != q) insert[-1] = NULLCHAR;
1043 snprintf(firstChessProgramNames, len, "%s\n%s%s", q, buf, insert);
1045 FloatToFront(&appData.recentEngineList, buf);
1047 ReplaceEngine(cps, i);
1053 int matched, min, sec;
1055 * Parse timeControl resource
1057 if (!ParseTimeControl(appData.timeControl, appData.timeIncrement,
1058 appData.movesPerSession)) {
1060 snprintf(buf, sizeof(buf), _("bad timeControl option %s"), appData.timeControl);
1061 DisplayFatalError(buf, 0, 2);
1065 * Parse searchTime resource
1067 if (*appData.searchTime != NULLCHAR) {
1068 matched = sscanf(appData.searchTime, "%d:%d", &min, &sec);
1070 searchTime = min * 60;
1071 } else if (matched == 2) {
1072 searchTime = min * 60 + sec;
1075 snprintf(buf, sizeof(buf), _("bad searchTime option %s"), appData.searchTime);
1076 DisplayFatalError(buf, 0, 2);
1085 ShowThinkingEvent(); // [HGM] thinking: make sure post/nopost state is set according to options
1086 startVariant = StringToVariant(appData.variant); // [HGM] nicks: remember original variant
1088 GetTimeMark(&programStartTime);
1089 srandom((programStartTime.ms + 1000*programStartTime.sec)*0x1001001); // [HGM] book: makes sure random is unpredictabe to msec level
1090 appData.seedBase = random() + (random()<<15);
1091 pauseStart = programStartTime; pauseStart.sec -= 100; // [HGM] matchpause: fake a pause that has long since ended
1093 ClearProgramStats();
1094 programStats.ok_to_send = 1;
1095 programStats.seen_stat = 0;
1098 * Initialize game list
1104 * Internet chess server status
1106 if (appData.icsActive) {
1107 appData.matchMode = FALSE;
1108 appData.matchGames = 0;
1110 appData.noChessProgram = !appData.zippyPlay;
1112 appData.zippyPlay = FALSE;
1113 appData.zippyTalk = FALSE;
1114 appData.noChessProgram = TRUE;
1116 if (*appData.icsHelper != NULLCHAR) {
1117 appData.useTelnet = TRUE;
1118 appData.telnetProgram = appData.icsHelper;
1121 appData.zippyTalk = appData.zippyPlay = FALSE;
1124 /* [AS] Initialize pv info list [HGM] and game state */
1128 for( i=0; i<=framePtr; i++ ) {
1129 pvInfoList[i].depth = -1;
1130 boards[i][EP_STATUS] = EP_NONE;
1131 for( j=0; j<BOARD_FILES-2; j++ ) boards[i][CASTLING][j] = NoRights;
1137 /* [AS] Adjudication threshold */
1138 adjudicateLossThreshold = appData.adjudicateLossThreshold;
1140 InitEngine(&first, 0);
1141 InitEngine(&second, 1);
1144 pairing.which = "pairing"; // pairing engine
1145 pairing.pr = NoProc;
1147 pairing.program = appData.pairingEngine;
1148 pairing.host = "localhost";
1151 if (appData.icsActive) {
1152 appData.clockMode = TRUE; /* changes dynamically in ICS mode */
1153 } else if (appData.noChessProgram) { // [HGM] st: searchTime mode now also is clockMode
1154 appData.clockMode = FALSE;
1155 first.sendTime = second.sendTime = 0;
1159 /* Override some settings from environment variables, for backward
1160 compatibility. Unfortunately it's not feasible to have the env
1161 vars just set defaults, at least in xboard. Ugh.
1163 if (appData.icsActive && (appData.zippyPlay || appData.zippyTalk)) {
1168 if (!appData.icsActive) {
1172 /* Check for variants that are supported only in ICS mode,
1173 or not at all. Some that are accepted here nevertheless
1174 have bugs; see comments below.
1176 VariantClass variant = StringToVariant(appData.variant);
1178 case VariantBughouse: /* need four players and two boards */
1179 case VariantKriegspiel: /* need to hide pieces and move details */
1180 /* case VariantFischeRandom: (Fabien: moved below) */
1181 len = snprintf(buf,MSG_SIZ, _("Variant %s supported only in ICS mode"), appData.variant);
1182 if( (len >= MSG_SIZ) && appData.debugMode )
1183 fprintf(debugFP, "InitBackEnd1: buffer truncated.\n");
1185 DisplayFatalError(buf, 0, 2);
1188 case VariantUnknown:
1189 case VariantLoadable:
1199 len = snprintf(buf, MSG_SIZ, _("Unknown variant name %s"), appData.variant);
1200 if( (len >= MSG_SIZ) && appData.debugMode )
1201 fprintf(debugFP, "InitBackEnd1: buffer truncated.\n");
1203 DisplayFatalError(buf, 0, 2);
1206 case VariantNormal: /* definitely works! */
1207 if(strcmp(appData.variant, "normal") && !appData.noChessProgram) { // [HGM] hope this is an engine-defined variant
1208 safeStrCpy(engineVariant, appData.variant, MSG_SIZ);
1211 case VariantXiangqi: /* [HGM] repetition rules not implemented */
1212 case VariantFairy: /* [HGM] TestLegality definitely off! */
1213 case VariantGothic: /* [HGM] should work */
1214 case VariantCapablanca: /* [HGM] should work */
1215 case VariantCourier: /* [HGM] initial forced moves not implemented */
1216 case VariantShogi: /* [HGM] could still mate with pawn drop */
1217 case VariantChu: /* [HGM] experimental */
1218 case VariantKnightmate: /* [HGM] should work */
1219 case VariantCylinder: /* [HGM] untested */
1220 case VariantFalcon: /* [HGM] untested */
1221 case VariantCrazyhouse: /* holdings not shown, ([HGM] fixed that!)
1222 offboard interposition not understood */
1223 case VariantWildCastle: /* pieces not automatically shuffled */
1224 case VariantNoCastle: /* pieces not automatically shuffled */
1225 case VariantFischeRandom: /* [HGM] works and shuffles pieces */
1226 case VariantLosers: /* should work except for win condition,
1227 and doesn't know captures are mandatory */
1228 case VariantSuicide: /* should work except for win condition,
1229 and doesn't know captures are mandatory */
1230 case VariantGiveaway: /* should work except for win condition,
1231 and doesn't know captures are mandatory */
1232 case VariantTwoKings: /* should work */
1233 case VariantAtomic: /* should work except for win condition */
1234 case Variant3Check: /* should work except for win condition */
1235 case VariantShatranj: /* should work except for all win conditions */
1236 case VariantMakruk: /* should work except for draw countdown */
1237 case VariantASEAN : /* should work except for draw countdown */
1238 case VariantBerolina: /* might work if TestLegality is off */
1239 case VariantCapaRandom: /* should work */
1240 case VariantJanus: /* should work */
1241 case VariantSuper: /* experimental */
1242 case VariantGreat: /* experimental, requires legality testing to be off */
1243 case VariantSChess: /* S-Chess, should work */
1244 case VariantGrand: /* should work */
1245 case VariantSpartan: /* should work */
1246 case VariantLion: /* should work */
1247 case VariantChuChess: /* should work */
1255 NextIntegerFromString (char ** str, long * value)
1260 while( *s == ' ' || *s == '\t' ) {
1266 if( *s >= '0' && *s <= '9' ) {
1267 while( *s >= '0' && *s <= '9' ) {
1268 *value = *value * 10 + (*s - '0');
1281 NextTimeControlFromString (char ** str, long * value)
1284 int result = NextIntegerFromString( str, &temp );
1287 *value = temp * 60; /* Minutes */
1288 if( **str == ':' ) {
1290 result = NextIntegerFromString( str, &temp );
1291 *value += temp; /* Seconds */
1299 NextSessionFromString (char ** str, int *moves, long * tc, long *inc, int *incType)
1300 { /* [HGM] routine added to read '+moves/time' for secondary time control. */
1301 int result = -1, type = 0; long temp, temp2;
1303 if(**str != ':') return -1; // old params remain in force!
1305 if(**str == '*') type = *(*str)++, temp = 0; // sandclock TC
1306 if( NextIntegerFromString( str, &temp ) ) return -1;
1307 if(type) { *moves = 0; *tc = temp * 500; *inc = temp * 1000; *incType = '*'; return 0; }
1310 /* time only: incremental or sudden-death time control */
1311 if(**str == '+') { /* increment follows; read it */
1313 if(**str == '!') type = *(*str)++; // Bronstein TC
1314 if(result = NextIntegerFromString( str, &temp2)) return -1;
1315 *inc = temp2 * 1000;
1316 if(**str == '.') { // read fraction of increment
1317 char *start = ++(*str);
1318 if(result = NextIntegerFromString( str, &temp2)) return -1;
1320 while(start++ < *str) temp2 /= 10;
1324 *moves = 0; *tc = temp * 1000; *incType = type;
1328 (*str)++; /* classical time control */
1329 result = NextIntegerFromString( str, &temp2); // NOTE: already converted to seconds by ParseTimeControl()
1341 GetTimeQuota (int movenr, int lastUsed, char *tcString)
1342 { /* [HGM] get time to add from the multi-session time-control string */
1343 int incType, moves=1; /* kludge to force reading of first session */
1344 long time, increment;
1347 if(!s || !*s) return 0; // empty TC string means we ran out of the last sudden-death version
1349 if(moves) NextSessionFromString(&s, &moves, &time, &increment, &incType);
1350 nextSession = s; suddenDeath = moves == 0 && increment == 0;
1351 if(movenr == -1) return time; /* last move before new session */
1352 if(incType == '*') increment = 0; else // for sandclock, time is added while not thinking
1353 if(incType == '!' && lastUsed < increment) increment = lastUsed;
1354 if(!moves) return increment; /* current session is incremental */
1355 if(movenr >= 0) movenr -= moves; /* we already finished this session */
1356 } while(movenr >= -1); /* try again for next session */
1358 return 0; // no new time quota on this move
1362 ParseTimeControl (char *tc, float ti, int mps)
1366 char buf[MSG_SIZ], buf2[MSG_SIZ], *mytc = tc;
1369 if(ti >= 0 && !strchr(tc, '+') && !strchr(tc, '/') ) mps = 0;
1370 if(!strchr(tc, '+') && !strchr(tc, '/') && sscanf(tc, "%d:%d", &min, &sec) >= 1)
1371 sprintf(mytc=buf2, "%d", 60*min+sec); // convert 'classical' min:sec tc string to seconds
1375 snprintf(buf, MSG_SIZ, ":%d/%s+%g", mps, mytc, ti);
1377 snprintf(buf, MSG_SIZ, ":%s+%g", mytc, ti);
1380 snprintf(buf, MSG_SIZ, ":%d/%s", mps, mytc);
1382 snprintf(buf, MSG_SIZ, ":%s", mytc);
1384 fullTimeControlString = StrSave(buf); // this should now be in PGN format
1386 if( NextTimeControlFromString( &tc, &tc1 ) != 0 ) {
1391 /* Parse second time control */
1394 if( NextTimeControlFromString( &tc, &tc2 ) != 0 ) {
1402 timeControl_2 = tc2 * 1000;
1412 timeControl = tc1 * 1000;
1415 timeIncrement = ti * 1000; /* convert to ms */
1416 movesPerSession = 0;
1419 movesPerSession = mps;
1427 if (appData.debugMode) {
1428 # ifdef __GIT_VERSION
1429 fprintf(debugFP, "Version: %s (%s)\n", programVersion, __GIT_VERSION);
1431 fprintf(debugFP, "Version: %s\n", programVersion);
1434 ASSIGN(currentDebugFile, appData.nameOfDebugFile); // [HGM] debug split: remember initial name in use
1436 set_cont_sequence(appData.wrapContSeq);
1437 if (appData.matchGames > 0) {
1438 appData.matchMode = TRUE;
1439 } else if (appData.matchMode) {
1440 appData.matchGames = 1;
1442 if(appData.matchMode && appData.sameColorGames > 0) /* [HGM] alternate: overrule matchGames */
1443 appData.matchGames = appData.sameColorGames;
1444 if(appData.rewindIndex > 1) { /* [HGM] autoinc: rewind implies auto-increment and overrules given index */
1445 if(appData.loadPositionIndex >= 0) appData.loadPositionIndex = -1;
1446 if(appData.loadGameIndex >= 0) appData.loadGameIndex = -1;
1449 if (appData.noChessProgram || first.protocolVersion == 1) {
1452 /* kludge: allow timeout for initial "feature" commands */
1454 DisplayMessage("", _("Starting chess program"));
1455 ScheduleDelayedEvent(InitBackEnd3, FEATURE_TIMEOUT);
1460 CalculateIndex (int index, int gameNr)
1461 { // [HGM] autoinc: absolute way to determine load index from game number (taking auto-inc and rewind into account)
1463 if(index > 0) return index; // fixed nmber
1464 if(index == 0) return 1;
1465 res = (index == -1 ? gameNr : (gameNr-1)/2 + 1); // autoinc
1466 if(appData.rewindIndex > 0) res = (res-1) % appData.rewindIndex + 1; // rewind
1471 LoadGameOrPosition (int gameNr)
1472 { // [HGM] taken out of MatchEvent and NextMatchGame (to combine it)
1473 if (*appData.loadGameFile != NULLCHAR) {
1474 if (!LoadGameFromFile(appData.loadGameFile,
1475 CalculateIndex(appData.loadGameIndex, gameNr),
1476 appData.loadGameFile, FALSE)) {
1477 DisplayFatalError(_("Bad game file"), 0, 1);
1480 } else if (*appData.loadPositionFile != NULLCHAR) {
1481 if (!LoadPositionFromFile(appData.loadPositionFile,
1482 CalculateIndex(appData.loadPositionIndex, gameNr),
1483 appData.loadPositionFile)) {
1484 DisplayFatalError(_("Bad position file"), 0, 1);
1492 ReserveGame (int gameNr, char resChar)
1494 FILE *tf = fopen(appData.tourneyFile, "r+");
1495 char *p, *q, c, buf[MSG_SIZ];
1496 if(tf == NULL) { nextGame = appData.matchGames + 1; return; } // kludge to terminate match
1497 safeStrCpy(buf, lastMsg, MSG_SIZ);
1498 DisplayMessage(_("Pick new game"), "");
1499 flock(fileno(tf), LOCK_EX); // lock the tourney file while we are messing with it
1500 ParseArgsFromFile(tf);
1501 p = q = appData.results;
1502 if(appData.debugMode) {
1503 char *r = appData.participants;
1504 fprintf(debugFP, "results = '%s'\n", p);
1505 while(*r) fprintf(debugFP, *r >= ' ' ? "%c" : "\\%03o", *r), r++;
1506 fprintf(debugFP, "\n");
1508 while(*q && *q != ' ') q++; // get first un-played game (could be beyond end!)
1510 q = malloc(strlen(p) + 2); // could be arbitrary long, but allow to extend by one!
1511 safeStrCpy(q, p, strlen(p) + 2);
1512 if(gameNr >= 0) q[gameNr] = resChar; // replace '*' with result
1513 if(appData.debugMode) fprintf(debugFP, "pick next game from '%s': %d\n", q, nextGame);
1514 if(nextGame <= appData.matchGames && resChar != ' ' && !abortMatch) { // reserve next game if tourney not yet done
1515 if(q[nextGame] == NULLCHAR) q[nextGame+1] = NULLCHAR; // append one char
1518 fseek(tf, -(strlen(p)+4), SEEK_END);
1520 if(c != '"') // depending on DOS or Unix line endings we can be one off
1521 fseek(tf, -(strlen(p)+2), SEEK_END);
1522 else fseek(tf, -(strlen(p)+3), SEEK_END);
1523 fprintf(tf, "%s\"\n", q); fclose(tf); // update, and flush by closing
1524 DisplayMessage(buf, "");
1525 free(p); appData.results = q;
1526 if(nextGame <= appData.matchGames && resChar != ' ' && !abortMatch &&
1527 (gameNr < 0 || nextGame / appData.defaultMatchGames != gameNr / appData.defaultMatchGames)) {
1528 int round = appData.defaultMatchGames * appData.tourneyType;
1529 if(gameNr < 0 || appData.tourneyType < 1 || // gauntlet engine can always stay loaded as first engine
1530 appData.tourneyType > 1 && nextGame/round != gameNr/round) // in multi-gauntlet change only after round
1531 UnloadEngine(&first); // next game belongs to other pairing;
1532 UnloadEngine(&second); // already unload the engines, so TwoMachinesEvent will load new ones.
1534 if(appData.debugMode) fprintf(debugFP, "Reserved, next=%d, nr=%d\n", nextGame, gameNr);
1538 MatchEvent (int mode)
1539 { // [HGM] moved out of InitBackend3, to make it callable when match starts through menu
1541 if(matchMode) { // already in match mode: switch it off
1543 if(!appData.tourneyFile[0]) appData.matchGames = matchGame; // kludge to let match terminate after next game.
1546 // if(gameMode != BeginningOfGame) {
1547 // DisplayError(_("You can only start a match from the initial position."), 0);
1551 if(mode == 2) appData.matchGames = appData.defaultMatchGames;
1552 /* Set up machine vs. machine match */
1554 NextTourneyGame(-1, &dummy); // sets appData.matchGames if this is tourney, to make sure ReserveGame knows it
1555 if(appData.tourneyFile[0]) {
1557 if(nextGame > appData.matchGames) {
1559 if(strchr(appData.results, '*') == NULL) {
1561 appData.tourneyCycles++;
1562 if(f = WriteTourneyFile(appData.results, NULL)) { // make a tourney file with increased number of cycles
1564 NextTourneyGame(-1, &dummy);
1566 if(nextGame <= appData.matchGames) {
1567 DisplayNote(_("You restarted an already completed tourney.\nOne more cycle will now be added to it.\nGames commence in 10 sec."));
1569 ScheduleDelayedEvent(NextMatchGame, 10000);
1574 snprintf(buf, MSG_SIZ, _("All games in tourney '%s' are already played or playing"), appData.tourneyFile);
1575 DisplayError(buf, 0);
1576 appData.tourneyFile[0] = 0;
1580 if (appData.noChessProgram) { // [HGM] in tourney engines are loaded automatically
1581 DisplayFatalError(_("Can't have a match with no chess programs"),
1586 matchGame = roundNr = 1;
1587 first.matchWins = second.matchWins = 0; // [HGM] match: needed in later matches
1591 char *comboLine = NULL; // [HGM] recent: WinBoard's first-engine combobox line
1594 InitBackEnd3 P((void))
1596 GameMode initialMode;
1600 if(!appData.icsActive && !appData.noChessProgram && !appData.matchMode && // mode involves only first engine
1601 !strcmp(appData.variant, "normal") && // no explicit variant request
1602 appData.NrRanks == -1 && appData.NrFiles == -1 && appData.holdingsSize == -1 && // no size overrides requested
1603 !SupportedVariant(first.variants, VariantNormal, 8, 8, 0, first.protocolVersion, "") && // but 'normal' won't work with engine
1604 !SupportedVariant(first.variants, VariantFischeRandom, 8, 8, 0, first.protocolVersion, "") ) { // nor will Chess960
1605 char c, *q = first.variants, *p = strchr(q, ',');
1606 if(p) *p = NULLCHAR;
1607 if(StringToVariant(q) != VariantUnknown) { // the engine can play a recognized variant, however
1609 if(sscanf(q, "%dx%d+%d_%c", &w, &h, &s, &c) == 4) // get size overrides the engine needs with it (if any)
1610 appData.NrFiles = w, appData.NrRanks = h, appData.holdingsSize = s, q = strchr(q, '_') + 1;
1611 ASSIGN(appData.variant, q); // fake user requested the first variant played by the engine
1612 Reset(TRUE, FALSE); // and re-initialize
1617 InitChessProgram(&first, startedFromSetupPosition);
1619 if(!appData.noChessProgram) { /* [HGM] tidy: redo program version to use name from myname feature */
1620 free(programVersion);
1621 programVersion = (char*) malloc(8 + strlen(PACKAGE_STRING) + strlen(first.tidy));
1622 sprintf(programVersion, "%s + %s", PACKAGE_STRING, first.tidy);
1623 FloatToFront(&appData.recentEngineList, comboLine ? comboLine : appData.firstChessProgram);
1626 if (appData.icsActive) {
1628 /* [DM] Make a console window if needed [HGM] merged ifs */
1634 if (*appData.icsCommPort != NULLCHAR)
1635 len = snprintf(buf, MSG_SIZ, _("Could not open comm port %s"),
1636 appData.icsCommPort);
1638 len = snprintf(buf, MSG_SIZ, _("Could not connect to host %s, port %s"),
1639 appData.icsHost, appData.icsPort);
1641 if( (len >= MSG_SIZ) && appData.debugMode )
1642 fprintf(debugFP, "InitBackEnd3: buffer truncated.\n");
1644 DisplayFatalError(buf, err, 1);
1649 AddInputSource(icsPR, FALSE, read_from_ics, &telnetISR);
1651 AddInputSource(NoProc, FALSE, read_from_player, &fromUserISR);
1652 if(appData.keepAlive) // [HGM] alive: schedule sending of dummy 'date' command
1653 ScheduleDelayedEvent(KeepAlive, appData.keepAlive*60*1000);
1654 } else if (appData.noChessProgram) {
1660 if (*appData.cmailGameName != NULLCHAR) {
1662 OpenLoopback(&cmailPR);
1664 AddInputSource(cmailPR, FALSE, CmailSigHandlerCallBack, &cmailISR);
1668 DisplayMessage("", "");
1669 if (StrCaseCmp(appData.initialMode, "") == 0) {
1670 initialMode = BeginningOfGame;
1671 if(!appData.icsActive && appData.noChessProgram) { // [HGM] could be fall-back
1672 gameMode = MachinePlaysBlack; // "Machine Black" might have been implicitly highlighted
1673 ModeHighlight(); // make sure XBoard knows it is highlighted, so it will un-highlight it
1674 gameMode = BeginningOfGame; // in case BeginningOfGame now means "Edit Position"
1677 } else if (StrCaseCmp(appData.initialMode, "TwoMachines") == 0) {
1678 initialMode = TwoMachinesPlay;
1679 } else if (StrCaseCmp(appData.initialMode, "AnalyzeFile") == 0) {
1680 initialMode = AnalyzeFile;
1681 } else if (StrCaseCmp(appData.initialMode, "Analysis") == 0) {
1682 initialMode = AnalyzeMode;
1683 } else if (StrCaseCmp(appData.initialMode, "MachineWhite") == 0) {
1684 initialMode = MachinePlaysWhite;
1685 } else if (StrCaseCmp(appData.initialMode, "MachineBlack") == 0) {
1686 initialMode = MachinePlaysBlack;
1687 } else if (StrCaseCmp(appData.initialMode, "EditGame") == 0) {
1688 initialMode = EditGame;
1689 } else if (StrCaseCmp(appData.initialMode, "EditPosition") == 0) {
1690 initialMode = EditPosition;
1691 } else if (StrCaseCmp(appData.initialMode, "Training") == 0) {
1692 initialMode = Training;
1694 len = snprintf(buf, MSG_SIZ, _("Unknown initialMode %s"), appData.initialMode);
1695 if( (len >= MSG_SIZ) && appData.debugMode )
1696 fprintf(debugFP, "InitBackEnd3: buffer truncated.\n");
1698 DisplayFatalError(buf, 0, 2);
1702 if (appData.matchMode) {
1703 if(appData.tourneyFile[0]) { // start tourney from command line
1705 if(f = fopen(appData.tourneyFile, "r")) {
1706 ParseArgsFromFile(f); // make sure tourney parmeters re known
1708 appData.clockMode = TRUE;
1710 } else appData.tourneyFile[0] = NULLCHAR; // for now ignore bad tourney file
1713 } else if (*appData.cmailGameName != NULLCHAR) {
1714 /* Set up cmail mode */
1715 ReloadCmailMsgEvent(TRUE);
1717 /* Set up other modes */
1718 if (initialMode == AnalyzeFile) {
1719 if (*appData.loadGameFile == NULLCHAR) {
1720 DisplayFatalError(_("AnalyzeFile mode requires a game file"), 0, 1);
1724 if (*appData.loadGameFile != NULLCHAR) {
1725 (void) LoadGameFromFile(appData.loadGameFile,
1726 appData.loadGameIndex,
1727 appData.loadGameFile, TRUE);
1728 } else if (*appData.loadPositionFile != NULLCHAR) {
1729 (void) LoadPositionFromFile(appData.loadPositionFile,
1730 appData.loadPositionIndex,
1731 appData.loadPositionFile);
1732 /* [HGM] try to make self-starting even after FEN load */
1733 /* to allow automatic setup of fairy variants with wtm */
1734 if(initialMode == BeginningOfGame && !blackPlaysFirst) {
1735 gameMode = BeginningOfGame;
1736 setboardSpoiledMachineBlack = 1;
1738 /* [HGM] loadPos: make that every new game uses the setup */
1739 /* from file as long as we do not switch variant */
1740 if(!blackPlaysFirst) {
1741 startedFromPositionFile = TRUE;
1742 CopyBoard(filePosition, boards[0]);
1743 CopyBoard(initialPosition, boards[0]);
1746 if (initialMode == AnalyzeMode) {
1747 if (appData.noChessProgram) {
1748 DisplayFatalError(_("Analysis mode requires a chess engine"), 0, 2);
1751 if (appData.icsActive) {
1752 DisplayFatalError(_("Analysis mode does not work with ICS mode"),0,2);
1756 } else if (initialMode == AnalyzeFile) {
1757 appData.showThinking = TRUE; // [HGM] thinking: moved out of ShowThinkingEvent
1758 ShowThinkingEvent();
1760 AnalysisPeriodicEvent(1);
1761 } else if (initialMode == MachinePlaysWhite) {
1762 if (appData.noChessProgram) {
1763 DisplayFatalError(_("MachineWhite mode requires a chess engine"),
1767 if (appData.icsActive) {
1768 DisplayFatalError(_("MachineWhite mode does not work with ICS mode"),
1772 MachineWhiteEvent();
1773 } else if (initialMode == MachinePlaysBlack) {
1774 if (appData.noChessProgram) {
1775 DisplayFatalError(_("MachineBlack mode requires a chess engine"),
1779 if (appData.icsActive) {
1780 DisplayFatalError(_("MachineBlack mode does not work with ICS mode"),
1784 MachineBlackEvent();
1785 } else if (initialMode == TwoMachinesPlay) {
1786 if (appData.noChessProgram) {
1787 DisplayFatalError(_("TwoMachines mode requires a chess engine"),
1791 if (appData.icsActive) {
1792 DisplayFatalError(_("TwoMachines mode does not work with ICS mode"),
1797 } else if (initialMode == EditGame) {
1799 } else if (initialMode == EditPosition) {
1800 EditPositionEvent();
1801 } else if (initialMode == Training) {
1802 if (*appData.loadGameFile == NULLCHAR) {
1803 DisplayFatalError(_("Training mode requires a game file"), 0, 2);
1812 HistorySet (char movelist[][2*MOVE_LEN], int first, int last, int current)
1814 DisplayBook(current+1);
1816 MoveHistorySet( movelist, first, last, current, pvInfoList );
1818 EvalGraphSet( first, last, current, pvInfoList );
1820 MakeEngineOutputTitle();
1824 * Establish will establish a contact to a remote host.port.
1825 * Sets icsPR to a ProcRef for a process (or pseudo-process)
1826 * used to talk to the host.
1827 * Returns 0 if okay, error code if not.
1834 if (*appData.icsCommPort != NULLCHAR) {
1835 /* Talk to the host through a serial comm port */
1836 return OpenCommPort(appData.icsCommPort, &icsPR);
1838 } else if (*appData.gateway != NULLCHAR) {
1839 if (*appData.remoteShell == NULLCHAR) {
1840 /* Use the rcmd protocol to run telnet program on a gateway host */
1841 snprintf(buf, sizeof(buf), "%s %s %s",
1842 appData.telnetProgram, appData.icsHost, appData.icsPort);
1843 return OpenRcmd(appData.gateway, appData.remoteUser, buf, &icsPR);
1846 /* Use the rsh program to run telnet program on a gateway host */
1847 if (*appData.remoteUser == NULLCHAR) {
1848 snprintf(buf, sizeof(buf), "%s %s %s %s %s", appData.remoteShell,
1849 appData.gateway, appData.telnetProgram,
1850 appData.icsHost, appData.icsPort);
1852 snprintf(buf, sizeof(buf), "%s %s -l %s %s %s %s",
1853 appData.remoteShell, appData.gateway,
1854 appData.remoteUser, appData.telnetProgram,
1855 appData.icsHost, appData.icsPort);
1857 return StartChildProcess(buf, "", &icsPR);
1860 } else if (appData.useTelnet) {
1861 return OpenTelnet(appData.icsHost, appData.icsPort, &icsPR);
1864 /* TCP socket interface differs somewhat between
1865 Unix and NT; handle details in the front end.
1867 return OpenTCP(appData.icsHost, appData.icsPort, &icsPR);
1872 EscapeExpand (char *p, char *q)
1873 { // [HGM] initstring: routine to shape up string arguments
1874 while(*p++ = *q++) if(p[-1] == '\\')
1876 case 'n': p[-1] = '\n'; break;
1877 case 'r': p[-1] = '\r'; break;
1878 case 't': p[-1] = '\t'; break;
1879 case '\\': p[-1] = '\\'; break;
1880 case 0: *p = 0; return;
1881 default: p[-1] = q[-1]; break;
1886 show_bytes (FILE *fp, char *buf, int count)
1889 if (*buf < 040 || *(unsigned char *) buf > 0177) {
1890 fprintf(fp, "\\%03o", *buf & 0xff);
1899 /* Returns an errno value */
1901 OutputMaybeTelnet (ProcRef pr, char *message, int count, int *outError)
1903 char buf[8192], *p, *q, *buflim;
1904 int left, newcount, outcount;
1906 if (*appData.icsCommPort != NULLCHAR || appData.useTelnet ||
1907 *appData.gateway != NULLCHAR) {
1908 if (appData.debugMode) {
1909 fprintf(debugFP, ">ICS: ");
1910 show_bytes(debugFP, message, count);
1911 fprintf(debugFP, "\n");
1913 return OutputToProcess(pr, message, count, outError);
1916 buflim = &buf[sizeof(buf)-1]; /* allow 1 byte for expanding last char */
1923 if (appData.debugMode) {
1924 fprintf(debugFP, ">ICS: ");
1925 show_bytes(debugFP, buf, newcount);
1926 fprintf(debugFP, "\n");
1928 outcount = OutputToProcess(pr, buf, newcount, outError);
1929 if (outcount < newcount) return -1; /* to be sure */
1936 } else if (((unsigned char) *p) == TN_IAC) {
1937 *q++ = (char) TN_IAC;
1944 if (appData.debugMode) {
1945 fprintf(debugFP, ">ICS: ");
1946 show_bytes(debugFP, buf, newcount);
1947 fprintf(debugFP, "\n");
1949 outcount = OutputToProcess(pr, buf, newcount, outError);
1950 if (outcount < newcount) return -1; /* to be sure */
1955 read_from_player (InputSourceRef isr, VOIDSTAR closure, char *message, int count, int error)
1957 int outError, outCount;
1958 static int gotEof = 0;
1961 /* Pass data read from player on to ICS */
1964 outCount = OutputMaybeTelnet(icsPR, message, count, &outError);
1965 if (outCount < count) {
1966 DisplayFatalError(_("Error writing to ICS"), outError, 1);
1968 if(have_sent_ICS_logon == 2) {
1969 if(ini = fopen(appData.icsLogon, "w")) { // save first two lines (presumably username & password) on init script file
1970 fprintf(ini, "%s", message);
1971 have_sent_ICS_logon = 3;
1973 have_sent_ICS_logon = 1;
1974 } else if(have_sent_ICS_logon == 3) {
1975 fprintf(ini, "%s", message);
1977 have_sent_ICS_logon = 1;
1979 } else if (count < 0) {
1980 RemoveInputSource(isr);
1981 DisplayFatalError(_("Error reading from keyboard"), error, 1);
1982 } else if (gotEof++ > 0) {
1983 RemoveInputSource(isr);
1984 DisplayFatalError(_("Got end of file from keyboard"), 0, 0);
1990 { // [HGM] alive: periodically send dummy (date) command to ICS to prevent time-out
1991 if(!connectionAlive) DisplayFatalError("No response from ICS", 0, 1);
1992 connectionAlive = FALSE; // only sticks if no response to 'date' command.
1993 SendToICS("date\n");
1994 if(appData.keepAlive) ScheduleDelayedEvent(KeepAlive, appData.keepAlive*60*1000);
1997 /* added routine for printf style output to ics */
1999 ics_printf (char *format, ...)
2001 char buffer[MSG_SIZ];
2004 va_start(args, format);
2005 vsnprintf(buffer, sizeof(buffer), format, args);
2006 buffer[sizeof(buffer)-1] = '\0';
2014 int count, outCount, outError;
2016 if (icsPR == NoProc) return;
2019 outCount = OutputMaybeTelnet(icsPR, s, count, &outError);
2020 if (outCount < count) {
2021 DisplayFatalError(_("Error writing to ICS"), outError, 1);
2025 /* This is used for sending logon scripts to the ICS. Sending
2026 without a delay causes problems when using timestamp on ICC
2027 (at least on my machine). */
2029 SendToICSDelayed (char *s, long msdelay)
2031 int count, outCount, outError;
2033 if (icsPR == NoProc) return;
2036 if (appData.debugMode) {
2037 fprintf(debugFP, ">ICS: ");
2038 show_bytes(debugFP, s, count);
2039 fprintf(debugFP, "\n");
2041 outCount = OutputToProcessDelayed(icsPR, s, count, &outError,
2043 if (outCount < count) {
2044 DisplayFatalError(_("Error writing to ICS"), outError, 1);
2049 /* Remove all highlighting escape sequences in s
2050 Also deletes any suffix starting with '('
2053 StripHighlightAndTitle (char *s)
2055 static char retbuf[MSG_SIZ];
2058 while (*s != NULLCHAR) {
2059 while (*s == '\033') {
2060 while (*s != NULLCHAR && !isalpha(*s)) s++;
2061 if (*s != NULLCHAR) s++;
2063 while (*s != NULLCHAR && *s != '\033') {
2064 if (*s == '(' || *s == '[') {
2075 /* Remove all highlighting escape sequences in s */
2077 StripHighlight (char *s)
2079 static char retbuf[MSG_SIZ];
2082 while (*s != NULLCHAR) {
2083 while (*s == '\033') {
2084 while (*s != NULLCHAR && !isalpha(*s)) s++;
2085 if (*s != NULLCHAR) s++;
2087 while (*s != NULLCHAR && *s != '\033') {
2095 char engineVariant[MSG_SIZ];
2096 char *variantNames[] = VARIANT_NAMES;
2098 VariantName (VariantClass v)
2100 if(v == VariantUnknown || *engineVariant) return engineVariant;
2101 return variantNames[v];
2105 /* Identify a variant from the strings the chess servers use or the
2106 PGN Variant tag names we use. */
2108 StringToVariant (char *e)
2112 VariantClass v = VariantNormal;
2113 int i, found = FALSE;
2114 char buf[MSG_SIZ], c;
2119 /* [HGM] skip over optional board-size prefixes */
2120 if( sscanf(e, "%dx%d_%c", &i, &i, &c) == 3 ||
2121 sscanf(e, "%dx%d+%d_%c", &i, &i, &i, &c) == 4 ) {
2122 while( *e++ != '_');
2125 if(StrCaseStr(e, "misc/")) { // [HGM] on FICS, misc/shogi is not shogi
2129 for (i=0; i<sizeof(variantNames)/sizeof(char*); i++) {
2130 if (p = StrCaseStr(e, variantNames[i])) {
2131 if(p && i >= VariantShogi && (p != e && !appData.icsActive || isalpha(p[strlen(variantNames[i])]))) continue;
2132 v = (VariantClass) i;
2139 if ((StrCaseStr(e, "fischer") && StrCaseStr(e, "random"))
2140 || StrCaseStr(e, "wild/fr")
2141 || StrCaseStr(e, "frc") || StrCaseStr(e, "960")) {
2142 v = VariantFischeRandom;
2143 } else if ((i = 4, p = StrCaseStr(e, "wild")) ||
2144 (i = 1, p = StrCaseStr(e, "w"))) {
2146 while (*p && (isspace(*p) || *p == '(' || *p == '/')) p++;
2153 case 0: /* FICS only, actually */
2155 /* Castling legal even if K starts on d-file */
2156 v = VariantWildCastle;
2161 /* Castling illegal even if K & R happen to start in
2162 normal positions. */
2163 v = VariantNoCastle;
2176 /* Castling legal iff K & R start in normal positions */
2182 /* Special wilds for position setup; unclear what to do here */
2183 v = VariantLoadable;
2186 /* Bizarre ICC game */
2187 v = VariantTwoKings;
2190 v = VariantKriegspiel;
2196 v = VariantFischeRandom;
2199 v = VariantCrazyhouse;
2202 v = VariantBughouse;
2208 /* Not quite the same as FICS suicide! */
2209 v = VariantGiveaway;
2215 v = VariantShatranj;
2218 /* Temporary names for future ICC types. The name *will* change in
2219 the next xboard/WinBoard release after ICC defines it. */
2257 v = VariantCapablanca;
2260 v = VariantKnightmate;
2266 v = VariantCylinder;
2272 v = VariantCapaRandom;
2275 v = VariantBerolina;
2287 /* Found "wild" or "w" in the string but no number;
2288 must assume it's normal chess. */
2292 len = snprintf(buf, MSG_SIZ, _("Unknown wild type %d"), wnum);
2293 if( (len >= MSG_SIZ) && appData.debugMode )
2294 fprintf(debugFP, "StringToVariant: buffer truncated.\n");
2296 DisplayError(buf, 0);
2302 if (appData.debugMode) {
2303 fprintf(debugFP, "recognized '%s' (%d) as variant %s\n",
2304 e, wnum, VariantName(v));
2309 static int leftover_start = 0, leftover_len = 0;
2310 char star_match[STAR_MATCH_N][MSG_SIZ];
2312 /* Test whether pattern is present at &buf[*index]; if so, return TRUE,
2313 advance *index beyond it, and set leftover_start to the new value of
2314 *index; else return FALSE. If pattern contains the character '*', it
2315 matches any sequence of characters not containing '\r', '\n', or the
2316 character following the '*' (if any), and the matched sequence(s) are
2317 copied into star_match.
2320 looking_at ( char *buf, int *index, char *pattern)
2322 char *bufp = &buf[*index], *patternp = pattern;
2324 char *matchp = star_match[0];
2327 if (*patternp == NULLCHAR) {
2328 *index = leftover_start = bufp - buf;
2332 if (*bufp == NULLCHAR) return FALSE;
2333 if (*patternp == '*') {
2334 if (*bufp == *(patternp + 1)) {
2336 matchp = star_match[++star_count];
2340 } else if (*bufp == '\n' || *bufp == '\r') {
2342 if (*patternp == NULLCHAR)
2347 *matchp++ = *bufp++;
2351 if (*patternp != *bufp) return FALSE;
2358 SendToPlayer (char *data, int length)
2360 int error, outCount;
2361 outCount = OutputToProcess(NoProc, data, length, &error);
2362 if (outCount < length) {
2363 DisplayFatalError(_("Error writing to display"), error, 1);
2368 PackHolding (char packed[], char *holding)
2378 switch (runlength) {
2389 sprintf(q, "%d", runlength);
2401 /* Telnet protocol requests from the front end */
2403 TelnetRequest (unsigned char ddww, unsigned char option)
2405 unsigned char msg[3];
2406 int outCount, outError;
2408 if (*appData.icsCommPort != NULLCHAR || appData.useTelnet) return;
2410 if (appData.debugMode) {
2411 char buf1[8], buf2[8], *ddwwStr, *optionStr;
2427 snprintf(buf1,sizeof(buf1)/sizeof(buf1[0]), "%d", ddww);
2436 snprintf(buf2,sizeof(buf2)/sizeof(buf2[0]), "%d", option);
2439 fprintf(debugFP, ">%s %s ", ddwwStr, optionStr);
2444 outCount = OutputToProcess(icsPR, (char *)msg, 3, &outError);
2446 DisplayFatalError(_("Error writing to ICS"), outError, 1);
2453 if (!appData.icsActive) return;
2454 TelnetRequest(TN_DO, TN_ECHO);
2460 if (!appData.icsActive) return;
2461 TelnetRequest(TN_DONT, TN_ECHO);
2465 CopyHoldings (Board board, char *holdings, ChessSquare lowestPiece)
2467 /* put the holdings sent to us by the server on the board holdings area */
2468 int i, j, holdingsColumn, holdingsStartRow, direction, countsColumn;
2472 if(gameInfo.holdingsWidth < 2) return;
2473 if(gameInfo.variant != VariantBughouse && board[HOLDINGS_SET])
2474 return; // prevent overwriting by pre-board holdings
2476 if( (int)lowestPiece >= BlackPawn ) {
2479 holdingsStartRow = BOARD_HEIGHT-1;
2482 holdingsColumn = BOARD_WIDTH-1;
2483 countsColumn = BOARD_WIDTH-2;
2484 holdingsStartRow = 0;
2488 for(i=0; i<BOARD_HEIGHT; i++) { /* clear holdings */
2489 board[i][holdingsColumn] = EmptySquare;
2490 board[i][countsColumn] = (ChessSquare) 0;
2492 while( (p=*holdings++) != NULLCHAR ) {
2493 piece = CharToPiece( ToUpper(p) );
2494 if(piece == EmptySquare) continue;
2495 /*j = (int) piece - (int) WhitePawn;*/
2496 j = PieceToNumber(piece);
2497 if(j >= gameInfo.holdingsSize) continue; /* ignore pieces that do not fit */
2498 if(j < 0) continue; /* should not happen */
2499 piece = (ChessSquare) ( (int)piece + (int)lowestPiece );
2500 board[holdingsStartRow+j*direction][holdingsColumn] = piece;
2501 board[holdingsStartRow+j*direction][countsColumn]++;
2507 VariantSwitch (Board board, VariantClass newVariant)
2509 int newHoldingsWidth, newWidth = 8, newHeight = 8, i, j;
2510 static Board oldBoard;
2512 startedFromPositionFile = FALSE;
2513 if(gameInfo.variant == newVariant) return;
2515 /* [HGM] This routine is called each time an assignment is made to
2516 * gameInfo.variant during a game, to make sure the board sizes
2517 * are set to match the new variant. If that means adding or deleting
2518 * holdings, we shift the playing board accordingly
2519 * This kludge is needed because in ICS observe mode, we get boards
2520 * of an ongoing game without knowing the variant, and learn about the
2521 * latter only later. This can be because of the move list we requested,
2522 * in which case the game history is refilled from the beginning anyway,
2523 * but also when receiving holdings of a crazyhouse game. In the latter
2524 * case we want to add those holdings to the already received position.
2528 if (appData.debugMode) {
2529 fprintf(debugFP, "Switch board from %s to %s\n",
2530 VariantName(gameInfo.variant), VariantName(newVariant));
2531 setbuf(debugFP, NULL);
2533 shuffleOpenings = 0; /* [HGM] shuffle */
2534 gameInfo.holdingsSize = 5; /* [HGM] prepare holdings */
2538 newWidth = 9; newHeight = 9;
2539 gameInfo.holdingsSize = 7;
2540 case VariantBughouse:
2541 case VariantCrazyhouse:
2542 newHoldingsWidth = 2; break;
2546 newHoldingsWidth = 2;
2547 gameInfo.holdingsSize = 8;
2550 case VariantCapablanca:
2551 case VariantCapaRandom:
2554 newHoldingsWidth = gameInfo.holdingsSize = 0;
2557 if(newWidth != gameInfo.boardWidth ||
2558 newHeight != gameInfo.boardHeight ||
2559 newHoldingsWidth != gameInfo.holdingsWidth ) {
2561 /* shift position to new playing area, if needed */
2562 if(newHoldingsWidth > gameInfo.holdingsWidth) {
2563 for(i=0; i<BOARD_HEIGHT; i++)
2564 for(j=BOARD_RGHT-1; j>=BOARD_LEFT; j--)
2565 board[i][j+newHoldingsWidth-gameInfo.holdingsWidth] =
2567 for(i=0; i<newHeight; i++) {
2568 board[i][0] = board[i][newWidth+2*newHoldingsWidth-1] = EmptySquare;
2569 board[i][1] = board[i][newWidth+2*newHoldingsWidth-2] = (ChessSquare) 0;
2571 } else if(newHoldingsWidth < gameInfo.holdingsWidth) {
2572 for(i=0; i<BOARD_HEIGHT; i++)
2573 for(j=BOARD_LEFT; j<BOARD_RGHT; j++)
2574 board[i][j+newHoldingsWidth-gameInfo.holdingsWidth] =
2577 board[HOLDINGS_SET] = 0;
2578 gameInfo.boardWidth = newWidth;
2579 gameInfo.boardHeight = newHeight;
2580 gameInfo.holdingsWidth = newHoldingsWidth;
2581 gameInfo.variant = newVariant;
2582 InitDrawingSizes(-2, 0);
2583 } else gameInfo.variant = newVariant;
2584 CopyBoard(oldBoard, board); // remember correctly formatted board
2585 InitPosition(FALSE); /* this sets up board[0], but also other stuff */
2586 DrawPosition(TRUE, currentMove ? boards[currentMove] : oldBoard);
2589 static int loggedOn = FALSE;
2591 /*-- Game start info cache: --*/
2593 char gs_kind[MSG_SIZ];
2594 static char player1Name[128] = "";
2595 static char player2Name[128] = "";
2596 static char cont_seq[] = "\n\\ ";
2597 static int player1Rating = -1;
2598 static int player2Rating = -1;
2599 /*----------------------------*/
2601 ColorClass curColor = ColorNormal;
2602 int suppressKibitz = 0;
2605 Boolean soughtPending = FALSE;
2606 Boolean seekGraphUp;
2607 #define MAX_SEEK_ADS 200
2609 char *seekAdList[MAX_SEEK_ADS];
2610 int ratingList[MAX_SEEK_ADS], xList[MAX_SEEK_ADS], yList[MAX_SEEK_ADS], seekNrList[MAX_SEEK_ADS], zList[MAX_SEEK_ADS];
2611 float tcList[MAX_SEEK_ADS];
2612 char colorList[MAX_SEEK_ADS];
2613 int nrOfSeekAds = 0;
2614 int minRating = 1010, maxRating = 2800;
2615 int hMargin = 10, vMargin = 20, h, w;
2616 extern int squareSize, lineGap;
2621 int x, y, color = 0, r = ratingList[i]; float tc = tcList[i];
2622 xList[i] = yList[i] = -100; // outside graph, so cannot be clicked
2623 if(r < minRating+100 && r >=0 ) r = minRating+100;
2624 if(r > maxRating) r = maxRating;
2625 if(tc < 1.f) tc = 1.f;
2626 if(tc > 95.f) tc = 95.f;
2627 x = (w-hMargin-squareSize/8-7)* log(tc)/log(95.) + hMargin;
2628 y = ((double)r - minRating)/(maxRating - minRating)
2629 * (h-vMargin-squareSize/8-1) + vMargin;
2630 if(ratingList[i] < 0) y = vMargin + squareSize/4;
2631 if(strstr(seekAdList[i], " u ")) color = 1;
2632 if(!strstr(seekAdList[i], "lightning") && // for now all wilds same color
2633 !strstr(seekAdList[i], "bullet") &&
2634 !strstr(seekAdList[i], "blitz") &&
2635 !strstr(seekAdList[i], "standard") ) color = 2;
2636 if(strstr(seekAdList[i], "(C) ")) color |= SQUARE; // plot computer seeks as squares
2637 DrawSeekDot(xList[i]=x+3*(color&~SQUARE), yList[i]=h-1-y, colorList[i]=color);
2641 PlotSingleSeekAd (int i)
2647 AddAd (char *handle, char *rating, int base, int inc, char rated, char *type, int nr, Boolean plot)
2649 char buf[MSG_SIZ], *ext = "";
2650 VariantClass v = StringToVariant(type);
2651 if(strstr(type, "wild")) {
2652 ext = type + 4; // append wild number
2653 if(v == VariantFischeRandom) type = "chess960"; else
2654 if(v == VariantLoadable) type = "setup"; else
2655 type = VariantName(v);
2657 snprintf(buf, MSG_SIZ, "%s (%s) %d %d %c %s%s", handle, rating, base, inc, rated, type, ext);
2658 if(nrOfSeekAds < MAX_SEEK_ADS-1) {
2659 if(seekAdList[nrOfSeekAds]) free(seekAdList[nrOfSeekAds]);
2660 ratingList[nrOfSeekAds] = -1; // for if seeker has no rating
2661 sscanf(rating, "%d", &ratingList[nrOfSeekAds]);
2662 tcList[nrOfSeekAds] = base + (2./3.)*inc;
2663 seekNrList[nrOfSeekAds] = nr;
2664 zList[nrOfSeekAds] = 0;
2665 seekAdList[nrOfSeekAds++] = StrSave(buf);
2666 if(plot) PlotSingleSeekAd(nrOfSeekAds-1);
2671 EraseSeekDot (int i)
2673 int x = xList[i], y = yList[i], d=squareSize/4, k;
2674 DrawSeekBackground(x-squareSize/8, y-squareSize/8, x+squareSize/8+1, y+squareSize/8+1);
2675 if(x < hMargin+d) DrawSeekAxis(hMargin, y-squareSize/8, hMargin, y+squareSize/8+1);
2676 // now replot every dot that overlapped
2677 for(k=0; k<nrOfSeekAds; k++) if(k != i) {
2678 int xx = xList[k], yy = yList[k];
2679 if(xx <= x+d && xx > x-d && yy <= y+d && yy > y-d)
2680 DrawSeekDot(xx, yy, colorList[k]);
2685 RemoveSeekAd (int nr)
2688 for(i=0; i<nrOfSeekAds; i++) if(seekNrList[i] == nr) {
2690 if(seekAdList[i]) free(seekAdList[i]);
2691 seekAdList[i] = seekAdList[--nrOfSeekAds];
2692 seekNrList[i] = seekNrList[nrOfSeekAds];
2693 ratingList[i] = ratingList[nrOfSeekAds];
2694 colorList[i] = colorList[nrOfSeekAds];
2695 tcList[i] = tcList[nrOfSeekAds];
2696 xList[i] = xList[nrOfSeekAds];
2697 yList[i] = yList[nrOfSeekAds];
2698 zList[i] = zList[nrOfSeekAds];
2699 seekAdList[nrOfSeekAds] = NULL;
2705 MatchSoughtLine (char *line)
2707 char handle[MSG_SIZ], rating[MSG_SIZ], type[MSG_SIZ];
2708 int nr, base, inc, u=0; char dummy;
2710 if(sscanf(line, "%d %s %s %d %d rated %s", &nr, rating, handle, &base, &inc, type) == 6 ||
2711 sscanf(line, "%d %s %s %s %d %d rated %c", &nr, rating, handle, type, &base, &inc, &dummy) == 7 ||
2713 (sscanf(line, "%d %s %s %d %d unrated %s", &nr, rating, handle, &base, &inc, type) == 6 ||
2714 sscanf(line, "%d %s %s %s %d %d unrated %c", &nr, rating, handle, type, &base, &inc, &dummy) == 7) ) {
2715 // match: compact and save the line
2716 AddAd(handle, rating, base, inc, u ? 'u' : 'r', type, nr, FALSE);
2726 if(!seekGraphUp) return FALSE;
2727 h = BOARD_HEIGHT * (squareSize + lineGap) + lineGap + 2*border;
2728 w = BOARD_WIDTH * (squareSize + lineGap) + lineGap + 2*border;
2730 DrawSeekBackground(0, 0, w, h);
2731 DrawSeekAxis(hMargin, h-1-vMargin, w-5, h-1-vMargin);
2732 DrawSeekAxis(hMargin, h-1-vMargin, hMargin, 5);
2733 for(i=0; i<4000; i+= 100) if(i>=minRating && i<maxRating) {
2734 int yy =((double)i - minRating)/(maxRating - minRating)*(h-vMargin-squareSize/8-1) + vMargin;
2736 DrawSeekAxis(hMargin-5, yy, hMargin+5*(i%500==0), yy); // rating ticks
2739 snprintf(buf, MSG_SIZ, "%d", i);
2740 DrawSeekText(buf, hMargin+squareSize/8+7, yy);
2743 DrawSeekText("unrated", hMargin+squareSize/8+7, h-1-vMargin-squareSize/4);
2744 for(i=1; i<100; i+=(i<10?1:5)) {
2745 int xx = (w-hMargin-squareSize/8-7)* log((double)i)/log(95.) + hMargin;
2746 DrawSeekAxis(xx, h-1-vMargin, xx, h-6-vMargin-3*(i%10==0)); // TC ticks
2747 if(i<=5 || (i>40 ? i%20 : i%10) == 0) {
2749 snprintf(buf, MSG_SIZ, "%d", i);
2750 DrawSeekText(buf, xx-2-3*(i>9), h-1-vMargin/2);
2753 for(i=0; i<nrOfSeekAds; i++) PlotSeekAd(i);
2758 SeekGraphClick (ClickType click, int x, int y, int moving)
2760 static int lastDown = 0, displayed = 0, lastSecond;
2761 if(y < 0) return FALSE;
2762 if(!(appData.seekGraph && appData.icsActive && loggedOn &&
2763 (gameMode == BeginningOfGame || gameMode == IcsIdle))) {
2764 if(!seekGraphUp) return FALSE;
2765 seekGraphUp = FALSE; // seek graph is up when it shouldn't be: take it down
2766 DrawPosition(TRUE, NULL);
2769 if(!seekGraphUp) { // initiate cration of seek graph by requesting seek-ad list
2770 if(click == Release || moving) return FALSE;
2772 soughtPending = TRUE;
2773 SendToICS(ics_prefix);
2774 SendToICS("sought\n"); // should this be "sought all"?
2775 } else { // issue challenge based on clicked ad
2776 int dist = 10000; int i, closest = 0, second = 0;
2777 for(i=0; i<nrOfSeekAds; i++) {
2778 int d = (x-xList[i])*(x-xList[i]) + (y-yList[i])*(y-yList[i]) + zList[i];
2779 if(d < dist) { dist = d; closest = i; }
2780 second += (d - zList[i] < 120); // count in-range ads
2781 if(click == Press && moving != 1 && zList[i]>0) zList[i] *= 0.8; // age priority
2785 second = (second > 1);
2786 if(displayed != closest || second != lastSecond) {
2787 DisplayMessage(second ? "!" : "", seekAdList[closest]);
2788 lastSecond = second; displayed = closest;
2790 if(click == Press) {
2791 if(moving == 2) zList[closest] = 100; // right-click; push to back on press
2794 } // on press 'hit', only show info
2795 if(moving == 2) return TRUE; // ignore right up-clicks on dot
2796 snprintf(buf, MSG_SIZ, "play %d\n", seekNrList[closest]);
2797 SendToICS(ics_prefix);
2799 return TRUE; // let incoming board of started game pop down the graph
2800 } else if(click == Release) { // release 'miss' is ignored
2801 zList[lastDown] = 100; // make future selection of the rejected ad more difficult
2802 if(moving == 2) { // right up-click
2803 nrOfSeekAds = 0; // refresh graph
2804 soughtPending = TRUE;
2805 SendToICS(ics_prefix);
2806 SendToICS("sought\n"); // should this be "sought all"?
2809 } else if(moving) { if(displayed >= 0) DisplayMessage("", ""); displayed = -1; return TRUE; }
2810 // press miss or release hit 'pop down' seek graph
2811 seekGraphUp = FALSE;
2812 DrawPosition(TRUE, NULL);
2818 read_from_ics (InputSourceRef isr, VOIDSTAR closure, char *data, int count, int error)
2820 #define BUF_SIZE (16*1024) /* overflowed at 8K with "inchannel 1" on FICS? */
2821 #define STARTED_NONE 0
2822 #define STARTED_MOVES 1
2823 #define STARTED_BOARD 2
2824 #define STARTED_OBSERVE 3
2825 #define STARTED_HOLDINGS 4
2826 #define STARTED_CHATTER 5
2827 #define STARTED_COMMENT 6
2828 #define STARTED_MOVES_NOHIDE 7
2830 static int started = STARTED_NONE;
2831 static char parse[20000];
2832 static int parse_pos = 0;
2833 static char buf[BUF_SIZE + 1];
2834 static int firstTime = TRUE, intfSet = FALSE;
2835 static ColorClass prevColor = ColorNormal;
2836 static int savingComment = FALSE;
2837 static int cmatch = 0; // continuation sequence match
2844 int backup; /* [DM] For zippy color lines */
2846 char talker[MSG_SIZ]; // [HGM] chat
2847 int channel, collective=0;
2849 connectionAlive = TRUE; // [HGM] alive: I think, therefore I am...
2851 if (appData.debugMode) {
2853 fprintf(debugFP, "<ICS: ");
2854 show_bytes(debugFP, data, count);
2855 fprintf(debugFP, "\n");
2859 if (appData.debugMode) { int f = forwardMostMove;
2860 fprintf(debugFP, "ics input %d, castling = %d %d %d %d %d %d\n", f,
2861 boards[f][CASTLING][0],boards[f][CASTLING][1],boards[f][CASTLING][2],
2862 boards[f][CASTLING][3],boards[f][CASTLING][4],boards[f][CASTLING][5]);
2865 /* If last read ended with a partial line that we couldn't parse,
2866 prepend it to the new read and try again. */
2867 if (leftover_len > 0) {
2868 for (i=0; i<leftover_len; i++)
2869 buf[i] = buf[leftover_start + i];
2872 /* copy new characters into the buffer */
2873 bp = buf + leftover_len;
2874 buf_len=leftover_len;
2875 for (i=0; i<count; i++)
2878 if (data[i] == '\r')
2881 // join lines split by ICS?
2882 if (!appData.noJoin)
2885 Joining just consists of finding matches against the
2886 continuation sequence, and discarding that sequence
2887 if found instead of copying it. So, until a match
2888 fails, there's nothing to do since it might be the
2889 complete sequence, and thus, something we don't want
2892 if (data[i] == cont_seq[cmatch])
2895 if (cmatch == strlen(cont_seq))
2897 cmatch = 0; // complete match. just reset the counter
2900 it's possible for the ICS to not include the space
2901 at the end of the last word, making our [correct]
2902 join operation fuse two separate words. the server
2903 does this when the space occurs at the width setting.
2905 if (!buf_len || buf[buf_len-1] != ' ')
2916 match failed, so we have to copy what matched before
2917 falling through and copying this character. In reality,
2918 this will only ever be just the newline character, but
2919 it doesn't hurt to be precise.
2921 strncpy(bp, cont_seq, cmatch);
2933 buf[buf_len] = NULLCHAR;
2934 // next_out = leftover_len; // [HGM] should we set this to 0, and not print it in advance?
2939 while (i < buf_len) {
2940 /* Deal with part of the TELNET option negotiation
2941 protocol. We refuse to do anything beyond the
2942 defaults, except that we allow the WILL ECHO option,
2943 which ICS uses to turn off password echoing when we are
2944 directly connected to it. We reject this option
2945 if localLineEditing mode is on (always on in xboard)
2946 and we are talking to port 23, which might be a real
2947 telnet server that will try to keep WILL ECHO on permanently.
2949 if (buf_len - i >= 3 && (unsigned char) buf[i] == TN_IAC) {
2950 static int remoteEchoOption = FALSE; /* telnet ECHO option */
2951 unsigned char option;
2953 switch ((unsigned char) buf[++i]) {
2955 if (appData.debugMode)
2956 fprintf(debugFP, "\n<WILL ");
2957 switch (option = (unsigned char) buf[++i]) {
2959 if (appData.debugMode)
2960 fprintf(debugFP, "ECHO ");
2961 /* Reply only if this is a change, according
2962 to the protocol rules. */
2963 if (remoteEchoOption) break;
2964 if (appData.localLineEditing &&
2965 atoi(appData.icsPort) == TN_PORT) {
2966 TelnetRequest(TN_DONT, TN_ECHO);
2969 TelnetRequest(TN_DO, TN_ECHO);
2970 remoteEchoOption = TRUE;
2974 if (appData.debugMode)
2975 fprintf(debugFP, "%d ", option);
2976 /* Whatever this is, we don't want it. */
2977 TelnetRequest(TN_DONT, option);
2982 if (appData.debugMode)
2983 fprintf(debugFP, "\n<WONT ");
2984 switch (option = (unsigned char) buf[++i]) {
2986 if (appData.debugMode)
2987 fprintf(debugFP, "ECHO ");
2988 /* Reply only if this is a change, according
2989 to the protocol rules. */
2990 if (!remoteEchoOption) break;
2992 TelnetRequest(TN_DONT, TN_ECHO);
2993 remoteEchoOption = FALSE;
2996 if (appData.debugMode)
2997 fprintf(debugFP, "%d ", (unsigned char) option);
2998 /* Whatever this is, it must already be turned
2999 off, because we never agree to turn on
3000 anything non-default, so according to the
3001 protocol rules, we don't reply. */
3006 if (appData.debugMode)
3007 fprintf(debugFP, "\n<DO ");
3008 switch (option = (unsigned char) buf[++i]) {
3010 /* Whatever this is, we refuse to do it. */
3011 if (appData.debugMode)
3012 fprintf(debugFP, "%d ", option);
3013 TelnetRequest(TN_WONT, option);
3018 if (appData.debugMode)
3019 fprintf(debugFP, "\n<DONT ");
3020 switch (option = (unsigned char) buf[++i]) {
3022 if (appData.debugMode)
3023 fprintf(debugFP, "%d ", option);
3024 /* Whatever this is, we are already not doing
3025 it, because we never agree to do anything
3026 non-default, so according to the protocol
3027 rules, we don't reply. */
3032 if (appData.debugMode)
3033 fprintf(debugFP, "\n<IAC ");
3034 /* Doubled IAC; pass it through */
3038 if (appData.debugMode)
3039 fprintf(debugFP, "\n<%d ", (unsigned char) buf[i]);
3040 /* Drop all other telnet commands on the floor */
3043 if (oldi > next_out)
3044 SendToPlayer(&buf[next_out], oldi - next_out);
3050 /* OK, this at least will *usually* work */
3051 if (!loggedOn && looking_at(buf, &i, "ics%")) {
3055 if (loggedOn && !intfSet) {
3056 if (ics_type == ICS_ICC) {
3057 snprintf(str, MSG_SIZ,
3058 "/set-quietly interface %s\n/set-quietly style 12\n",
3060 if(appData.seekGraph && appData.autoRefresh) // [HGM] seekgraph
3061 strcat(str, "/set-2 51 1\n/set seek 1\n");
3062 } else if (ics_type == ICS_CHESSNET) {
3063 snprintf(str, MSG_SIZ, "/style 12\n");
3065 safeStrCpy(str, "alias $ @\n$set interface ", sizeof(str)/sizeof(str[0]));
3066 strcat(str, programVersion);
3067 strcat(str, "\n$iset startpos 1\n$iset ms 1\n");
3068 if(appData.seekGraph && appData.autoRefresh) // [HGM] seekgraph
3069 strcat(str, "$iset seekremove 1\n$set seek 1\n");
3071 strcat(str, "$iset nohighlight 1\n");
3073 strcat(str, "$iset lock 1\n$style 12\n");
3076 NotifyFrontendLogin();
3080 if (started == STARTED_COMMENT) {
3081 /* Accumulate characters in comment */
3082 parse[parse_pos++] = buf[i];
3083 if (buf[i] == '\n') {
3084 parse[parse_pos] = NULLCHAR;
3085 if(chattingPartner>=0) {
3087 snprintf(mess, MSG_SIZ, "%s%s", talker, parse);
3088 OutputChatMessage(chattingPartner, mess);
3089 if(collective == 1) { // broadcasted talk also goes to private chatbox of talker
3091 talker[strlen(talker+1)-1] = NULLCHAR; // strip closing delimiter
3092 for(p=0; p<MAX_CHAT; p++) if(!StrCaseCmp(talker+1, chatPartner[p])) {
3093 snprintf(mess, MSG_SIZ, "%s: %s", chatPartner[chattingPartner], parse);
3094 OutputChatMessage(p, mess);
3098 chattingPartner = -1;
3099 if(collective != 3) next_out = i+1; // [HGM] suppress printing in ICS window
3102 if(!suppressKibitz) // [HGM] kibitz
3103 AppendComment(forwardMostMove, StripHighlight(parse), TRUE);
3104 else { // [HGM kibitz: divert memorized engine kibitz to engine-output window
3105 int nrDigit = 0, nrAlph = 0, j;
3106 if(parse_pos > MSG_SIZ - 30) // defuse unreasonably long input
3107 { parse_pos = MSG_SIZ-30; parse[parse_pos - 1] = '\n'; }
3108 parse[parse_pos] = NULLCHAR;
3109 // try to be smart: if it does not look like search info, it should go to
3110 // ICS interaction window after all, not to engine-output window.
3111 for(j=0; j<parse_pos; j++) { // count letters and digits
3112 nrDigit += (parse[j] >= '0' && parse[j] <= '9');
3113 nrAlph += (parse[j] >= 'a' && parse[j] <= 'z');
3114 nrAlph += (parse[j] >= 'A' && parse[j] <= 'Z');
3116 if(nrAlph < 9*nrDigit) { // if more than 10% digit we assume search info
3117 int depth=0; float score;
3118 if(sscanf(parse, "!!! %f/%d", &score, &depth) == 2 && depth>0) {
3119 // [HGM] kibitz: save kibitzed opponent info for PGN and eval graph
3120 pvInfoList[forwardMostMove-1].depth = depth;
3121 pvInfoList[forwardMostMove-1].score = 100*score;
3123 OutputKibitz(suppressKibitz, parse);
3126 if(gameMode == IcsObserving) // restore original ICS messages
3127 /* TRANSLATORS: to 'kibitz' is to send a message to all players and the game observers */
3128 snprintf(tmp, MSG_SIZ, "%s kibitzes: %s", star_match[0], parse);
3130 /* TRANSLATORS: to 'kibitz' is to send a message to all players and the game observers */
3131 snprintf(tmp, MSG_SIZ, _("your opponent kibitzes: %s"), parse);
3132 SendToPlayer(tmp, strlen(tmp));
3134 next_out = i+1; // [HGM] suppress printing in ICS window
3136 started = STARTED_NONE;
3138 /* Don't match patterns against characters in comment */
3143 if (started == STARTED_CHATTER) {
3144 if (buf[i] != '\n') {
3145 /* Don't match patterns against characters in chatter */
3149 started = STARTED_NONE;
3150 if(suppressKibitz) next_out = i+1;
3153 /* Kludge to deal with rcmd protocol */
3154 if (firstTime && looking_at(buf, &i, "\001*")) {
3155 DisplayFatalError(&buf[1], 0, 1);
3161 if (!loggedOn && looking_at(buf, &i, "chessclub.com")) {
3164 if (appData.debugMode)
3165 fprintf(debugFP, "ics_type %d\n", ics_type);
3168 if (!loggedOn && looking_at(buf, &i, "freechess.org")) {
3169 ics_type = ICS_FICS;
3171 if (appData.debugMode)
3172 fprintf(debugFP, "ics_type %d\n", ics_type);
3175 if (!loggedOn && looking_at(buf, &i, "chess.net")) {
3176 ics_type = ICS_CHESSNET;
3178 if (appData.debugMode)
3179 fprintf(debugFP, "ics_type %d\n", ics_type);
3184 (looking_at(buf, &i, "\"*\" is *a registered name") ||
3185 looking_at(buf, &i, "Logging you in as \"*\"") ||
3186 looking_at(buf, &i, "will be \"*\""))) {
3187 safeStrCpy(ics_handle, star_match[0], sizeof(ics_handle)/sizeof(ics_handle[0]));
3191 if (loggedOn && !have_set_title && ics_handle[0] != NULLCHAR) {
3193 snprintf(buf, sizeof(buf), "%s@%s", ics_handle, appData.icsHost);
3194 DisplayIcsInteractionTitle(buf);
3195 have_set_title = TRUE;
3198 /* skip finger notes */
3199 if (started == STARTED_NONE &&
3200 ((buf[i] == ' ' && isdigit(buf[i+1])) ||
3201 (buf[i] == '1' && buf[i+1] == '0')) &&
3202 buf[i+2] == ':' && buf[i+3] == ' ') {
3203 started = STARTED_CHATTER;
3209 // [HGM] seekgraph: recognize sought lines and end-of-sought message
3210 if(appData.seekGraph) {
3211 if(soughtPending && MatchSoughtLine(buf+i)) {
3212 i = strstr(buf+i, "rated") - buf;
3213 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3214 next_out = leftover_start = i;
3215 started = STARTED_CHATTER;
3216 suppressKibitz = TRUE;
3219 if((gameMode == IcsIdle || gameMode == BeginningOfGame)
3220 && looking_at(buf, &i, "* ads displayed")) {
3221 soughtPending = FALSE;
3226 if(appData.autoRefresh) {
3227 if(looking_at(buf, &i, "* (*) seeking * * * * *\"play *\" to respond)\n")) {
3228 int s = (ics_type == ICS_ICC); // ICC format differs
3230 AddAd(star_match[0], star_match[1], atoi(star_match[2+s]), atoi(star_match[3+s]),
3231 star_match[4+s][0], star_match[5-3*s], atoi(star_match[7]), TRUE);
3232 looking_at(buf, &i, "*% "); // eat prompt
3233 if(oldi > 0 && buf[oldi-1] == '\n') oldi--; // suppress preceding LF, if any
3234 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3235 next_out = i; // suppress
3238 if(looking_at(buf, &i, "\nAds removed: *\n") || looking_at(buf, &i, "\031(51 * *\031)")) {
3239 char *p = star_match[0];
3241 if(seekGraphUp) RemoveSeekAd(atoi(p));
3242 while(*p && *p++ != ' '); // next
3244 looking_at(buf, &i, "*% "); // eat prompt
3245 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3252 /* skip formula vars */
3253 if (started == STARTED_NONE &&
3254 buf[i] == 'f' && isdigit(buf[i+1]) && buf[i+2] == ':') {
3255 started = STARTED_CHATTER;
3260 // [HGM] kibitz: try to recognize opponent engine-score kibitzes, to divert them to engine-output window
3261 if (appData.autoKibitz && started == STARTED_NONE &&
3262 !appData.icsEngineAnalyze && // [HGM] [DM] ICS analyze
3263 (gameMode == IcsPlayingWhite || gameMode == IcsPlayingBlack || gameMode == IcsObserving)) {
3264 if((looking_at(buf, &i, "\n* kibitzes: ") || looking_at(buf, &i, "\n* whispers: ") ||
3265 looking_at(buf, &i, "* kibitzes: ") || looking_at(buf, &i, "* whispers: ")) &&
3266 (StrStr(star_match[0], gameInfo.white) == star_match[0] ||
3267 StrStr(star_match[0], gameInfo.black) == star_match[0] )) { // kibitz of self or opponent
3268 suppressKibitz = TRUE;
3269 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3271 if((StrStr(star_match[0], gameInfo.white) == star_match[0]
3272 && (gameMode == IcsPlayingWhite)) ||
3273 (StrStr(star_match[0], gameInfo.black) == star_match[0]
3274 && (gameMode == IcsPlayingBlack)) ) // opponent kibitz
3275 started = STARTED_CHATTER; // own kibitz we simply discard
3277 started = STARTED_COMMENT; // make sure it will be collected in parse[]
3278 parse_pos = 0; parse[0] = NULLCHAR;
3279 savingComment = TRUE;
3280 suppressKibitz = gameMode != IcsObserving ? 2 :
3281 (StrStr(star_match[0], gameInfo.white) == NULL) + 1;
3285 if((looking_at(buf, &i, "\nkibitzed to *\n") || looking_at(buf, &i, "kibitzed to *\n") ||
3286 looking_at(buf, &i, "\n(kibitzed to *\n") || looking_at(buf, &i, "(kibitzed to *\n"))
3287 && atoi(star_match[0])) {
3288 // suppress the acknowledgements of our own autoKibitz
3290 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3291 if(p = strchr(star_match[0], ' ')) p[1] = NULLCHAR; // clip off "players)" on FICS
3292 SendToPlayer(star_match[0], strlen(star_match[0]));
3293 if(looking_at(buf, &i, "*% ")) // eat prompt
3294 suppressKibitz = FALSE;
3298 } // [HGM] kibitz: end of patch
3300 if(looking_at(buf, &i, "* rating adjustment: * --> *\n")) continue;
3302 // [HGM] chat: intercept tells by users for which we have an open chat window
3304 if(started == STARTED_NONE && (looking_at(buf, &i, "* tells you:") || looking_at(buf, &i, "* says:") ||
3305 looking_at(buf, &i, "* whispers:") ||
3306 looking_at(buf, &i, "* kibitzes:") ||
3307 looking_at(buf, &i, "* shouts:") ||
3308 looking_at(buf, &i, "* c-shouts:") ||
3309 looking_at(buf, &i, "--> * ") ||
3310 looking_at(buf, &i, "*(*):") && (sscanf(star_match[1], "%d", &channel),1) ||
3311 looking_at(buf, &i, "*(*)(*):") && (sscanf(star_match[2], "%d", &channel),1) ||
3312 looking_at(buf, &i, "*(*)(*)(*):") && (sscanf(star_match[3], "%d", &channel),1) ||
3313 looking_at(buf, &i, "*(*)(*)(*)(*):") && sscanf(star_match[4], "%d", &channel) == 1 )) {
3315 sscanf(star_match[0], "%[^(]", talker+1); // strip (C) or (U) off ICS handle
3316 chattingPartner = -1; collective = 0;
3318 if(channel >= 0) // channel broadcast; look if there is a chatbox for this channel
3319 for(p=0; p<MAX_CHAT; p++) {
3321 if(chatPartner[p][0] >= '0' && chatPartner[p][0] <= '9' && channel == atoi(chatPartner[p])) {
3322 talker[0] = '['; strcat(talker, "] ");
3323 Colorize((channel == 1 ? ColorChannel1 : ColorChannel), FALSE);
3324 chattingPartner = p; break;
3327 if(buf[i-3] == 'e') // kibitz; look if there is a KIBITZ chatbox
3328 for(p=0; p<MAX_CHAT; p++) {
3330 if(!strcmp("kibitzes", chatPartner[p])) {
3331 talker[0] = '['; strcat(talker, "] ");
3332 chattingPartner = p; break;
3335 if(buf[i-3] == 'r') // whisper; look if there is a WHISPER chatbox
3336 for(p=0; p<MAX_CHAT; p++) {
3338 if(!strcmp("whispers", chatPartner[p])) {
3339 talker[0] = '['; strcat(talker, "] ");
3340 chattingPartner = p; break;
3343 if(buf[i-3] == 't' || buf[oldi+2] == '>') {// shout, c-shout or it; look if there is a 'shouts' chatbox
3344 if(buf[i-8] == '-' && buf[i-3] == 't')
3345 for(p=0; p<MAX_CHAT; p++) { // c-shout; check if dedicatesd c-shout box exists
3347 if(!strcmp("c-shouts", chatPartner[p])) {
3348 talker[0] = '('; strcat(talker, ") "); Colorize(ColorSShout, FALSE);
3349 chattingPartner = p; break;
3352 if(chattingPartner < 0)
3353 for(p=0; p<MAX_CHAT; p++) {
3355 if(!strcmp("shouts", chatPartner[p])) {
3356 if(buf[oldi+2] == '>') { talker[0] = '<'; strcat(talker, "> "); Colorize(ColorShout, FALSE); }
3357 else if(buf[i-8] == '-') { talker[0] = '('; strcat(talker, ") "); Colorize(ColorSShout, FALSE); }
3358 else { talker[0] = '['; strcat(talker, "] "); Colorize(ColorShout, FALSE); }
3359 chattingPartner = p; break;
3363 if(chattingPartner<0) // if not, look if there is a chatbox for this indivdual
3364 for(p=0; p<MAX_CHAT; p++) if(!StrCaseCmp(talker+1, chatPartner[p])) {
3366 Colorize(ColorTell, FALSE);
3367 if(collective) safeStrCpy(talker, "broadcasts: ", MSG_SIZ);
3369 chattingPartner = p; break;
3371 if(chattingPartner<0) i = oldi, safeStrCpy(lastTalker, talker+1, MSG_SIZ); else {
3372 Colorize(curColor, TRUE); // undo the bogus colorations we just made to trigger the souds
3373 started = STARTED_COMMENT;
3374 parse_pos = 0; parse[0] = NULLCHAR;
3375 savingComment = 3 + chattingPartner; // counts as TRUE
3376 if(collective == 3) i = oldi; else {
3377 suppressKibitz = TRUE;
3378 if(oldi > 0 && buf[oldi-1] == '\n') oldi--;
3379 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3383 } // [HGM] chat: end of patch
3386 if (appData.zippyTalk || appData.zippyPlay) {
3387 /* [DM] Backup address for color zippy lines */
3389 if (loggedOn == TRUE)
3390 if (ZippyControl(buf, &backup) || ZippyConverse(buf, &backup) ||
3391 (appData.zippyPlay && ZippyMatch(buf, &backup)));
3393 } // [DM] 'else { ' deleted
3395 /* Regular tells and says */
3396 (tkind = 1, looking_at(buf, &i, "* tells you: ")) ||
3397 looking_at(buf, &i, "* (your partner) tells you: ") ||
3398 looking_at(buf, &i, "* says: ") ||
3399 /* Don't color "message" or "messages" output */
3400 (tkind = 5, looking_at(buf, &i, "*. * (*:*): ")) ||
3401 looking_at(buf, &i, "*. * at *:*: ") ||
3402 looking_at(buf, &i, "--* (*:*): ") ||
3403 /* Message notifications (same color as tells) */
3404 looking_at(buf, &i, "* has left a message ") ||
3405 looking_at(buf, &i, "* just sent you a message:\n") ||
3406 /* Whispers and kibitzes */
3407 (tkind = 2, looking_at(buf, &i, "* whispers: ")) ||
3408 looking_at(buf, &i, "* kibitzes: ") ||
3410 (tkind = 3, looking_at(buf, &i, "*(*: "))) {
3412 if (tkind == 1 && strchr(star_match[0], ':')) {
3413 /* Avoid "tells you:" spoofs in channels */
3416 if (star_match[0][0] == NULLCHAR ||
3417 strchr(star_match[0], ' ') ||
3418 (tkind == 3 && strchr(star_match[1], ' '))) {
3419 /* Reject bogus matches */
3422 if (appData.colorize) {
3423 if (oldi > next_out) {
3424 SendToPlayer(&buf[next_out], oldi - next_out);
3429 Colorize(ColorTell, FALSE);
3430 curColor = ColorTell;
3433 Colorize(ColorKibitz, FALSE);
3434 curColor = ColorKibitz;
3437 p = strrchr(star_match[1], '(');
3444 Colorize(ColorChannel1, FALSE);
3445 curColor = ColorChannel1;
3447 Colorize(ColorChannel, FALSE);
3448 curColor = ColorChannel;
3452 curColor = ColorNormal;
3456 if (started == STARTED_NONE && appData.autoComment &&
3457 (gameMode == IcsObserving ||
3458 gameMode == IcsPlayingWhite ||
3459 gameMode == IcsPlayingBlack)) {
3460 parse_pos = i - oldi;
3461 memcpy(parse, &buf[oldi], parse_pos);
3462 parse[parse_pos] = NULLCHAR;
3463 started = STARTED_COMMENT;
3464 savingComment = TRUE;
3465 } else if(collective != 3) {
3466 started = STARTED_CHATTER;
3467 savingComment = FALSE;
3474 if (looking_at(buf, &i, "* s-shouts: ") ||
3475 looking_at(buf, &i, "* c-shouts: ")) {
3476 if (appData.colorize) {
3477 if (oldi > next_out) {
3478 SendToPlayer(&buf[next_out], oldi - next_out);
3481 Colorize(ColorSShout, FALSE);
3482 curColor = ColorSShout;
3485 started = STARTED_CHATTER;
3489 if (looking_at(buf, &i, "--->")) {
3494 if (looking_at(buf, &i, "* shouts: ") ||
3495 looking_at(buf, &i, "--> ")) {
3496 if (appData.colorize) {
3497 if (oldi > next_out) {
3498 SendToPlayer(&buf[next_out], oldi - next_out);
3501 Colorize(ColorShout, FALSE);
3502 curColor = ColorShout;
3505 started = STARTED_CHATTER;
3509 if (looking_at( buf, &i, "Challenge:")) {
3510 if (appData.colorize) {
3511 if (oldi > next_out) {
3512 SendToPlayer(&buf[next_out], oldi - next_out);
3515 Colorize(ColorChallenge, FALSE);
3516 curColor = ColorChallenge;
3522 if (looking_at(buf, &i, "* offers you") ||
3523 looking_at(buf, &i, "* offers to be") ||
3524 looking_at(buf, &i, "* would like to") ||
3525 looking_at(buf, &i, "* requests to") ||
3526 looking_at(buf, &i, "Your opponent offers") ||
3527 looking_at(buf, &i, "Your opponent requests")) {
3529 if (appData.colorize) {
3530 if (oldi > next_out) {
3531 SendToPlayer(&buf[next_out], oldi - next_out);
3534 Colorize(ColorRequest, FALSE);
3535 curColor = ColorRequest;
3540 if (looking_at(buf, &i, "* (*) seeking")) {
3541 if (appData.colorize) {
3542 if (oldi > next_out) {
3543 SendToPlayer(&buf[next_out], oldi - next_out);
3546 Colorize(ColorSeek, FALSE);
3547 curColor = ColorSeek;
3552 if(i < backup) { i = backup; continue; } // [HGM] for if ZippyControl matches, but the colorie code doesn't
3554 if (looking_at(buf, &i, "\\ ")) {
3555 if (prevColor != ColorNormal) {
3556 if (oldi > next_out) {
3557 SendToPlayer(&buf[next_out], oldi - next_out);
3560 Colorize(prevColor, TRUE);
3561 curColor = prevColor;
3563 if (savingComment) {
3564 parse_pos = i - oldi;
3565 memcpy(parse, &buf[oldi], parse_pos);
3566 parse[parse_pos] = NULLCHAR;
3567 started = STARTED_COMMENT;
3568 if(savingComment >= 3) // [HGM] chat: continuation of line for chat box
3569 chattingPartner = savingComment - 3; // kludge to remember the box
3571 started = STARTED_CHATTER;
3576 if (looking_at(buf, &i, "Black Strength :") ||
3577 looking_at(buf, &i, "<<< style 10 board >>>") ||
3578 looking_at(buf, &i, "<10>") ||
3579 looking_at(buf, &i, "#@#")) {
3580 /* Wrong board style */
3582 SendToICS(ics_prefix);
3583 SendToICS("set style 12\n");
3584 SendToICS(ics_prefix);
3585 SendToICS("refresh\n");
3589 if (looking_at(buf, &i, "login:")) {
3590 if (!have_sent_ICS_logon) {
3592 have_sent_ICS_logon = 1;
3593 else // no init script was found
3594 have_sent_ICS_logon = (appData.autoCreateLogon ? 2 : 1); // flag that we should capture username + password
3595 } else { // we have sent (or created) the InitScript, but apparently the ICS rejected it
3596 have_sent_ICS_logon = (appData.autoCreateLogon ? 2 : 1); // request creation of a new script
3601 if (ics_getting_history != H_GETTING_MOVES /*smpos kludge*/ &&
3602 (looking_at(buf, &i, "\n<12> ") ||
3603 looking_at(buf, &i, "<12> "))) {
3605 if (oldi > next_out) {
3606 SendToPlayer(&buf[next_out], oldi - next_out);
3609 started = STARTED_BOARD;
3614 if ((started == STARTED_NONE && looking_at(buf, &i, "\n<b1> ")) ||
3615 looking_at(buf, &i, "<b1> ")) {
3616 if (oldi > next_out) {
3617 SendToPlayer(&buf[next_out], oldi - next_out);
3620 started = STARTED_HOLDINGS;
3625 if (looking_at(buf, &i, "* *vs. * *--- *")) {
3627 /* Header for a move list -- first line */
3629 switch (ics_getting_history) {
3633 case BeginningOfGame:
3634 /* User typed "moves" or "oldmoves" while we
3635 were idle. Pretend we asked for these
3636 moves and soak them up so user can step
3637 through them and/or save them.
3640 gameMode = IcsObserving;
3643 ics_getting_history = H_GOT_UNREQ_HEADER;
3645 case EditGame: /*?*/
3646 case EditPosition: /*?*/
3647 /* Should above feature work in these modes too? */
3648 /* For now it doesn't */
3649 ics_getting_history = H_GOT_UNWANTED_HEADER;
3652 ics_getting_history = H_GOT_UNWANTED_HEADER;
3657 /* Is this the right one? */
3658 if (gameInfo.white && gameInfo.black &&
3659 strcmp(gameInfo.white, star_match[0]) == 0 &&
3660 strcmp(gameInfo.black, star_match[2]) == 0) {
3662 ics_getting_history = H_GOT_REQ_HEADER;
3665 case H_GOT_REQ_HEADER:
3666 case H_GOT_UNREQ_HEADER:
3667 case H_GOT_UNWANTED_HEADER:
3668 case H_GETTING_MOVES:
3669 /* Should not happen */
3670 DisplayError(_("Error gathering move list: two headers"), 0);
3671 ics_getting_history = H_FALSE;
3675 /* Save player ratings into gameInfo if needed */
3676 if ((ics_getting_history == H_GOT_REQ_HEADER ||
3677 ics_getting_history == H_GOT_UNREQ_HEADER) &&
3678 (gameInfo.whiteRating == -1 ||
3679 gameInfo.blackRating == -1)) {
3681 gameInfo.whiteRating = string_to_rating(star_match[1]);
3682 gameInfo.blackRating = string_to_rating(star_match[3]);
3683 if (appData.debugMode)
3684 fprintf(debugFP, "Ratings from header: W %d, B %d\n",
3685 gameInfo.whiteRating, gameInfo.blackRating);
3690 if (looking_at(buf, &i,
3691 "* * match, initial time: * minute*, increment: * second")) {
3692 /* Header for a move list -- second line */
3693 /* Initial board will follow if this is a wild game */
3694 if (gameInfo.event != NULL) free(gameInfo.event);
3695 snprintf(str, MSG_SIZ, "ICS %s %s match", star_match[0], star_match[1]);
3696 gameInfo.event = StrSave(str);
3697 /* [HGM] we switched variant. Translate boards if needed. */
3698 VariantSwitch(boards[currentMove], StringToVariant(gameInfo.event));
3702 if (looking_at(buf, &i, "Move ")) {
3703 /* Beginning of a move list */
3704 switch (ics_getting_history) {
3706 /* Normally should not happen */
3707 /* Maybe user hit reset while we were parsing */
3710 /* Happens if we are ignoring a move list that is not
3711 * the one we just requested. Common if the user
3712 * tries to observe two games without turning off
3715 case H_GETTING_MOVES:
3716 /* Should not happen */
3717 DisplayError(_("Error gathering move list: nested"), 0);
3718 ics_getting_history = H_FALSE;
3720 case H_GOT_REQ_HEADER:
3721 ics_getting_history = H_GETTING_MOVES;
3722 started = STARTED_MOVES;
3724 if (oldi > next_out) {
3725 SendToPlayer(&buf[next_out], oldi - next_out);
3728 case H_GOT_UNREQ_HEADER:
3729 ics_getting_history = H_GETTING_MOVES;
3730 started = STARTED_MOVES_NOHIDE;
3733 case H_GOT_UNWANTED_HEADER:
3734 ics_getting_history = H_FALSE;
3740 if (looking_at(buf, &i, "% ") ||
3741 ((started == STARTED_MOVES || started == STARTED_MOVES_NOHIDE)
3742 && looking_at(buf, &i, "}*"))) { char *bookHit = NULL; // [HGM] book
3743 if(soughtPending && nrOfSeekAds) { // [HGM] seekgraph: on ICC sought-list has no termination line
3744 soughtPending = FALSE;
3748 if(suppressKibitz) next_out = i;
3749 savingComment = FALSE;
3753 case STARTED_MOVES_NOHIDE:
3754 memcpy(&parse[parse_pos], &buf[oldi], i - oldi);
3755 parse[parse_pos + i - oldi] = NULLCHAR;
3756 ParseGameHistory(parse);
3758 if (appData.zippyPlay && first.initDone) {
3759 FeedMovesToProgram(&first, forwardMostMove);
3760 if (gameMode == IcsPlayingWhite) {
3761 if (WhiteOnMove(forwardMostMove)) {
3762 if (first.sendTime) {
3763 if (first.useColors) {
3764 SendToProgram("black\n", &first);
3766 SendTimeRemaining(&first, TRUE);
3768 if (first.useColors) {
3769 SendToProgram("white\n", &first); // [HGM] book: made sending of "go\n" book dependent
3771 bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE); // [HGM] book: probe book for initial pos
3772 first.maybeThinking = TRUE;
3774 if (first.usePlayother) {
3775 if (first.sendTime) {
3776 SendTimeRemaining(&first, TRUE);
3778 SendToProgram("playother\n", &first);
3784 } else if (gameMode == IcsPlayingBlack) {
3785 if (!WhiteOnMove(forwardMostMove)) {
3786 if (first.sendTime) {
3787 if (first.useColors) {
3788 SendToProgram("white\n", &first);
3790 SendTimeRemaining(&first, FALSE);
3792 if (first.useColors) {
3793 SendToProgram("black\n", &first);
3795 bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE);
3796 first.maybeThinking = TRUE;
3798 if (first.usePlayother) {
3799 if (first.sendTime) {
3800 SendTimeRemaining(&first, FALSE);
3802 SendToProgram("playother\n", &first);
3811 if (gameMode == IcsObserving && ics_gamenum == -1) {
3812 /* Moves came from oldmoves or moves command
3813 while we weren't doing anything else.
3815 currentMove = forwardMostMove;
3816 ClearHighlights();/*!!could figure this out*/
3817 flipView = appData.flipView;
3818 DrawPosition(TRUE, boards[currentMove]);
3819 DisplayBothClocks();
3820 snprintf(str, MSG_SIZ, "%s %s %s",
3821 gameInfo.white, _("vs."), gameInfo.black);
3825 /* Moves were history of an active game */
3826 if (gameInfo.resultDetails != NULL) {
3827 free(gameInfo.resultDetails);
3828 gameInfo.resultDetails = NULL;
3831 HistorySet(parseList, backwardMostMove,
3832 forwardMostMove, currentMove-1);
3833 DisplayMove(currentMove - 1);
3834 if (started == STARTED_MOVES) next_out = i;
3835 started = STARTED_NONE;
3836 ics_getting_history = H_FALSE;
3839 case STARTED_OBSERVE:
3840 started = STARTED_NONE;
3841 SendToICS(ics_prefix);
3842 SendToICS("refresh\n");
3848 if(bookHit) { // [HGM] book: simulate book reply
3849 static char bookMove[MSG_SIZ]; // a bit generous?
3851 programStats.nodes = programStats.depth = programStats.time =
3852 programStats.score = programStats.got_only_move = 0;
3853 sprintf(programStats.movelist, "%s (xbook)", bookHit);
3855 safeStrCpy(bookMove, "move ", sizeof(bookMove)/sizeof(bookMove[0]));
3856 strcat(bookMove, bookHit);
3857 HandleMachineMove(bookMove, &first);
3862 if ((started == STARTED_MOVES || started == STARTED_BOARD ||
3863 started == STARTED_HOLDINGS ||
3864 started == STARTED_MOVES_NOHIDE) && i >= leftover_len) {
3865 /* Accumulate characters in move list or board */
3866 parse[parse_pos++] = buf[i];
3869 /* Start of game messages. Mostly we detect start of game
3870 when the first board image arrives. On some versions
3871 of the ICS, though, we need to do a "refresh" after starting
3872 to observe in order to get the current board right away. */
3873 if (looking_at(buf, &i, "Adding game * to observation list")) {
3874 started = STARTED_OBSERVE;