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 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 StartChessProgram P((ChessProgramState *cps));
196 void SendToProgram P((char *message, ChessProgramState *cps));
197 void SendMoveToProgram P((int moveNum, ChessProgramState *cps));
198 void ReceiveFromProgram P((InputSourceRef isr, VOIDSTAR closure,
199 char *buf, int count, int error));
200 void SendTimeControl P((ChessProgramState *cps,
201 int mps, long tc, int inc, int sd, int st));
202 char *TimeControlTagValue P((void));
203 void Attention P((ChessProgramState *cps));
204 void FeedMovesToProgram P((ChessProgramState *cps, int upto));
205 int ResurrectChessProgram P((void));
206 void DisplayComment P((int moveNumber, char *text));
207 void DisplayMove P((int moveNumber));
209 void ParseGameHistory P((char *game));
210 void ParseBoard12 P((char *string));
211 void KeepAlive P((void));
212 void StartClocks P((void));
213 void SwitchClocks P((int nr));
214 void StopClocks P((void));
215 void ResetClocks P((void));
216 char *PGNDate P((void));
217 void SetGameInfo P((void));
218 int RegisterMove P((void));
219 void MakeRegisteredMove P((void));
220 void TruncateGame P((void));
221 int looking_at P((char *, int *, char *));
222 void CopyPlayerNameIntoFileName P((char **, char *));
223 char *SavePart P((char *));
224 int SaveGameOldStyle P((FILE *));
225 int SaveGamePGN P((FILE *));
226 int CheckFlags P((void));
227 long NextTickLength P((long));
228 void CheckTimeControl P((void));
229 void show_bytes P((FILE *, char *, int));
230 int string_to_rating P((char *str));
231 void ParseFeatures P((char* args, ChessProgramState *cps));
232 void InitBackEnd3 P((void));
233 void FeatureDone P((ChessProgramState* cps, int val));
234 void InitChessProgram P((ChessProgramState *cps, int setup));
235 void OutputKibitz(int window, char *text);
236 int PerpetualChase(int first, int last);
237 int EngineOutputIsUp();
238 void InitDrawingSizes(int x, int y);
239 void NextMatchGame P((void));
240 int NextTourneyGame P((int nr, int *swap));
241 int Pairing P((int nr, int nPlayers, int *w, int *b, int *sync));
242 FILE *WriteTourneyFile P((char *results, FILE *f));
243 void DisplayTwoMachinesTitle P(());
244 static void ExcludeClick P((int index));
245 void ToggleSecond P((void));
246 void PauseEngine P((ChessProgramState *cps));
247 static int NonStandardBoardSize P((VariantClass v, int w, int h, int s));
250 extern void ConsoleCreate();
253 ChessProgramState *WhitePlayer();
254 int VerifyDisplayMode P(());
256 char *GetInfoFromComment( int, char * ); // [HGM] PV time: returns stripped comment
257 void InitEngineUCI( const char * iniDir, ChessProgramState * cps ); // [HGM] moved here from winboard.c
258 char *ProbeBook P((int moveNr, char *book)); // [HGM] book: returns a book move
259 char *SendMoveToBookUser P((int nr, ChessProgramState *cps, int initial)); // [HGM] book
260 void ics_update_width P((int new_width));
261 extern char installDir[MSG_SIZ];
262 VariantClass startVariant; /* [HGM] nicks: initial variant */
265 extern int tinyLayout, smallLayout;
266 ChessProgramStats programStats;
267 char lastPV[2][2*MSG_SIZ]; /* [HGM] pv: last PV in thinking output of each engine */
269 static int exiting = 0; /* [HGM] moved to top */
270 static int setboardSpoiledMachineBlack = 0 /*, errorExitFlag = 0*/;
271 int startedFromPositionFile = FALSE; Board filePosition; /* [HGM] loadPos */
272 Board partnerBoard; /* [HGM] bughouse: for peeking at partner game */
273 int partnerHighlight[2];
274 Boolean partnerBoardValid = 0;
275 char partnerStatus[MSG_SIZ];
277 Boolean originalFlip;
278 Boolean twoBoards = 0;
279 char endingGame = 0; /* [HGM] crash: flag to prevent recursion of GameEnds() */
280 int whiteNPS, blackNPS; /* [HGM] nps: for easily making clocks aware of NPS */
281 VariantClass currentlyInitializedVariant; /* [HGM] variantswitch */
282 int lastIndex = 0; /* [HGM] autoinc: last game/position used in match mode */
283 Boolean connectionAlive;/* [HGM] alive: ICS connection status from probing */
284 int opponentKibitzes;
285 int lastSavedGame; /* [HGM] save: ID of game */
286 char chatPartner[MAX_CHAT][MSG_SIZ]; /* [HGM] chat: list of chatting partners */
287 extern int chatCount;
289 char marker[BOARD_RANKS][BOARD_FILES]; /* [HGM] marks for target squares */
290 char legal[BOARD_RANKS][BOARD_FILES]; /* [HGM] legal target squares */
291 char lastMsg[MSG_SIZ];
292 char lastTalker[MSG_SIZ];
293 ChessSquare pieceSweep = EmptySquare;
294 ChessSquare promoSweep = EmptySquare, defaultPromoChoice;
295 int promoDefaultAltered;
296 int keepInfo = 0; /* [HGM] to protect PGN tags in auto-step game analysis */
297 static int initPing = -1;
299 /* States for ics_getting_history */
301 #define H_REQUESTED 1
302 #define H_GOT_REQ_HEADER 2
303 #define H_GOT_UNREQ_HEADER 3
304 #define H_GETTING_MOVES 4
305 #define H_GOT_UNWANTED_HEADER 5
307 /* whosays values for GameEnds */
316 /* Maximum number of games in a cmail message */
317 #define CMAIL_MAX_GAMES 20
319 /* Different types of move when calling RegisterMove */
321 #define CMAIL_RESIGN 1
323 #define CMAIL_ACCEPT 3
325 /* Different types of result to remember for each game */
326 #define CMAIL_NOT_RESULT 0
327 #define CMAIL_OLD_RESULT 1
328 #define CMAIL_NEW_RESULT 2
330 /* Telnet protocol constants */
341 safeStrCpy (char *dst, const char *src, size_t count)
344 assert( dst != NULL );
345 assert( src != NULL );
348 for(i=0; i<count; i++) if((dst[i] = src[i]) == NULLCHAR) break;
349 if( i == count && dst[count-1] != NULLCHAR)
351 dst[ count-1 ] = '\0'; // make sure incomplete copy still null-terminated
352 if(appData.debugMode)
353 fprintf(debugFP, "safeStrCpy: copying %s into %s didn't work, not enough space %d\n",src,dst, (int)count);
359 /* Some compiler can't cast u64 to double
360 * This function do the job for us:
362 * We use the highest bit for cast, this only
363 * works if the highest bit is not
364 * in use (This should not happen)
366 * We used this for all compiler
369 u64ToDouble (u64 value)
372 u64 tmp = value & u64Const(0x7fffffffffffffff);
373 r = (double)(s64)tmp;
374 if (value & u64Const(0x8000000000000000))
375 r += 9.2233720368547758080e18; /* 2^63 */
379 /* Fake up flags for now, as we aren't keeping track of castling
380 availability yet. [HGM] Change of logic: the flag now only
381 indicates the type of castlings allowed by the rule of the game.
382 The actual rights themselves are maintained in the array
383 castlingRights, as part of the game history, and are not probed
389 int flags = F_ALL_CASTLE_OK;
390 if ((index % 2) == 0) flags |= F_WHITE_ON_MOVE;
391 switch (gameInfo.variant) {
393 flags &= ~F_ALL_CASTLE_OK;
394 case VariantGiveaway: // [HGM] moved this case label one down: seems Giveaway does have castling on ICC!
395 flags |= F_IGNORE_CHECK;
397 flags |= F_MANDATORY_CAPTURE; //[HGM] losers: sets flag so TestLegality rejects non-capts if capts exist
400 flags |= F_IGNORE_CHECK | F_ATOMIC_CAPTURE;
402 case VariantKriegspiel:
403 flags |= F_KRIEGSPIEL_CAPTURE;
405 case VariantCapaRandom:
406 case VariantFischeRandom:
407 flags |= F_FRC_TYPE_CASTLING; /* [HGM] enable this through flag */
408 case VariantNoCastle:
409 case VariantShatranj:
414 flags &= ~F_ALL_CASTLE_OK;
417 case VariantChuChess:
419 flags |= F_NULL_MOVE;
424 if(appData.fischerCastling) flags |= F_FRC_TYPE_CASTLING, flags &= ~F_ALL_CASTLE_OK; // [HGM] fischer
428 FILE *gameFileFP, *debugFP, *serverFP;
429 char *currentDebugFile; // [HGM] debug split: to remember name
432 [AS] Note: sometimes, the sscanf() function is used to parse the input
433 into a fixed-size buffer. Because of this, we must be prepared to
434 receive strings as long as the size of the input buffer, which is currently
435 set to 4K for Windows and 8K for the rest.
436 So, we must either allocate sufficiently large buffers here, or
437 reduce the size of the input buffer in the input reading part.
440 char cmailMove[CMAIL_MAX_GAMES][MOVE_LEN], cmailMsg[MSG_SIZ];
441 char bookOutput[MSG_SIZ*10], thinkOutput[MSG_SIZ*10], lastHint[MSG_SIZ];
442 char thinkOutput1[MSG_SIZ*10];
444 ChessProgramState first, second, pairing;
446 /* premove variables */
449 int premoveFromX = 0;
450 int premoveFromY = 0;
451 int premovePromoChar = 0;
453 Boolean alarmSounded;
454 /* end premove variables */
456 char *ics_prefix = "$";
457 enum ICS_TYPE ics_type = ICS_GENERIC;
459 int currentMove = 0, forwardMostMove = 0, backwardMostMove = 0;
460 int pauseExamForwardMostMove = 0;
461 int nCmailGames = 0, nCmailResults = 0, nCmailMovesRegistered = 0;
462 int cmailMoveRegistered[CMAIL_MAX_GAMES], cmailResult[CMAIL_MAX_GAMES];
463 int cmailMsgLoaded = FALSE, cmailMailedMove = FALSE;
464 int cmailOldMove = -1, firstMove = TRUE, flipView = FALSE;
465 int blackPlaysFirst = FALSE, startedFromSetupPosition = FALSE;
466 int searchTime = 0, pausing = FALSE, pauseExamInvalid = FALSE;
467 int whiteFlag = FALSE, blackFlag = FALSE;
468 int userOfferedDraw = FALSE;
469 int ics_user_moved = 0, ics_gamenum = -1, ics_getting_history = H_FALSE;
470 int matchMode = FALSE, hintRequested = FALSE, bookRequested = FALSE;
471 int cmailMoveType[CMAIL_MAX_GAMES];
472 long ics_clock_paused = 0;
473 ProcRef icsPR = NoProc, cmailPR = NoProc;
474 InputSourceRef telnetISR = NULL, fromUserISR = NULL, cmailISR = NULL;
475 GameMode gameMode = BeginningOfGame;
476 char moveList[MAX_MOVES][MOVE_LEN], parseList[MAX_MOVES][MOVE_LEN * 2];
477 char *commentList[MAX_MOVES], *cmailCommentList[CMAIL_MAX_GAMES];
478 ChessProgramStats_Move pvInfoList[MAX_MOVES]; /* [AS] Info about engine thinking */
479 int hiddenThinkOutputState = 0; /* [AS] */
480 int adjudicateLossThreshold = 0; /* [AS] Automatic adjudication */
481 int adjudicateLossPlies = 6;
482 char white_holding[64], black_holding[64];
483 TimeMark lastNodeCountTime;
484 long lastNodeCount=0;
485 int shiftKey, controlKey; // [HGM] set by mouse handler
487 int have_sent_ICS_logon = 0;
489 int suddenDeath, whiteStartMove, blackStartMove; /* [HGM] for implementation of 'any per time' sessions, as in first part of byoyomi TC */
490 long whiteTimeRemaining, blackTimeRemaining, timeControl, timeIncrement, lastWhite, lastBlack, activePartnerTime;
491 Boolean adjustedClock;
492 long timeControl_2; /* [AS] Allow separate time controls */
493 char *fullTimeControlString = NULL, *nextSession, *whiteTC, *blackTC, activePartner; /* [HGM] secondary TC: merge of MPS, TC and inc */
494 long timeRemaining[2][MAX_MOVES];
495 int matchGame = 0, nextGame = 0, roundNr = 0;
496 Boolean waitingForGame = FALSE, startingEngine = FALSE;
497 TimeMark programStartTime, pauseStart;
498 char ics_handle[MSG_SIZ];
499 int have_set_title = 0;
501 /* animateTraining preserves the state of appData.animate
502 * when Training mode is activated. This allows the
503 * response to be animated when appData.animate == TRUE and
504 * appData.animateDragging == TRUE.
506 Boolean animateTraining;
512 Board boards[MAX_MOVES];
513 /* [HGM] Following 7 needed for accurate legality tests: */
514 signed char castlingRank[BOARD_FILES]; // and corresponding ranks
515 signed char initialRights[BOARD_FILES];
516 int nrCastlingRights; // For TwoKings, or to implement castling-unknown status
517 int initialRulePlies, FENrulePlies;
518 FILE *serverMoves = NULL; // next two for broadcasting (/serverMoves option)
520 Boolean shuffleOpenings;
521 int mute; // mute all sounds
523 // [HGM] vari: next 12 to save and restore variations
524 #define MAX_VARIATIONS 10
525 int framePtr = MAX_MOVES-1; // points to free stack entry
527 int savedFirst[MAX_VARIATIONS];
528 int savedLast[MAX_VARIATIONS];
529 int savedFramePtr[MAX_VARIATIONS];
530 char *savedDetails[MAX_VARIATIONS];
531 ChessMove savedResult[MAX_VARIATIONS];
533 void PushTail P((int firstMove, int lastMove));
534 Boolean PopTail P((Boolean annotate));
535 void PushInner P((int firstMove, int lastMove));
536 void PopInner P((Boolean annotate));
537 void CleanupTail P((void));
539 ChessSquare FIDEArray[2][BOARD_FILES] = {
540 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
541 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
542 { BlackRook, BlackKnight, BlackBishop, BlackQueen,
543 BlackKing, BlackBishop, BlackKnight, BlackRook }
546 ChessSquare twoKingsArray[2][BOARD_FILES] = {
547 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
548 WhiteKing, WhiteKing, WhiteKnight, WhiteRook },
549 { BlackRook, BlackKnight, BlackBishop, BlackQueen,
550 BlackKing, BlackKing, BlackKnight, BlackRook }
553 ChessSquare KnightmateArray[2][BOARD_FILES] = {
554 { WhiteRook, WhiteMan, WhiteBishop, WhiteQueen,
555 WhiteUnicorn, WhiteBishop, WhiteMan, WhiteRook },
556 { BlackRook, BlackMan, BlackBishop, BlackQueen,
557 BlackUnicorn, BlackBishop, BlackMan, BlackRook }
560 ChessSquare SpartanArray[2][BOARD_FILES] = {
561 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
562 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
563 { BlackAlfil, BlackMarshall, BlackKing, BlackDragon,
564 BlackDragon, BlackKing, BlackAngel, BlackAlfil }
567 ChessSquare fairyArray[2][BOARD_FILES] = { /* [HGM] Queen side differs from King side */
568 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen,
569 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
570 { BlackCardinal, BlackAlfil, BlackMarshall, BlackAngel,
571 BlackKing, BlackMarshall, BlackAlfil, BlackCardinal }
574 ChessSquare ShatranjArray[2][BOARD_FILES] = { /* [HGM] (movGen knows about Shatranj Q and P) */
575 { WhiteRook, WhiteKnight, WhiteAlfil, WhiteKing,
576 WhiteFerz, WhiteAlfil, WhiteKnight, WhiteRook },
577 { BlackRook, BlackKnight, BlackAlfil, BlackKing,
578 BlackFerz, BlackAlfil, BlackKnight, BlackRook }
581 ChessSquare makrukArray[2][BOARD_FILES] = { /* [HGM] (movGen knows about Shatranj Q and P) */
582 { WhiteRook, WhiteKnight, WhiteMan, WhiteKing,
583 WhiteFerz, WhiteMan, WhiteKnight, WhiteRook },
584 { BlackRook, BlackKnight, BlackMan, BlackFerz,
585 BlackKing, BlackMan, BlackKnight, BlackRook }
588 ChessSquare aseanArray[2][BOARD_FILES] = { /* [HGM] (movGen knows about Shatranj Q and P) */
589 { WhiteRook, WhiteKnight, WhiteMan, WhiteFerz,
590 WhiteKing, WhiteMan, WhiteKnight, WhiteRook },
591 { BlackRook, BlackKnight, BlackMan, BlackFerz,
592 BlackKing, BlackMan, BlackKnight, BlackRook }
595 ChessSquare lionArray[2][BOARD_FILES] = {
596 { WhiteRook, WhiteLion, WhiteBishop, WhiteQueen,
597 WhiteKing, WhiteBishop, WhiteKnight, WhiteRook },
598 { BlackRook, BlackLion, BlackBishop, BlackQueen,
599 BlackKing, BlackBishop, BlackKnight, BlackRook }
603 #if (BOARD_FILES>=10)
604 ChessSquare ShogiArray[2][BOARD_FILES] = {
605 { WhiteQueen, WhiteKnight, WhiteFerz, WhiteWazir,
606 WhiteKing, WhiteWazir, WhiteFerz, WhiteKnight, WhiteQueen },
607 { BlackQueen, BlackKnight, BlackFerz, BlackWazir,
608 BlackKing, BlackWazir, BlackFerz, BlackKnight, BlackQueen }
611 ChessSquare XiangqiArray[2][BOARD_FILES] = {
612 { WhiteRook, WhiteKnight, WhiteAlfil, WhiteFerz,
613 WhiteWazir, WhiteFerz, WhiteAlfil, WhiteKnight, WhiteRook },
614 { BlackRook, BlackKnight, BlackAlfil, BlackFerz,
615 BlackWazir, BlackFerz, BlackAlfil, BlackKnight, BlackRook }
618 ChessSquare CapablancaArray[2][BOARD_FILES] = {
619 { WhiteRook, WhiteKnight, WhiteAngel, WhiteBishop, WhiteQueen,
620 WhiteKing, WhiteBishop, WhiteMarshall, WhiteKnight, WhiteRook },
621 { BlackRook, BlackKnight, BlackAngel, BlackBishop, BlackQueen,
622 BlackKing, BlackBishop, BlackMarshall, BlackKnight, BlackRook }
625 ChessSquare GreatArray[2][BOARD_FILES] = {
626 { WhiteDragon, WhiteKnight, WhiteAlfil, WhiteGrasshopper, WhiteKing,
627 WhiteSilver, WhiteCardinal, WhiteAlfil, WhiteKnight, WhiteDragon },
628 { BlackDragon, BlackKnight, BlackAlfil, BlackGrasshopper, BlackKing,
629 BlackSilver, BlackCardinal, BlackAlfil, BlackKnight, BlackDragon },
632 ChessSquare JanusArray[2][BOARD_FILES] = {
633 { WhiteRook, WhiteAngel, WhiteKnight, WhiteBishop, WhiteKing,
634 WhiteQueen, WhiteBishop, WhiteKnight, WhiteAngel, WhiteRook },
635 { BlackRook, BlackAngel, BlackKnight, BlackBishop, BlackKing,
636 BlackQueen, BlackBishop, BlackKnight, BlackAngel, BlackRook }
639 ChessSquare GrandArray[2][BOARD_FILES] = {
640 { EmptySquare, WhiteKnight, WhiteBishop, WhiteQueen, WhiteKing,
641 WhiteMarshall, WhiteAngel, WhiteBishop, WhiteKnight, EmptySquare },
642 { EmptySquare, BlackKnight, BlackBishop, BlackQueen, BlackKing,
643 BlackMarshall, BlackAngel, BlackBishop, BlackKnight, EmptySquare }
646 ChessSquare ChuChessArray[2][BOARD_FILES] = {
647 { WhiteMan, WhiteKnight, WhiteBishop, WhiteCardinal, WhiteLion,
648 WhiteQueen, WhiteDragon, WhiteBishop, WhiteKnight, WhiteMan },
649 { BlackMan, BlackKnight, BlackBishop, BlackDragon, BlackQueen,
650 BlackLion, BlackCardinal, BlackBishop, BlackKnight, BlackMan }
654 ChessSquare GothicArray[2][BOARD_FILES] = {
655 { WhiteRook, WhiteKnight, WhiteBishop, WhiteQueen, WhiteMarshall,
656 WhiteKing, WhiteAngel, WhiteBishop, WhiteKnight, WhiteRook },
657 { BlackRook, BlackKnight, BlackBishop, BlackQueen, BlackMarshall,
658 BlackKing, BlackAngel, BlackBishop, BlackKnight, BlackRook }
661 #define GothicArray CapablancaArray
665 ChessSquare FalconArray[2][BOARD_FILES] = {
666 { WhiteRook, WhiteKnight, WhiteBishop, WhiteFalcon, WhiteQueen,
667 WhiteKing, WhiteFalcon, WhiteBishop, WhiteKnight, WhiteRook },
668 { BlackRook, BlackKnight, BlackBishop, BlackFalcon, BlackQueen,
669 BlackKing, BlackFalcon, BlackBishop, BlackKnight, BlackRook }
672 #define FalconArray CapablancaArray
675 #else // !(BOARD_FILES>=10)
676 #define XiangqiPosition FIDEArray
677 #define CapablancaArray FIDEArray
678 #define GothicArray FIDEArray
679 #define GreatArray FIDEArray
680 #endif // !(BOARD_FILES>=10)
682 #if (BOARD_FILES>=12)
683 ChessSquare CourierArray[2][BOARD_FILES] = {
684 { WhiteRook, WhiteKnight, WhiteAlfil, WhiteBishop, WhiteMan, WhiteKing,
685 WhiteFerz, WhiteWazir, WhiteBishop, WhiteAlfil, WhiteKnight, WhiteRook },
686 { BlackRook, BlackKnight, BlackAlfil, BlackBishop, BlackMan, BlackKing,
687 BlackFerz, BlackWazir, BlackBishop, BlackAlfil, BlackKnight, BlackRook }
689 ChessSquare ChuArray[6][BOARD_FILES] = {
690 { WhiteLance, WhiteUnicorn, WhiteMan, WhiteFerz, WhiteWazir, WhiteKing,
691 WhiteAlfil, WhiteWazir, WhiteFerz, WhiteMan, WhiteUnicorn, WhiteLance },
692 { BlackLance, BlackUnicorn, BlackMan, BlackFerz, BlackWazir, BlackAlfil,
693 BlackKing, BlackWazir, BlackFerz, BlackMan, BlackUnicorn, BlackLance },
694 { WhiteCannon, EmptySquare, WhiteBishop, EmptySquare, WhiteNightrider, WhiteMarshall,
695 WhiteAngel, WhiteNightrider, EmptySquare, WhiteBishop, EmptySquare, WhiteCannon },
696 { BlackCannon, EmptySquare, BlackBishop, EmptySquare, BlackNightrider, BlackAngel,
697 BlackMarshall, BlackNightrider, EmptySquare, BlackBishop, EmptySquare, BlackCannon },
698 { WhiteFalcon, WhiteSilver, WhiteRook, WhiteCardinal, WhiteDragon, WhiteLion,
699 WhiteQueen, WhiteDragon, WhiteCardinal, WhiteRook, WhiteSilver, WhiteFalcon },
700 { BlackFalcon, BlackSilver, BlackRook, BlackCardinal, BlackDragon, BlackQueen,
701 BlackLion, BlackDragon, BlackCardinal, BlackRook, BlackSilver, BlackFalcon }
703 #else // !(BOARD_FILES>=12)
704 #define CourierArray CapablancaArray
705 #define ChuArray CapablancaArray
706 #endif // !(BOARD_FILES>=12)
709 Board initialPosition;
712 /* Convert str to a rating. Checks for special cases of "----",
714 "++++", etc. Also strips ()'s */
716 string_to_rating (char *str)
718 while(*str && !isdigit(*str)) ++str;
720 return 0; /* One of the special "no rating" cases */
728 /* Init programStats */
729 programStats.movelist[0] = 0;
730 programStats.depth = 0;
731 programStats.nr_moves = 0;
732 programStats.moves_left = 0;
733 programStats.nodes = 0;
734 programStats.time = -1; // [HGM] PGNtime: make invalid to recognize engine output
735 programStats.score = 0;
736 programStats.got_only_move = 0;
737 programStats.got_fail = 0;
738 programStats.line_is_book = 0;
743 { // [HGM] moved some code here from InitBackend1 that has to be done after both engines have contributed their settings
744 if (appData.firstPlaysBlack) {
745 first.twoMachinesColor = "black\n";
746 second.twoMachinesColor = "white\n";
748 first.twoMachinesColor = "white\n";
749 second.twoMachinesColor = "black\n";
752 first.other = &second;
753 second.other = &first;
756 if(appData.timeOddsMode) {
757 norm = appData.timeOdds[0];
758 if(norm > appData.timeOdds[1]) norm = appData.timeOdds[1];
760 first.timeOdds = appData.timeOdds[0]/norm;
761 second.timeOdds = appData.timeOdds[1]/norm;
764 if(programVersion) free(programVersion);
765 if (appData.noChessProgram) {
766 programVersion = (char*) malloc(5 + strlen(PACKAGE_STRING));
767 sprintf(programVersion, "%s", PACKAGE_STRING);
769 /* [HGM] tidy: use tidy name, in stead of full pathname (which was probably a bug due to / vs \ ) */
770 programVersion = (char*) malloc(8 + strlen(PACKAGE_STRING) + strlen(first.tidy));
771 sprintf(programVersion, "%s + %s", PACKAGE_STRING, first.tidy);
776 UnloadEngine (ChessProgramState *cps)
778 /* Kill off first chess program */
779 if (cps->isr != NULL)
780 RemoveInputSource(cps->isr);
783 if (cps->pr != NoProc) {
785 DoSleep( appData.delayBeforeQuit );
786 SendToProgram("quit\n", cps);
787 DestroyChildProcess(cps->pr, 4 + cps->useSigterm);
790 if(appData.debugMode) fprintf(debugFP, "Unload %s\n", cps->which);
794 ClearOptions (ChessProgramState *cps)
797 cps->nrOptions = cps->comboCnt = 0;
798 for(i=0; i<MAX_OPTIONS; i++) {
799 cps->option[i].min = cps->option[i].max = cps->option[i].value = 0;
800 cps->option[i].textValue = 0;
804 char *engineNames[] = {
805 /* TRANSLATORS: "first" is the first of possible two chess engines. It is inserted into strings
806 such as "%s engine" / "%s chess program" / "%s machine" - all meaning the same thing */
808 /* TRANSLATORS: "second" is the second 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 */
814 InitEngine (ChessProgramState *cps, int n)
815 { // [HGM] all engine initialiation put in a function that does one engine
819 cps->which = engineNames[n];
820 cps->maybeThinking = FALSE;
824 cps->sendDrawOffers = 1;
826 cps->program = appData.chessProgram[n];
827 cps->host = appData.host[n];
828 cps->dir = appData.directory[n];
829 cps->initString = appData.engInitString[n];
830 cps->computerString = appData.computerString[n];
831 cps->useSigint = TRUE;
832 cps->useSigterm = TRUE;
833 cps->reuse = appData.reuse[n];
834 cps->nps = appData.NPS[n]; // [HGM] nps: copy nodes per second
835 cps->useSetboard = FALSE;
837 cps->usePing = FALSE;
840 cps->usePlayother = FALSE;
841 cps->useColors = TRUE;
842 cps->useUsermove = FALSE;
843 cps->sendICS = FALSE;
844 cps->sendName = appData.icsActive;
845 cps->sdKludge = FALSE;
846 cps->stKludge = FALSE;
847 if(cps->tidy == NULL) cps->tidy = (char*) malloc(MSG_SIZ);
848 TidyProgramName(cps->program, cps->host, cps->tidy);
850 ASSIGN(cps->variants, appData.variant);
851 cps->analysisSupport = 2; /* detect */
852 cps->analyzing = FALSE;
853 cps->initDone = FALSE;
855 cps->pseudo = appData.pseudo[n];
857 /* New features added by Tord: */
858 cps->useFEN960 = FALSE;
859 cps->useOOCastle = TRUE;
860 /* End of new features added by Tord. */
861 cps->fenOverride = appData.fenOverride[n];
863 /* [HGM] time odds: set factor for each machine */
864 cps->timeOdds = appData.timeOdds[n];
866 /* [HGM] secondary TC: how to handle sessions that do not fit in 'level'*/
867 cps->accumulateTC = appData.accumulateTC[n];
868 cps->maxNrOfSessions = 1;
873 cps->drawDepth = appData.drawDepth[n];
874 cps->supportsNPS = UNKNOWN;
875 cps->memSize = FALSE;
876 cps->maxCores = FALSE;
877 ASSIGN(cps->egtFormats, "");
880 cps->optionSettings = appData.engOptions[n];
882 cps->scoreIsAbsolute = appData.scoreIsAbsolute[n]; /* [AS] */
883 cps->isUCI = appData.isUCI[n]; /* [AS] */
884 cps->hasOwnBookUCI = appData.hasOwnBookUCI[n]; /* [AS] */
887 if (appData.protocolVersion[n] > PROTOVER
888 || appData.protocolVersion[n] < 1)
893 len = snprintf(buf, MSG_SIZ, _("protocol version %d not supported"),
894 appData.protocolVersion[n]);
895 if( (len >= MSG_SIZ) && appData.debugMode )
896 fprintf(debugFP, "InitBackEnd1: buffer truncated.\n");
898 DisplayFatalError(buf, 0, 2);
902 cps->protocolVersion = appData.protocolVersion[n];
905 InitEngineUCI( installDir, cps ); // [HGM] moved here from winboard.c, to make available in xboard
906 ParseFeatures(appData.featureDefaults, cps);
909 ChessProgramState *savCps;
917 if(WaitForEngine(savCps, LoadEngine)) return;
918 CommonEngineInit(); // recalculate time odds
919 if(gameInfo.variant != StringToVariant(appData.variant)) {
920 // we changed variant when loading the engine; this forces us to reset
921 Reset(TRUE, savCps != &first);
922 oldMode = BeginningOfGame; // to prevent restoring old mode
924 InitChessProgram(savCps, FALSE);
925 if(gameMode == EditGame) SendToProgram("force\n", savCps); // in EditGame mode engine must be in force mode
926 DisplayMessage("", "");
927 if (startedFromSetupPosition) SendBoard(savCps, backwardMostMove);
928 for (i = backwardMostMove; i < currentMove; i++) SendMoveToProgram(i, savCps);
931 if(oldMode == AnalyzeMode) AnalyzeModeEvent();
935 ReplaceEngine (ChessProgramState *cps, int n)
937 oldMode = gameMode; // remember mode, so it can be restored after loading sequence is complete
939 if(oldMode != BeginningOfGame) EditGameEvent();
942 appData.noChessProgram = FALSE;
943 appData.clockMode = TRUE;
946 if(n) return; // only startup first engine immediately; second can wait
947 savCps = cps; // parameter to LoadEngine passed as globals, to allow scheduled calling :-(
951 extern char *engineName, *engineDir, *engineChoice, *engineLine, *nickName, *params;
952 extern Boolean isUCI, hasBook, storeVariant, v1, addToList, useNick;
954 static char resetOptions[] =
955 "-reuse -firstIsUCI false -firstHasOwnBookUCI true -firstTimeOdds 1 "
956 "-firstInitString \"" INIT_STRING "\" -firstComputerString \"" COMPUTER_STRING "\" "
957 "-firstFeatures \"\" -firstLogo \"\" -firstAccumulateTC 1 "
958 "-firstOptions \"\" -firstNPS -1 -fn \"\" -firstScoreAbs false";
961 FloatToFront(char **list, char *engineLine)
963 char buf[MSG_SIZ], tidy[MSG_SIZ], *p = buf, *q, *r = buf;
965 if(appData.recentEngines <= 0) return;
966 TidyProgramName(engineLine, "localhost", tidy+1);
967 tidy[0] = buf[0] = '\n'; strcat(tidy, "\n");
968 strncpy(buf+1, *list, MSG_SIZ-50);
969 if(p = strstr(buf, tidy)) { // tidy name appears in list
970 q = strchr(++p, '\n'); if(q == NULL) return; // malformed, don't touch
971 while(*p++ = *++q); // squeeze out
973 strcat(tidy, buf+1); // put list behind tidy name
974 p = tidy + 1; while(q = strchr(p, '\n')) i++, r = p, p = q + 1; // count entries in new list
975 if(i > appData.recentEngines) *r = NULLCHAR; // if maximum rached, strip off last
976 ASSIGN(*list, tidy+1);
979 char *insert, *wbOptions; // point in ChessProgramNames were we should insert new engine
982 Load (ChessProgramState *cps, int i)
984 char *p, *q, buf[MSG_SIZ], command[MSG_SIZ], buf2[MSG_SIZ], buf3[MSG_SIZ], jar;
985 if(engineLine && engineLine[0]) { // an engine was selected from the combo box
986 snprintf(buf, MSG_SIZ, "-fcp %s", engineLine);
987 SwapEngines(i); // kludge to parse -f* / -first* like it is -s* / -second*
988 ParseArgsFromString(resetOptions); appData.pvSAN[0] = FALSE;
989 FREE(appData.fenOverride[0]); appData.fenOverride[0] = NULL;
990 appData.firstProtocolVersion = PROTOVER;
991 ParseArgsFromString(buf);
993 ReplaceEngine(cps, i);
994 FloatToFront(&appData.recentEngineList, engineLine);
998 while(q = strchr(p, SLASH)) p = q+1;
999 if(*p== NULLCHAR) { DisplayError(_("You did not specify the engine executable"), 0); return; }
1000 if(engineDir[0] != NULLCHAR) {
1001 ASSIGN(appData.directory[i], engineDir); p = engineName;
1002 } else if(p != engineName) { // derive directory from engine path, when not given
1004 ASSIGN(appData.directory[i], engineName);
1006 if(SLASH == '/' && p - engineName > 1) *(p -= 2) = '.'; // for XBoard use ./exeName as command after split!
1007 } else { ASSIGN(appData.directory[i], "."); }
1008 jar = (strstr(p, ".jar") == p + strlen(p) - 4);
1010 if(strchr(p, ' ') && !strchr(p, '"')) snprintf(buf2, MSG_SIZ, "\"%s\"", p), p = buf2; // quote if it contains spaces
1011 snprintf(command, MSG_SIZ, "%s %s", p, params);
1014 if(jar) { snprintf(buf3, MSG_SIZ, "java -jar %s", p); p = buf3; }
1015 ASSIGN(appData.chessProgram[i], p);
1016 appData.isUCI[i] = isUCI;
1017 appData.protocolVersion[i] = v1 ? 1 : PROTOVER;
1018 appData.hasOwnBookUCI[i] = hasBook;
1019 if(!nickName[0]) useNick = FALSE;
1020 if(useNick) ASSIGN(appData.pgnName[i], nickName);
1024 q = firstChessProgramNames;
1025 if(nickName[0]) snprintf(buf, MSG_SIZ, "\"%s\" -fcp ", nickName); else buf[0] = NULLCHAR;
1026 quote = strchr(p, '"') ? '\'' : '"'; // use single quotes around engine command if it contains double quotes
1027 snprintf(buf+strlen(buf), MSG_SIZ-strlen(buf), "%c%s%c -fd \"%s\"%s%s%s%s%s%s%s%s\n",
1028 quote, p, quote, appData.directory[i],
1029 useNick ? " -fn \"" : "",
1030 useNick ? nickName : "",
1031 useNick ? "\"" : "",
1032 v1 ? " -firstProtocolVersion 1" : "",
1033 hasBook ? "" : " -fNoOwnBookUCI",
1034 isUCI ? (isUCI == TRUE ? " -fUCI" : gameInfo.variant == VariantShogi ? " -fUSI" : " -fUCCI") : "",
1035 storeVariant ? " -variant " : "",
1036 storeVariant ? VariantName(gameInfo.variant) : "");
1037 if(wbOptions && wbOptions[0]) snprintf(buf+strlen(buf)-1, MSG_SIZ-strlen(buf), " %s\n", wbOptions);
1038 firstChessProgramNames = malloc(len = strlen(q) + strlen(buf) + 1);
1039 if(insert != q) insert[-1] = NULLCHAR;
1040 snprintf(firstChessProgramNames, len, "%s\n%s%s", q, buf, insert);
1042 FloatToFront(&appData.recentEngineList, buf);
1044 ReplaceEngine(cps, i);
1050 int matched, min, sec;
1052 * Parse timeControl resource
1054 if (!ParseTimeControl(appData.timeControl, appData.timeIncrement,
1055 appData.movesPerSession)) {
1057 snprintf(buf, sizeof(buf), _("bad timeControl option %s"), appData.timeControl);
1058 DisplayFatalError(buf, 0, 2);
1062 * Parse searchTime resource
1064 if (*appData.searchTime != NULLCHAR) {
1065 matched = sscanf(appData.searchTime, "%d:%d", &min, &sec);
1067 searchTime = min * 60;
1068 } else if (matched == 2) {
1069 searchTime = min * 60 + sec;
1072 snprintf(buf, sizeof(buf), _("bad searchTime option %s"), appData.searchTime);
1073 DisplayFatalError(buf, 0, 2);
1082 ShowThinkingEvent(); // [HGM] thinking: make sure post/nopost state is set according to options
1083 startVariant = StringToVariant(appData.variant); // [HGM] nicks: remember original variant
1085 GetTimeMark(&programStartTime);
1086 srandom((programStartTime.ms + 1000*programStartTime.sec)*0x1001001); // [HGM] book: makes sure random is unpredictabe to msec level
1087 appData.seedBase = random() + (random()<<15);
1088 pauseStart = programStartTime; pauseStart.sec -= 100; // [HGM] matchpause: fake a pause that has long since ended
1090 ClearProgramStats();
1091 programStats.ok_to_send = 1;
1092 programStats.seen_stat = 0;
1095 * Initialize game list
1101 * Internet chess server status
1103 if (appData.icsActive) {
1104 appData.matchMode = FALSE;
1105 appData.matchGames = 0;
1107 appData.noChessProgram = !appData.zippyPlay;
1109 appData.zippyPlay = FALSE;
1110 appData.zippyTalk = FALSE;
1111 appData.noChessProgram = TRUE;
1113 if (*appData.icsHelper != NULLCHAR) {
1114 appData.useTelnet = TRUE;
1115 appData.telnetProgram = appData.icsHelper;
1118 appData.zippyTalk = appData.zippyPlay = FALSE;
1121 /* [AS] Initialize pv info list [HGM] and game state */
1125 for( i=0; i<=framePtr; i++ ) {
1126 pvInfoList[i].depth = -1;
1127 boards[i][EP_STATUS] = EP_NONE;
1128 for( j=0; j<BOARD_FILES-2; j++ ) boards[i][CASTLING][j] = NoRights;
1134 /* [AS] Adjudication threshold */
1135 adjudicateLossThreshold = appData.adjudicateLossThreshold;
1137 InitEngine(&first, 0);
1138 InitEngine(&second, 1);
1141 pairing.which = "pairing"; // pairing engine
1142 pairing.pr = NoProc;
1144 pairing.program = appData.pairingEngine;
1145 pairing.host = "localhost";
1148 if (appData.icsActive) {
1149 appData.clockMode = TRUE; /* changes dynamically in ICS mode */
1150 } else if (appData.noChessProgram) { // [HGM] st: searchTime mode now also is clockMode
1151 appData.clockMode = FALSE;
1152 first.sendTime = second.sendTime = 0;
1156 /* Override some settings from environment variables, for backward
1157 compatibility. Unfortunately it's not feasible to have the env
1158 vars just set defaults, at least in xboard. Ugh.
1160 if (appData.icsActive && (appData.zippyPlay || appData.zippyTalk)) {
1165 if (!appData.icsActive) {
1169 /* Check for variants that are supported only in ICS mode,
1170 or not at all. Some that are accepted here nevertheless
1171 have bugs; see comments below.
1173 VariantClass variant = StringToVariant(appData.variant);
1175 case VariantBughouse: /* need four players and two boards */
1176 case VariantKriegspiel: /* need to hide pieces and move details */
1177 /* case VariantFischeRandom: (Fabien: moved below) */
1178 len = snprintf(buf,MSG_SIZ, _("Variant %s supported only in ICS mode"), appData.variant);
1179 if( (len >= MSG_SIZ) && appData.debugMode )
1180 fprintf(debugFP, "InitBackEnd1: buffer truncated.\n");
1182 DisplayFatalError(buf, 0, 2);
1185 case VariantUnknown:
1186 case VariantLoadable:
1196 len = snprintf(buf, MSG_SIZ, _("Unknown variant name %s"), appData.variant);
1197 if( (len >= MSG_SIZ) && appData.debugMode )
1198 fprintf(debugFP, "InitBackEnd1: buffer truncated.\n");
1200 DisplayFatalError(buf, 0, 2);
1203 case VariantXiangqi: /* [HGM] repetition rules not implemented */
1204 case VariantFairy: /* [HGM] TestLegality definitely off! */
1205 case VariantGothic: /* [HGM] should work */
1206 case VariantCapablanca: /* [HGM] should work */
1207 case VariantCourier: /* [HGM] initial forced moves not implemented */
1208 case VariantShogi: /* [HGM] could still mate with pawn drop */
1209 case VariantChu: /* [HGM] experimental */
1210 case VariantKnightmate: /* [HGM] should work */
1211 case VariantCylinder: /* [HGM] untested */
1212 case VariantFalcon: /* [HGM] untested */
1213 case VariantCrazyhouse: /* holdings not shown, ([HGM] fixed that!)
1214 offboard interposition not understood */
1215 case VariantNormal: /* definitely works! */
1216 case VariantWildCastle: /* pieces not automatically shuffled */
1217 case VariantNoCastle: /* pieces not automatically shuffled */
1218 case VariantFischeRandom: /* [HGM] works and shuffles pieces */
1219 case VariantLosers: /* should work except for win condition,
1220 and doesn't know captures are mandatory */
1221 case VariantSuicide: /* should work except for win condition,
1222 and doesn't know captures are mandatory */
1223 case VariantGiveaway: /* should work except for win condition,
1224 and doesn't know captures are mandatory */
1225 case VariantTwoKings: /* should work */
1226 case VariantAtomic: /* should work except for win condition */
1227 case Variant3Check: /* should work except for win condition */
1228 case VariantShatranj: /* should work except for all win conditions */
1229 case VariantMakruk: /* should work except for draw countdown */
1230 case VariantASEAN : /* should work except for draw countdown */
1231 case VariantBerolina: /* might work if TestLegality is off */
1232 case VariantCapaRandom: /* should work */
1233 case VariantJanus: /* should work */
1234 case VariantSuper: /* experimental */
1235 case VariantGreat: /* experimental, requires legality testing to be off */
1236 case VariantSChess: /* S-Chess, should work */
1237 case VariantGrand: /* should work */
1238 case VariantSpartan: /* should work */
1239 case VariantLion: /* should work */
1240 case VariantChuChess: /* should work */
1248 NextIntegerFromString (char ** str, long * value)
1253 while( *s == ' ' || *s == '\t' ) {
1259 if( *s >= '0' && *s <= '9' ) {
1260 while( *s >= '0' && *s <= '9' ) {
1261 *value = *value * 10 + (*s - '0');
1274 NextTimeControlFromString (char ** str, long * value)
1277 int result = NextIntegerFromString( str, &temp );
1280 *value = temp * 60; /* Minutes */
1281 if( **str == ':' ) {
1283 result = NextIntegerFromString( str, &temp );
1284 *value += temp; /* Seconds */
1292 NextSessionFromString (char ** str, int *moves, long * tc, long *inc, int *incType)
1293 { /* [HGM] routine added to read '+moves/time' for secondary time control. */
1294 int result = -1, type = 0; long temp, temp2;
1296 if(**str != ':') return -1; // old params remain in force!
1298 if(**str == '*') type = *(*str)++, temp = 0; // sandclock TC
1299 if( NextIntegerFromString( str, &temp ) ) return -1;
1300 if(type) { *moves = 0; *tc = temp * 500; *inc = temp * 1000; *incType = '*'; return 0; }
1303 /* time only: incremental or sudden-death time control */
1304 if(**str == '+') { /* increment follows; read it */
1306 if(**str == '!') type = *(*str)++; // Bronstein TC
1307 if(result = NextIntegerFromString( str, &temp2)) return -1;
1308 *inc = temp2 * 1000;
1309 if(**str == '.') { // read fraction of increment
1310 char *start = ++(*str);
1311 if(result = NextIntegerFromString( str, &temp2)) return -1;
1313 while(start++ < *str) temp2 /= 10;
1317 *moves = 0; *tc = temp * 1000; *incType = type;
1321 (*str)++; /* classical time control */
1322 result = NextIntegerFromString( str, &temp2); // NOTE: already converted to seconds by ParseTimeControl()
1334 GetTimeQuota (int movenr, int lastUsed, char *tcString)
1335 { /* [HGM] get time to add from the multi-session time-control string */
1336 int incType, moves=1; /* kludge to force reading of first session */
1337 long time, increment;
1340 if(!s || !*s) return 0; // empty TC string means we ran out of the last sudden-death version
1342 if(moves) NextSessionFromString(&s, &moves, &time, &increment, &incType);
1343 nextSession = s; suddenDeath = moves == 0 && increment == 0;
1344 if(movenr == -1) return time; /* last move before new session */
1345 if(incType == '*') increment = 0; else // for sandclock, time is added while not thinking
1346 if(incType == '!' && lastUsed < increment) increment = lastUsed;
1347 if(!moves) return increment; /* current session is incremental */
1348 if(movenr >= 0) movenr -= moves; /* we already finished this session */
1349 } while(movenr >= -1); /* try again for next session */
1351 return 0; // no new time quota on this move
1355 ParseTimeControl (char *tc, float ti, int mps)
1359 char buf[MSG_SIZ], buf2[MSG_SIZ], *mytc = tc;
1362 if(ti >= 0 && !strchr(tc, '+') && !strchr(tc, '/') ) mps = 0;
1363 if(!strchr(tc, '+') && !strchr(tc, '/') && sscanf(tc, "%d:%d", &min, &sec) >= 1)
1364 sprintf(mytc=buf2, "%d", 60*min+sec); // convert 'classical' min:sec tc string to seconds
1368 snprintf(buf, MSG_SIZ, ":%d/%s+%g", mps, mytc, ti);
1370 snprintf(buf, MSG_SIZ, ":%s+%g", mytc, ti);
1373 snprintf(buf, MSG_SIZ, ":%d/%s", mps, mytc);
1375 snprintf(buf, MSG_SIZ, ":%s", mytc);
1377 fullTimeControlString = StrSave(buf); // this should now be in PGN format
1379 if( NextTimeControlFromString( &tc, &tc1 ) != 0 ) {
1384 /* Parse second time control */
1387 if( NextTimeControlFromString( &tc, &tc2 ) != 0 ) {
1395 timeControl_2 = tc2 * 1000;
1405 timeControl = tc1 * 1000;
1408 timeIncrement = ti * 1000; /* convert to ms */
1409 movesPerSession = 0;
1412 movesPerSession = mps;
1420 if (appData.debugMode) {
1421 # ifdef __GIT_VERSION
1422 fprintf(debugFP, "Version: %s (%s)\n", programVersion, __GIT_VERSION);
1424 fprintf(debugFP, "Version: %s\n", programVersion);
1427 ASSIGN(currentDebugFile, appData.nameOfDebugFile); // [HGM] debug split: remember initial name in use
1429 set_cont_sequence(appData.wrapContSeq);
1430 if (appData.matchGames > 0) {
1431 appData.matchMode = TRUE;
1432 } else if (appData.matchMode) {
1433 appData.matchGames = 1;
1435 if(appData.matchMode && appData.sameColorGames > 0) /* [HGM] alternate: overrule matchGames */
1436 appData.matchGames = appData.sameColorGames;
1437 if(appData.rewindIndex > 1) { /* [HGM] autoinc: rewind implies auto-increment and overrules given index */
1438 if(appData.loadPositionIndex >= 0) appData.loadPositionIndex = -1;
1439 if(appData.loadGameIndex >= 0) appData.loadGameIndex = -1;
1442 if (appData.noChessProgram || first.protocolVersion == 1) {
1445 /* kludge: allow timeout for initial "feature" commands */
1447 DisplayMessage("", _("Starting chess program"));
1448 ScheduleDelayedEvent(InitBackEnd3, FEATURE_TIMEOUT);
1453 CalculateIndex (int index, int gameNr)
1454 { // [HGM] autoinc: absolute way to determine load index from game number (taking auto-inc and rewind into account)
1456 if(index > 0) return index; // fixed nmber
1457 if(index == 0) return 1;
1458 res = (index == -1 ? gameNr : (gameNr-1)/2 + 1); // autoinc
1459 if(appData.rewindIndex > 0) res = (res-1) % appData.rewindIndex + 1; // rewind
1464 LoadGameOrPosition (int gameNr)
1465 { // [HGM] taken out of MatchEvent and NextMatchGame (to combine it)
1466 if (*appData.loadGameFile != NULLCHAR) {
1467 if (!LoadGameFromFile(appData.loadGameFile,
1468 CalculateIndex(appData.loadGameIndex, gameNr),
1469 appData.loadGameFile, FALSE)) {
1470 DisplayFatalError(_("Bad game file"), 0, 1);
1473 } else if (*appData.loadPositionFile != NULLCHAR) {
1474 if (!LoadPositionFromFile(appData.loadPositionFile,
1475 CalculateIndex(appData.loadPositionIndex, gameNr),
1476 appData.loadPositionFile)) {
1477 DisplayFatalError(_("Bad position file"), 0, 1);
1485 ReserveGame (int gameNr, char resChar)
1487 FILE *tf = fopen(appData.tourneyFile, "r+");
1488 char *p, *q, c, buf[MSG_SIZ];
1489 if(tf == NULL) { nextGame = appData.matchGames + 1; return; } // kludge to terminate match
1490 safeStrCpy(buf, lastMsg, MSG_SIZ);
1491 DisplayMessage(_("Pick new game"), "");
1492 flock(fileno(tf), LOCK_EX); // lock the tourney file while we are messing with it
1493 ParseArgsFromFile(tf);
1494 p = q = appData.results;
1495 if(appData.debugMode) {
1496 char *r = appData.participants;
1497 fprintf(debugFP, "results = '%s'\n", p);
1498 while(*r) fprintf(debugFP, *r >= ' ' ? "%c" : "\\%03o", *r), r++;
1499 fprintf(debugFP, "\n");
1501 while(*q && *q != ' ') q++; // get first un-played game (could be beyond end!)
1503 q = malloc(strlen(p) + 2); // could be arbitrary long, but allow to extend by one!
1504 safeStrCpy(q, p, strlen(p) + 2);
1505 if(gameNr >= 0) q[gameNr] = resChar; // replace '*' with result
1506 if(appData.debugMode) fprintf(debugFP, "pick next game from '%s': %d\n", q, nextGame);
1507 if(nextGame <= appData.matchGames && resChar != ' ' && !abortMatch) { // reserve next game if tourney not yet done
1508 if(q[nextGame] == NULLCHAR) q[nextGame+1] = NULLCHAR; // append one char
1511 fseek(tf, -(strlen(p)+4), SEEK_END);
1513 if(c != '"') // depending on DOS or Unix line endings we can be one off
1514 fseek(tf, -(strlen(p)+2), SEEK_END);
1515 else fseek(tf, -(strlen(p)+3), SEEK_END);
1516 fprintf(tf, "%s\"\n", q); fclose(tf); // update, and flush by closing
1517 DisplayMessage(buf, "");
1518 free(p); appData.results = q;
1519 if(nextGame <= appData.matchGames && resChar != ' ' && !abortMatch &&
1520 (gameNr < 0 || nextGame / appData.defaultMatchGames != gameNr / appData.defaultMatchGames)) {
1521 int round = appData.defaultMatchGames * appData.tourneyType;
1522 if(gameNr < 0 || appData.tourneyType < 1 || // gauntlet engine can always stay loaded as first engine
1523 appData.tourneyType > 1 && nextGame/round != gameNr/round) // in multi-gauntlet change only after round
1524 UnloadEngine(&first); // next game belongs to other pairing;
1525 UnloadEngine(&second); // already unload the engines, so TwoMachinesEvent will load new ones.
1527 if(appData.debugMode) fprintf(debugFP, "Reserved, next=%d, nr=%d\n", nextGame, gameNr);
1531 MatchEvent (int mode)
1532 { // [HGM] moved out of InitBackend3, to make it callable when match starts through menu
1534 if(matchMode) { // already in match mode: switch it off
1536 if(!appData.tourneyFile[0]) appData.matchGames = matchGame; // kludge to let match terminate after next game.
1539 // if(gameMode != BeginningOfGame) {
1540 // DisplayError(_("You can only start a match from the initial position."), 0);
1544 if(mode == 2) appData.matchGames = appData.defaultMatchGames;
1545 /* Set up machine vs. machine match */
1547 NextTourneyGame(-1, &dummy); // sets appData.matchGames if this is tourney, to make sure ReserveGame knows it
1548 if(appData.tourneyFile[0]) {
1550 if(nextGame > appData.matchGames) {
1552 if(strchr(appData.results, '*') == NULL) {
1554 appData.tourneyCycles++;
1555 if(f = WriteTourneyFile(appData.results, NULL)) { // make a tourney file with increased number of cycles
1557 NextTourneyGame(-1, &dummy);
1559 if(nextGame <= appData.matchGames) {
1560 DisplayNote(_("You restarted an already completed tourney.\nOne more cycle will now be added to it.\nGames commence in 10 sec."));
1562 ScheduleDelayedEvent(NextMatchGame, 10000);
1567 snprintf(buf, MSG_SIZ, _("All games in tourney '%s' are already played or playing"), appData.tourneyFile);
1568 DisplayError(buf, 0);
1569 appData.tourneyFile[0] = 0;
1573 if (appData.noChessProgram) { // [HGM] in tourney engines are loaded automatically
1574 DisplayFatalError(_("Can't have a match with no chess programs"),
1579 matchGame = roundNr = 1;
1580 first.matchWins = second.matchWins = 0; // [HGM] match: needed in later matches
1584 char *comboLine = NULL; // [HGM] recent: WinBoard's first-engine combobox line
1587 InitBackEnd3 P((void))
1589 GameMode initialMode;
1593 if(!appData.icsActive && !appData.noChessProgram && !appData.matchMode && // mode involves only first engine
1594 !strcmp(appData.variant, "normal") && // no explicit variant request
1595 appData.NrRanks == -1 && appData.NrFiles == -1 && appData.holdingsSize == -1 && // no size overrides requested
1596 !SupportedVariant(first.variants, VariantNormal, 8, 8, 0, first.protocolVersion, "") && // but 'normal' won't work with engine
1597 !SupportedVariant(first.variants, VariantFischeRandom, 8, 8, 0, first.protocolVersion, "") ) { // nor will Chess960
1598 char c, *q = first.variants, *p = strchr(q, ',');
1599 if(p) *p = NULLCHAR;
1600 if(StringToVariant(q) != VariantUnknown) { // the engine can play a recognized variant, however
1602 if(sscanf(q, "%dx%d+%d_%c", &w, &h, &s, &c) == 4) // get size overrides the engine needs with it (if any)
1603 appData.NrFiles = w, appData.NrRanks = h, appData.holdingsSize = s, q = strchr(q, '_') + 1;
1604 ASSIGN(appData.variant, q); // fake user requested the first variant played by the engine
1605 Reset(TRUE, FALSE); // and re-initialize
1610 InitChessProgram(&first, startedFromSetupPosition);
1612 if(!appData.noChessProgram) { /* [HGM] tidy: redo program version to use name from myname feature */
1613 free(programVersion);
1614 programVersion = (char*) malloc(8 + strlen(PACKAGE_STRING) + strlen(first.tidy));
1615 sprintf(programVersion, "%s + %s", PACKAGE_STRING, first.tidy);
1616 FloatToFront(&appData.recentEngineList, comboLine ? comboLine : appData.firstChessProgram);
1619 if (appData.icsActive) {
1621 /* [DM] Make a console window if needed [HGM] merged ifs */
1627 if (*appData.icsCommPort != NULLCHAR)
1628 len = snprintf(buf, MSG_SIZ, _("Could not open comm port %s"),
1629 appData.icsCommPort);
1631 len = snprintf(buf, MSG_SIZ, _("Could not connect to host %s, port %s"),
1632 appData.icsHost, appData.icsPort);
1634 if( (len >= MSG_SIZ) && appData.debugMode )
1635 fprintf(debugFP, "InitBackEnd3: buffer truncated.\n");
1637 DisplayFatalError(buf, err, 1);
1642 AddInputSource(icsPR, FALSE, read_from_ics, &telnetISR);
1644 AddInputSource(NoProc, FALSE, read_from_player, &fromUserISR);
1645 if(appData.keepAlive) // [HGM] alive: schedule sending of dummy 'date' command
1646 ScheduleDelayedEvent(KeepAlive, appData.keepAlive*60*1000);
1647 } else if (appData.noChessProgram) {
1653 if (*appData.cmailGameName != NULLCHAR) {
1655 OpenLoopback(&cmailPR);
1657 AddInputSource(cmailPR, FALSE, CmailSigHandlerCallBack, &cmailISR);
1661 DisplayMessage("", "");
1662 if (StrCaseCmp(appData.initialMode, "") == 0) {
1663 initialMode = BeginningOfGame;
1664 if(!appData.icsActive && appData.noChessProgram) { // [HGM] could be fall-back
1665 gameMode = MachinePlaysBlack; // "Machine Black" might have been implicitly highlighted
1666 ModeHighlight(); // make sure XBoard knows it is highlighted, so it will un-highlight it
1667 gameMode = BeginningOfGame; // in case BeginningOfGame now means "Edit Position"
1670 } else if (StrCaseCmp(appData.initialMode, "TwoMachines") == 0) {
1671 initialMode = TwoMachinesPlay;
1672 } else if (StrCaseCmp(appData.initialMode, "AnalyzeFile") == 0) {
1673 initialMode = AnalyzeFile;
1674 } else if (StrCaseCmp(appData.initialMode, "Analysis") == 0) {
1675 initialMode = AnalyzeMode;
1676 } else if (StrCaseCmp(appData.initialMode, "MachineWhite") == 0) {
1677 initialMode = MachinePlaysWhite;
1678 } else if (StrCaseCmp(appData.initialMode, "MachineBlack") == 0) {
1679 initialMode = MachinePlaysBlack;
1680 } else if (StrCaseCmp(appData.initialMode, "EditGame") == 0) {
1681 initialMode = EditGame;
1682 } else if (StrCaseCmp(appData.initialMode, "EditPosition") == 0) {
1683 initialMode = EditPosition;
1684 } else if (StrCaseCmp(appData.initialMode, "Training") == 0) {
1685 initialMode = Training;
1687 len = snprintf(buf, MSG_SIZ, _("Unknown initialMode %s"), appData.initialMode);
1688 if( (len >= MSG_SIZ) && appData.debugMode )
1689 fprintf(debugFP, "InitBackEnd3: buffer truncated.\n");
1691 DisplayFatalError(buf, 0, 2);
1695 if (appData.matchMode) {
1696 if(appData.tourneyFile[0]) { // start tourney from command line
1698 if(f = fopen(appData.tourneyFile, "r")) {
1699 ParseArgsFromFile(f); // make sure tourney parmeters re known
1701 appData.clockMode = TRUE;
1703 } else appData.tourneyFile[0] = NULLCHAR; // for now ignore bad tourney file
1706 } else if (*appData.cmailGameName != NULLCHAR) {
1707 /* Set up cmail mode */
1708 ReloadCmailMsgEvent(TRUE);
1710 /* Set up other modes */
1711 if (initialMode == AnalyzeFile) {
1712 if (*appData.loadGameFile == NULLCHAR) {
1713 DisplayFatalError(_("AnalyzeFile mode requires a game file"), 0, 1);
1717 if (*appData.loadGameFile != NULLCHAR) {
1718 (void) LoadGameFromFile(appData.loadGameFile,
1719 appData.loadGameIndex,
1720 appData.loadGameFile, TRUE);
1721 } else if (*appData.loadPositionFile != NULLCHAR) {
1722 (void) LoadPositionFromFile(appData.loadPositionFile,
1723 appData.loadPositionIndex,
1724 appData.loadPositionFile);
1725 /* [HGM] try to make self-starting even after FEN load */
1726 /* to allow automatic setup of fairy variants with wtm */
1727 if(initialMode == BeginningOfGame && !blackPlaysFirst) {
1728 gameMode = BeginningOfGame;
1729 setboardSpoiledMachineBlack = 1;
1731 /* [HGM] loadPos: make that every new game uses the setup */
1732 /* from file as long as we do not switch variant */
1733 if(!blackPlaysFirst) {
1734 startedFromPositionFile = TRUE;
1735 CopyBoard(filePosition, boards[0]);
1738 if (initialMode == AnalyzeMode) {
1739 if (appData.noChessProgram) {
1740 DisplayFatalError(_("Analysis mode requires a chess engine"), 0, 2);
1743 if (appData.icsActive) {
1744 DisplayFatalError(_("Analysis mode does not work with ICS mode"),0,2);
1748 } else if (initialMode == AnalyzeFile) {
1749 appData.showThinking = TRUE; // [HGM] thinking: moved out of ShowThinkingEvent
1750 ShowThinkingEvent();
1752 AnalysisPeriodicEvent(1);
1753 } else if (initialMode == MachinePlaysWhite) {
1754 if (appData.noChessProgram) {
1755 DisplayFatalError(_("MachineWhite mode requires a chess engine"),
1759 if (appData.icsActive) {
1760 DisplayFatalError(_("MachineWhite mode does not work with ICS mode"),
1764 MachineWhiteEvent();
1765 } else if (initialMode == MachinePlaysBlack) {
1766 if (appData.noChessProgram) {
1767 DisplayFatalError(_("MachineBlack mode requires a chess engine"),
1771 if (appData.icsActive) {
1772 DisplayFatalError(_("MachineBlack mode does not work with ICS mode"),
1776 MachineBlackEvent();
1777 } else if (initialMode == TwoMachinesPlay) {
1778 if (appData.noChessProgram) {
1779 DisplayFatalError(_("TwoMachines mode requires a chess engine"),
1783 if (appData.icsActive) {
1784 DisplayFatalError(_("TwoMachines mode does not work with ICS mode"),
1789 } else if (initialMode == EditGame) {
1791 } else if (initialMode == EditPosition) {
1792 EditPositionEvent();
1793 } else if (initialMode == Training) {
1794 if (*appData.loadGameFile == NULLCHAR) {
1795 DisplayFatalError(_("Training mode requires a game file"), 0, 2);
1804 HistorySet (char movelist[][2*MOVE_LEN], int first, int last, int current)
1806 DisplayBook(current+1);
1808 MoveHistorySet( movelist, first, last, current, pvInfoList );
1810 EvalGraphSet( first, last, current, pvInfoList );
1812 MakeEngineOutputTitle();
1816 * Establish will establish a contact to a remote host.port.
1817 * Sets icsPR to a ProcRef for a process (or pseudo-process)
1818 * used to talk to the host.
1819 * Returns 0 if okay, error code if not.
1826 if (*appData.icsCommPort != NULLCHAR) {
1827 /* Talk to the host through a serial comm port */
1828 return OpenCommPort(appData.icsCommPort, &icsPR);
1830 } else if (*appData.gateway != NULLCHAR) {
1831 if (*appData.remoteShell == NULLCHAR) {
1832 /* Use the rcmd protocol to run telnet program on a gateway host */
1833 snprintf(buf, sizeof(buf), "%s %s %s",
1834 appData.telnetProgram, appData.icsHost, appData.icsPort);
1835 return OpenRcmd(appData.gateway, appData.remoteUser, buf, &icsPR);
1838 /* Use the rsh program to run telnet program on a gateway host */
1839 if (*appData.remoteUser == NULLCHAR) {
1840 snprintf(buf, sizeof(buf), "%s %s %s %s %s", appData.remoteShell,
1841 appData.gateway, appData.telnetProgram,
1842 appData.icsHost, appData.icsPort);
1844 snprintf(buf, sizeof(buf), "%s %s -l %s %s %s %s",
1845 appData.remoteShell, appData.gateway,
1846 appData.remoteUser, appData.telnetProgram,
1847 appData.icsHost, appData.icsPort);
1849 return StartChildProcess(buf, "", &icsPR);
1852 } else if (appData.useTelnet) {
1853 return OpenTelnet(appData.icsHost, appData.icsPort, &icsPR);
1856 /* TCP socket interface differs somewhat between
1857 Unix and NT; handle details in the front end.
1859 return OpenTCP(appData.icsHost, appData.icsPort, &icsPR);
1864 EscapeExpand (char *p, char *q)
1865 { // [HGM] initstring: routine to shape up string arguments
1866 while(*p++ = *q++) if(p[-1] == '\\')
1868 case 'n': p[-1] = '\n'; break;
1869 case 'r': p[-1] = '\r'; break;
1870 case 't': p[-1] = '\t'; break;
1871 case '\\': p[-1] = '\\'; break;
1872 case 0: *p = 0; return;
1873 default: p[-1] = q[-1]; break;
1878 show_bytes (FILE *fp, char *buf, int count)
1881 if (*buf < 040 || *(unsigned char *) buf > 0177) {
1882 fprintf(fp, "\\%03o", *buf & 0xff);
1891 /* Returns an errno value */
1893 OutputMaybeTelnet (ProcRef pr, char *message, int count, int *outError)
1895 char buf[8192], *p, *q, *buflim;
1896 int left, newcount, outcount;
1898 if (*appData.icsCommPort != NULLCHAR || appData.useTelnet ||
1899 *appData.gateway != NULLCHAR) {
1900 if (appData.debugMode) {
1901 fprintf(debugFP, ">ICS: ");
1902 show_bytes(debugFP, message, count);
1903 fprintf(debugFP, "\n");
1905 return OutputToProcess(pr, message, count, outError);
1908 buflim = &buf[sizeof(buf)-1]; /* allow 1 byte for expanding last char */
1915 if (appData.debugMode) {
1916 fprintf(debugFP, ">ICS: ");
1917 show_bytes(debugFP, buf, newcount);
1918 fprintf(debugFP, "\n");
1920 outcount = OutputToProcess(pr, buf, newcount, outError);
1921 if (outcount < newcount) return -1; /* to be sure */
1928 } else if (((unsigned char) *p) == TN_IAC) {
1929 *q++ = (char) TN_IAC;
1936 if (appData.debugMode) {
1937 fprintf(debugFP, ">ICS: ");
1938 show_bytes(debugFP, buf, newcount);
1939 fprintf(debugFP, "\n");
1941 outcount = OutputToProcess(pr, buf, newcount, outError);
1942 if (outcount < newcount) return -1; /* to be sure */
1947 read_from_player (InputSourceRef isr, VOIDSTAR closure, char *message, int count, int error)
1949 int outError, outCount;
1950 static int gotEof = 0;
1953 /* Pass data read from player on to ICS */
1956 outCount = OutputMaybeTelnet(icsPR, message, count, &outError);
1957 if (outCount < count) {
1958 DisplayFatalError(_("Error writing to ICS"), outError, 1);
1960 if(have_sent_ICS_logon == 2) {
1961 if(ini = fopen(appData.icsLogon, "w")) { // save first two lines (presumably username & password) on init script file
1962 fprintf(ini, "%s", message);
1963 have_sent_ICS_logon = 3;
1965 have_sent_ICS_logon = 1;
1966 } else if(have_sent_ICS_logon == 3) {
1967 fprintf(ini, "%s", message);
1969 have_sent_ICS_logon = 1;
1971 } else if (count < 0) {
1972 RemoveInputSource(isr);
1973 DisplayFatalError(_("Error reading from keyboard"), error, 1);
1974 } else if (gotEof++ > 0) {
1975 RemoveInputSource(isr);
1976 DisplayFatalError(_("Got end of file from keyboard"), 0, 0);
1982 { // [HGM] alive: periodically send dummy (date) command to ICS to prevent time-out
1983 if(!connectionAlive) DisplayFatalError("No response from ICS", 0, 1);
1984 connectionAlive = FALSE; // only sticks if no response to 'date' command.
1985 SendToICS("date\n");
1986 if(appData.keepAlive) ScheduleDelayedEvent(KeepAlive, appData.keepAlive*60*1000);
1989 /* added routine for printf style output to ics */
1991 ics_printf (char *format, ...)
1993 char buffer[MSG_SIZ];
1996 va_start(args, format);
1997 vsnprintf(buffer, sizeof(buffer), format, args);
1998 buffer[sizeof(buffer)-1] = '\0';
2006 int count, outCount, outError;
2008 if (icsPR == NoProc) return;
2011 outCount = OutputMaybeTelnet(icsPR, s, count, &outError);
2012 if (outCount < count) {
2013 DisplayFatalError(_("Error writing to ICS"), outError, 1);
2017 /* This is used for sending logon scripts to the ICS. Sending
2018 without a delay causes problems when using timestamp on ICC
2019 (at least on my machine). */
2021 SendToICSDelayed (char *s, long msdelay)
2023 int count, outCount, outError;
2025 if (icsPR == NoProc) return;
2028 if (appData.debugMode) {
2029 fprintf(debugFP, ">ICS: ");
2030 show_bytes(debugFP, s, count);
2031 fprintf(debugFP, "\n");
2033 outCount = OutputToProcessDelayed(icsPR, s, count, &outError,
2035 if (outCount < count) {
2036 DisplayFatalError(_("Error writing to ICS"), outError, 1);
2041 /* Remove all highlighting escape sequences in s
2042 Also deletes any suffix starting with '('
2045 StripHighlightAndTitle (char *s)
2047 static char retbuf[MSG_SIZ];
2050 while (*s != NULLCHAR) {
2051 while (*s == '\033') {
2052 while (*s != NULLCHAR && !isalpha(*s)) s++;
2053 if (*s != NULLCHAR) s++;
2055 while (*s != NULLCHAR && *s != '\033') {
2056 if (*s == '(' || *s == '[') {
2067 /* Remove all highlighting escape sequences in s */
2069 StripHighlight (char *s)
2071 static char retbuf[MSG_SIZ];
2074 while (*s != NULLCHAR) {
2075 while (*s == '\033') {
2076 while (*s != NULLCHAR && !isalpha(*s)) s++;
2077 if (*s != NULLCHAR) s++;
2079 while (*s != NULLCHAR && *s != '\033') {
2087 char engineVariant[MSG_SIZ];
2088 char *variantNames[] = VARIANT_NAMES;
2090 VariantName (VariantClass v)
2092 if(v == VariantUnknown || *engineVariant) return engineVariant;
2093 return variantNames[v];
2097 /* Identify a variant from the strings the chess servers use or the
2098 PGN Variant tag names we use. */
2100 StringToVariant (char *e)
2104 VariantClass v = VariantNormal;
2105 int i, found = FALSE;
2111 /* [HGM] skip over optional board-size prefixes */
2112 if( sscanf(e, "%dx%d_", &i, &i) == 2 ||
2113 sscanf(e, "%dx%d+%d_", &i, &i, &i) == 3 ) {
2114 while( *e++ != '_');
2117 if(StrCaseStr(e, "misc/")) { // [HGM] on FICS, misc/shogi is not shogi
2121 for (i=0; i<sizeof(variantNames)/sizeof(char*); i++) {
2122 if (p = StrCaseStr(e, variantNames[i])) {
2123 if(p && i >= VariantShogi && isalpha(p[strlen(variantNames[i])])) continue;
2124 v = (VariantClass) i;
2131 if ((StrCaseStr(e, "fischer") && StrCaseStr(e, "random"))
2132 || StrCaseStr(e, "wild/fr")
2133 || StrCaseStr(e, "frc") || StrCaseStr(e, "960")) {
2134 v = VariantFischeRandom;
2135 } else if ((i = 4, p = StrCaseStr(e, "wild")) ||
2136 (i = 1, p = StrCaseStr(e, "w"))) {
2138 while (*p && (isspace(*p) || *p == '(' || *p == '/')) p++;
2145 case 0: /* FICS only, actually */
2147 /* Castling legal even if K starts on d-file */
2148 v = VariantWildCastle;
2153 /* Castling illegal even if K & R happen to start in
2154 normal positions. */
2155 v = VariantNoCastle;
2168 /* Castling legal iff K & R start in normal positions */
2174 /* Special wilds for position setup; unclear what to do here */
2175 v = VariantLoadable;
2178 /* Bizarre ICC game */
2179 v = VariantTwoKings;
2182 v = VariantKriegspiel;
2188 v = VariantFischeRandom;
2191 v = VariantCrazyhouse;
2194 v = VariantBughouse;
2200 /* Not quite the same as FICS suicide! */
2201 v = VariantGiveaway;
2207 v = VariantShatranj;
2210 /* Temporary names for future ICC types. The name *will* change in
2211 the next xboard/WinBoard release after ICC defines it. */
2249 v = VariantCapablanca;
2252 v = VariantKnightmate;
2258 v = VariantCylinder;
2264 v = VariantCapaRandom;
2267 v = VariantBerolina;
2279 /* Found "wild" or "w" in the string but no number;
2280 must assume it's normal chess. */
2284 len = snprintf(buf, MSG_SIZ, _("Unknown wild type %d"), wnum);
2285 if( (len >= MSG_SIZ) && appData.debugMode )
2286 fprintf(debugFP, "StringToVariant: buffer truncated.\n");
2288 DisplayError(buf, 0);
2294 if (appData.debugMode) {
2295 fprintf(debugFP, "recognized '%s' (%d) as variant %s\n",
2296 e, wnum, VariantName(v));
2301 static int leftover_start = 0, leftover_len = 0;
2302 char star_match[STAR_MATCH_N][MSG_SIZ];
2304 /* Test whether pattern is present at &buf[*index]; if so, return TRUE,
2305 advance *index beyond it, and set leftover_start to the new value of
2306 *index; else return FALSE. If pattern contains the character '*', it
2307 matches any sequence of characters not containing '\r', '\n', or the
2308 character following the '*' (if any), and the matched sequence(s) are
2309 copied into star_match.
2312 looking_at ( char *buf, int *index, char *pattern)
2314 char *bufp = &buf[*index], *patternp = pattern;
2316 char *matchp = star_match[0];
2319 if (*patternp == NULLCHAR) {
2320 *index = leftover_start = bufp - buf;
2324 if (*bufp == NULLCHAR) return FALSE;
2325 if (*patternp == '*') {
2326 if (*bufp == *(patternp + 1)) {
2328 matchp = star_match[++star_count];
2332 } else if (*bufp == '\n' || *bufp == '\r') {
2334 if (*patternp == NULLCHAR)
2339 *matchp++ = *bufp++;
2343 if (*patternp != *bufp) return FALSE;
2350 SendToPlayer (char *data, int length)
2352 int error, outCount;
2353 outCount = OutputToProcess(NoProc, data, length, &error);
2354 if (outCount < length) {
2355 DisplayFatalError(_("Error writing to display"), error, 1);
2360 PackHolding (char packed[], char *holding)
2370 switch (runlength) {
2381 sprintf(q, "%d", runlength);
2393 /* Telnet protocol requests from the front end */
2395 TelnetRequest (unsigned char ddww, unsigned char option)
2397 unsigned char msg[3];
2398 int outCount, outError;
2400 if (*appData.icsCommPort != NULLCHAR || appData.useTelnet) return;
2402 if (appData.debugMode) {
2403 char buf1[8], buf2[8], *ddwwStr, *optionStr;
2419 snprintf(buf1,sizeof(buf1)/sizeof(buf1[0]), "%d", ddww);
2428 snprintf(buf2,sizeof(buf2)/sizeof(buf2[0]), "%d", option);
2431 fprintf(debugFP, ">%s %s ", ddwwStr, optionStr);
2436 outCount = OutputToProcess(icsPR, (char *)msg, 3, &outError);
2438 DisplayFatalError(_("Error writing to ICS"), outError, 1);
2445 if (!appData.icsActive) return;
2446 TelnetRequest(TN_DO, TN_ECHO);
2452 if (!appData.icsActive) return;
2453 TelnetRequest(TN_DONT, TN_ECHO);
2457 CopyHoldings (Board board, char *holdings, ChessSquare lowestPiece)
2459 /* put the holdings sent to us by the server on the board holdings area */
2460 int i, j, holdingsColumn, holdingsStartRow, direction, countsColumn;
2464 if(gameInfo.holdingsWidth < 2) return;
2465 if(gameInfo.variant != VariantBughouse && board[HOLDINGS_SET])
2466 return; // prevent overwriting by pre-board holdings
2468 if( (int)lowestPiece >= BlackPawn ) {
2471 holdingsStartRow = BOARD_HEIGHT-1;
2474 holdingsColumn = BOARD_WIDTH-1;
2475 countsColumn = BOARD_WIDTH-2;
2476 holdingsStartRow = 0;
2480 for(i=0; i<BOARD_HEIGHT; i++) { /* clear holdings */
2481 board[i][holdingsColumn] = EmptySquare;
2482 board[i][countsColumn] = (ChessSquare) 0;
2484 while( (p=*holdings++) != NULLCHAR ) {
2485 piece = CharToPiece( ToUpper(p) );
2486 if(piece == EmptySquare) continue;
2487 /*j = (int) piece - (int) WhitePawn;*/
2488 j = PieceToNumber(piece);
2489 if(j >= gameInfo.holdingsSize) continue; /* ignore pieces that do not fit */
2490 if(j < 0) continue; /* should not happen */
2491 piece = (ChessSquare) ( (int)piece + (int)lowestPiece );
2492 board[holdingsStartRow+j*direction][holdingsColumn] = piece;
2493 board[holdingsStartRow+j*direction][countsColumn]++;
2499 VariantSwitch (Board board, VariantClass newVariant)
2501 int newHoldingsWidth, newWidth = 8, newHeight = 8, i, j;
2502 static Board oldBoard;
2504 startedFromPositionFile = FALSE;
2505 if(gameInfo.variant == newVariant) return;
2507 /* [HGM] This routine is called each time an assignment is made to
2508 * gameInfo.variant during a game, to make sure the board sizes
2509 * are set to match the new variant. If that means adding or deleting
2510 * holdings, we shift the playing board accordingly
2511 * This kludge is needed because in ICS observe mode, we get boards
2512 * of an ongoing game without knowing the variant, and learn about the
2513 * latter only later. This can be because of the move list we requested,
2514 * in which case the game history is refilled from the beginning anyway,
2515 * but also when receiving holdings of a crazyhouse game. In the latter
2516 * case we want to add those holdings to the already received position.
2520 if (appData.debugMode) {
2521 fprintf(debugFP, "Switch board from %s to %s\n",
2522 VariantName(gameInfo.variant), VariantName(newVariant));
2523 setbuf(debugFP, NULL);
2525 shuffleOpenings = 0; /* [HGM] shuffle */
2526 gameInfo.holdingsSize = 5; /* [HGM] prepare holdings */
2530 newWidth = 9; newHeight = 9;
2531 gameInfo.holdingsSize = 7;
2532 case VariantBughouse:
2533 case VariantCrazyhouse:
2534 newHoldingsWidth = 2; break;
2538 newHoldingsWidth = 2;
2539 gameInfo.holdingsSize = 8;
2542 case VariantCapablanca:
2543 case VariantCapaRandom:
2546 newHoldingsWidth = gameInfo.holdingsSize = 0;
2549 if(newWidth != gameInfo.boardWidth ||
2550 newHeight != gameInfo.boardHeight ||
2551 newHoldingsWidth != gameInfo.holdingsWidth ) {
2553 /* shift position to new playing area, if needed */
2554 if(newHoldingsWidth > gameInfo.holdingsWidth) {
2555 for(i=0; i<BOARD_HEIGHT; i++)
2556 for(j=BOARD_RGHT-1; j>=BOARD_LEFT; j--)
2557 board[i][j+newHoldingsWidth-gameInfo.holdingsWidth] =
2559 for(i=0; i<newHeight; i++) {
2560 board[i][0] = board[i][newWidth+2*newHoldingsWidth-1] = EmptySquare;
2561 board[i][1] = board[i][newWidth+2*newHoldingsWidth-2] = (ChessSquare) 0;
2563 } else if(newHoldingsWidth < gameInfo.holdingsWidth) {
2564 for(i=0; i<BOARD_HEIGHT; i++)
2565 for(j=BOARD_LEFT; j<BOARD_RGHT; j++)
2566 board[i][j+newHoldingsWidth-gameInfo.holdingsWidth] =
2569 board[HOLDINGS_SET] = 0;
2570 gameInfo.boardWidth = newWidth;
2571 gameInfo.boardHeight = newHeight;
2572 gameInfo.holdingsWidth = newHoldingsWidth;
2573 gameInfo.variant = newVariant;
2574 InitDrawingSizes(-2, 0);
2575 } else gameInfo.variant = newVariant;
2576 CopyBoard(oldBoard, board); // remember correctly formatted board
2577 InitPosition(FALSE); /* this sets up board[0], but also other stuff */
2578 DrawPosition(TRUE, currentMove ? boards[currentMove] : oldBoard);
2581 static int loggedOn = FALSE;
2583 /*-- Game start info cache: --*/
2585 char gs_kind[MSG_SIZ];
2586 static char player1Name[128] = "";
2587 static char player2Name[128] = "";
2588 static char cont_seq[] = "\n\\ ";
2589 static int player1Rating = -1;
2590 static int player2Rating = -1;
2591 /*----------------------------*/
2593 ColorClass curColor = ColorNormal;
2594 int suppressKibitz = 0;
2597 Boolean soughtPending = FALSE;
2598 Boolean seekGraphUp;
2599 #define MAX_SEEK_ADS 200
2601 char *seekAdList[MAX_SEEK_ADS];
2602 int ratingList[MAX_SEEK_ADS], xList[MAX_SEEK_ADS], yList[MAX_SEEK_ADS], seekNrList[MAX_SEEK_ADS], zList[MAX_SEEK_ADS];
2603 float tcList[MAX_SEEK_ADS];
2604 char colorList[MAX_SEEK_ADS];
2605 int nrOfSeekAds = 0;
2606 int minRating = 1010, maxRating = 2800;
2607 int hMargin = 10, vMargin = 20, h, w;
2608 extern int squareSize, lineGap;
2613 int x, y, color = 0, r = ratingList[i]; float tc = tcList[i];
2614 xList[i] = yList[i] = -100; // outside graph, so cannot be clicked
2615 if(r < minRating+100 && r >=0 ) r = minRating+100;
2616 if(r > maxRating) r = maxRating;
2617 if(tc < 1.f) tc = 1.f;
2618 if(tc > 95.f) tc = 95.f;
2619 x = (w-hMargin-squareSize/8-7)* log(tc)/log(95.) + hMargin;
2620 y = ((double)r - minRating)/(maxRating - minRating)
2621 * (h-vMargin-squareSize/8-1) + vMargin;
2622 if(ratingList[i] < 0) y = vMargin + squareSize/4;
2623 if(strstr(seekAdList[i], " u ")) color = 1;
2624 if(!strstr(seekAdList[i], "lightning") && // for now all wilds same color
2625 !strstr(seekAdList[i], "bullet") &&
2626 !strstr(seekAdList[i], "blitz") &&
2627 !strstr(seekAdList[i], "standard") ) color = 2;
2628 if(strstr(seekAdList[i], "(C) ")) color |= SQUARE; // plot computer seeks as squares
2629 DrawSeekDot(xList[i]=x+3*(color&~SQUARE), yList[i]=h-1-y, colorList[i]=color);
2633 PlotSingleSeekAd (int i)
2639 AddAd (char *handle, char *rating, int base, int inc, char rated, char *type, int nr, Boolean plot)
2641 char buf[MSG_SIZ], *ext = "";
2642 VariantClass v = StringToVariant(type);
2643 if(strstr(type, "wild")) {
2644 ext = type + 4; // append wild number
2645 if(v == VariantFischeRandom) type = "chess960"; else
2646 if(v == VariantLoadable) type = "setup"; else
2647 type = VariantName(v);
2649 snprintf(buf, MSG_SIZ, "%s (%s) %d %d %c %s%s", handle, rating, base, inc, rated, type, ext);
2650 if(nrOfSeekAds < MAX_SEEK_ADS-1) {
2651 if(seekAdList[nrOfSeekAds]) free(seekAdList[nrOfSeekAds]);
2652 ratingList[nrOfSeekAds] = -1; // for if seeker has no rating
2653 sscanf(rating, "%d", &ratingList[nrOfSeekAds]);
2654 tcList[nrOfSeekAds] = base + (2./3.)*inc;
2655 seekNrList[nrOfSeekAds] = nr;
2656 zList[nrOfSeekAds] = 0;
2657 seekAdList[nrOfSeekAds++] = StrSave(buf);
2658 if(plot) PlotSingleSeekAd(nrOfSeekAds-1);
2663 EraseSeekDot (int i)
2665 int x = xList[i], y = yList[i], d=squareSize/4, k;
2666 DrawSeekBackground(x-squareSize/8, y-squareSize/8, x+squareSize/8+1, y+squareSize/8+1);
2667 if(x < hMargin+d) DrawSeekAxis(hMargin, y-squareSize/8, hMargin, y+squareSize/8+1);
2668 // now replot every dot that overlapped
2669 for(k=0; k<nrOfSeekAds; k++) if(k != i) {
2670 int xx = xList[k], yy = yList[k];
2671 if(xx <= x+d && xx > x-d && yy <= y+d && yy > y-d)
2672 DrawSeekDot(xx, yy, colorList[k]);
2677 RemoveSeekAd (int nr)
2680 for(i=0; i<nrOfSeekAds; i++) if(seekNrList[i] == nr) {
2682 if(seekAdList[i]) free(seekAdList[i]);
2683 seekAdList[i] = seekAdList[--nrOfSeekAds];
2684 seekNrList[i] = seekNrList[nrOfSeekAds];
2685 ratingList[i] = ratingList[nrOfSeekAds];
2686 colorList[i] = colorList[nrOfSeekAds];
2687 tcList[i] = tcList[nrOfSeekAds];
2688 xList[i] = xList[nrOfSeekAds];
2689 yList[i] = yList[nrOfSeekAds];
2690 zList[i] = zList[nrOfSeekAds];
2691 seekAdList[nrOfSeekAds] = NULL;
2697 MatchSoughtLine (char *line)
2699 char handle[MSG_SIZ], rating[MSG_SIZ], type[MSG_SIZ];
2700 int nr, base, inc, u=0; char dummy;
2702 if(sscanf(line, "%d %s %s %d %d rated %s", &nr, rating, handle, &base, &inc, type) == 6 ||
2703 sscanf(line, "%d %s %s %s %d %d rated %c", &nr, rating, handle, type, &base, &inc, &dummy) == 7 ||
2705 (sscanf(line, "%d %s %s %d %d unrated %s", &nr, rating, handle, &base, &inc, type) == 6 ||
2706 sscanf(line, "%d %s %s %s %d %d unrated %c", &nr, rating, handle, type, &base, &inc, &dummy) == 7) ) {
2707 // match: compact and save the line
2708 AddAd(handle, rating, base, inc, u ? 'u' : 'r', type, nr, FALSE);
2718 if(!seekGraphUp) return FALSE;
2719 h = BOARD_HEIGHT * (squareSize + lineGap) + lineGap;
2720 w = BOARD_WIDTH * (squareSize + lineGap) + lineGap;
2722 DrawSeekBackground(0, 0, w, h);
2723 DrawSeekAxis(hMargin, h-1-vMargin, w-5, h-1-vMargin);
2724 DrawSeekAxis(hMargin, h-1-vMargin, hMargin, 5);
2725 for(i=0; i<4000; i+= 100) if(i>=minRating && i<maxRating) {
2726 int yy =((double)i - minRating)/(maxRating - minRating)*(h-vMargin-squareSize/8-1) + vMargin;
2728 DrawSeekAxis(hMargin-5, yy, hMargin+5*(i%500==0), yy); // rating ticks
2731 snprintf(buf, MSG_SIZ, "%d", i);
2732 DrawSeekText(buf, hMargin+squareSize/8+7, yy);
2735 DrawSeekText("unrated", hMargin+squareSize/8+7, h-1-vMargin-squareSize/4);
2736 for(i=1; i<100; i+=(i<10?1:5)) {
2737 int xx = (w-hMargin-squareSize/8-7)* log((double)i)/log(95.) + hMargin;
2738 DrawSeekAxis(xx, h-1-vMargin, xx, h-6-vMargin-3*(i%10==0)); // TC ticks
2739 if(i<=5 || (i>40 ? i%20 : i%10) == 0) {
2741 snprintf(buf, MSG_SIZ, "%d", i);
2742 DrawSeekText(buf, xx-2-3*(i>9), h-1-vMargin/2);
2745 for(i=0; i<nrOfSeekAds; i++) PlotSeekAd(i);
2750 SeekGraphClick (ClickType click, int x, int y, int moving)
2752 static int lastDown = 0, displayed = 0, lastSecond;
2753 if(y < 0) return FALSE;
2754 if(!(appData.seekGraph && appData.icsActive && loggedOn &&
2755 (gameMode == BeginningOfGame || gameMode == IcsIdle))) {
2756 if(!seekGraphUp) return FALSE;
2757 seekGraphUp = FALSE; // seek graph is up when it shouldn't be: take it down
2758 DrawPosition(TRUE, NULL);
2761 if(!seekGraphUp) { // initiate cration of seek graph by requesting seek-ad list
2762 if(click == Release || moving) return FALSE;
2764 soughtPending = TRUE;
2765 SendToICS(ics_prefix);
2766 SendToICS("sought\n"); // should this be "sought all"?
2767 } else { // issue challenge based on clicked ad
2768 int dist = 10000; int i, closest = 0, second = 0;
2769 for(i=0; i<nrOfSeekAds; i++) {
2770 int d = (x-xList[i])*(x-xList[i]) + (y-yList[i])*(y-yList[i]) + zList[i];
2771 if(d < dist) { dist = d; closest = i; }
2772 second += (d - zList[i] < 120); // count in-range ads
2773 if(click == Press && moving != 1 && zList[i]>0) zList[i] *= 0.8; // age priority
2777 second = (second > 1);
2778 if(displayed != closest || second != lastSecond) {
2779 DisplayMessage(second ? "!" : "", seekAdList[closest]);
2780 lastSecond = second; displayed = closest;
2782 if(click == Press) {
2783 if(moving == 2) zList[closest] = 100; // right-click; push to back on press
2786 } // on press 'hit', only show info
2787 if(moving == 2) return TRUE; // ignore right up-clicks on dot
2788 snprintf(buf, MSG_SIZ, "play %d\n", seekNrList[closest]);
2789 SendToICS(ics_prefix);
2791 return TRUE; // let incoming board of started game pop down the graph
2792 } else if(click == Release) { // release 'miss' is ignored
2793 zList[lastDown] = 100; // make future selection of the rejected ad more difficult
2794 if(moving == 2) { // right up-click
2795 nrOfSeekAds = 0; // refresh graph
2796 soughtPending = TRUE;
2797 SendToICS(ics_prefix);
2798 SendToICS("sought\n"); // should this be "sought all"?
2801 } else if(moving) { if(displayed >= 0) DisplayMessage("", ""); displayed = -1; return TRUE; }
2802 // press miss or release hit 'pop down' seek graph
2803 seekGraphUp = FALSE;
2804 DrawPosition(TRUE, NULL);
2810 read_from_ics (InputSourceRef isr, VOIDSTAR closure, char *data, int count, int error)
2812 #define BUF_SIZE (16*1024) /* overflowed at 8K with "inchannel 1" on FICS? */
2813 #define STARTED_NONE 0
2814 #define STARTED_MOVES 1
2815 #define STARTED_BOARD 2
2816 #define STARTED_OBSERVE 3
2817 #define STARTED_HOLDINGS 4
2818 #define STARTED_CHATTER 5
2819 #define STARTED_COMMENT 6
2820 #define STARTED_MOVES_NOHIDE 7
2822 static int started = STARTED_NONE;
2823 static char parse[20000];
2824 static int parse_pos = 0;
2825 static char buf[BUF_SIZE + 1];
2826 static int firstTime = TRUE, intfSet = FALSE;
2827 static ColorClass prevColor = ColorNormal;
2828 static int savingComment = FALSE;
2829 static int cmatch = 0; // continuation sequence match
2836 int backup; /* [DM] For zippy color lines */
2838 char talker[MSG_SIZ]; // [HGM] chat
2839 int channel, collective=0;
2841 connectionAlive = TRUE; // [HGM] alive: I think, therefore I am...
2843 if (appData.debugMode) {
2845 fprintf(debugFP, "<ICS: ");
2846 show_bytes(debugFP, data, count);
2847 fprintf(debugFP, "\n");
2851 if (appData.debugMode) { int f = forwardMostMove;
2852 fprintf(debugFP, "ics input %d, castling = %d %d %d %d %d %d\n", f,
2853 boards[f][CASTLING][0],boards[f][CASTLING][1],boards[f][CASTLING][2],
2854 boards[f][CASTLING][3],boards[f][CASTLING][4],boards[f][CASTLING][5]);
2857 /* If last read ended with a partial line that we couldn't parse,
2858 prepend it to the new read and try again. */
2859 if (leftover_len > 0) {
2860 for (i=0; i<leftover_len; i++)
2861 buf[i] = buf[leftover_start + i];
2864 /* copy new characters into the buffer */
2865 bp = buf + leftover_len;
2866 buf_len=leftover_len;
2867 for (i=0; i<count; i++)
2870 if (data[i] == '\r')
2873 // join lines split by ICS?
2874 if (!appData.noJoin)
2877 Joining just consists of finding matches against the
2878 continuation sequence, and discarding that sequence
2879 if found instead of copying it. So, until a match
2880 fails, there's nothing to do since it might be the
2881 complete sequence, and thus, something we don't want
2884 if (data[i] == cont_seq[cmatch])
2887 if (cmatch == strlen(cont_seq))
2889 cmatch = 0; // complete match. just reset the counter
2892 it's possible for the ICS to not include the space
2893 at the end of the last word, making our [correct]
2894 join operation fuse two separate words. the server
2895 does this when the space occurs at the width setting.
2897 if (!buf_len || buf[buf_len-1] != ' ')
2908 match failed, so we have to copy what matched before
2909 falling through and copying this character. In reality,
2910 this will only ever be just the newline character, but
2911 it doesn't hurt to be precise.
2913 strncpy(bp, cont_seq, cmatch);
2925 buf[buf_len] = NULLCHAR;
2926 // next_out = leftover_len; // [HGM] should we set this to 0, and not print it in advance?
2931 while (i < buf_len) {
2932 /* Deal with part of the TELNET option negotiation
2933 protocol. We refuse to do anything beyond the
2934 defaults, except that we allow the WILL ECHO option,
2935 which ICS uses to turn off password echoing when we are
2936 directly connected to it. We reject this option
2937 if localLineEditing mode is on (always on in xboard)
2938 and we are talking to port 23, which might be a real
2939 telnet server that will try to keep WILL ECHO on permanently.
2941 if (buf_len - i >= 3 && (unsigned char) buf[i] == TN_IAC) {
2942 static int remoteEchoOption = FALSE; /* telnet ECHO option */
2943 unsigned char option;
2945 switch ((unsigned char) buf[++i]) {
2947 if (appData.debugMode)
2948 fprintf(debugFP, "\n<WILL ");
2949 switch (option = (unsigned char) buf[++i]) {
2951 if (appData.debugMode)
2952 fprintf(debugFP, "ECHO ");
2953 /* Reply only if this is a change, according
2954 to the protocol rules. */
2955 if (remoteEchoOption) break;
2956 if (appData.localLineEditing &&
2957 atoi(appData.icsPort) == TN_PORT) {
2958 TelnetRequest(TN_DONT, TN_ECHO);
2961 TelnetRequest(TN_DO, TN_ECHO);
2962 remoteEchoOption = TRUE;
2966 if (appData.debugMode)
2967 fprintf(debugFP, "%d ", option);
2968 /* Whatever this is, we don't want it. */
2969 TelnetRequest(TN_DONT, option);
2974 if (appData.debugMode)
2975 fprintf(debugFP, "\n<WONT ");
2976 switch (option = (unsigned char) buf[++i]) {
2978 if (appData.debugMode)
2979 fprintf(debugFP, "ECHO ");
2980 /* Reply only if this is a change, according
2981 to the protocol rules. */
2982 if (!remoteEchoOption) break;
2984 TelnetRequest(TN_DONT, TN_ECHO);
2985 remoteEchoOption = FALSE;
2988 if (appData.debugMode)
2989 fprintf(debugFP, "%d ", (unsigned char) option);
2990 /* Whatever this is, it must already be turned
2991 off, because we never agree to turn on
2992 anything non-default, so according to the
2993 protocol rules, we don't reply. */
2998 if (appData.debugMode)
2999 fprintf(debugFP, "\n<DO ");
3000 switch (option = (unsigned char) buf[++i]) {
3002 /* Whatever this is, we refuse to do it. */
3003 if (appData.debugMode)
3004 fprintf(debugFP, "%d ", option);
3005 TelnetRequest(TN_WONT, option);
3010 if (appData.debugMode)
3011 fprintf(debugFP, "\n<DONT ");
3012 switch (option = (unsigned char) buf[++i]) {
3014 if (appData.debugMode)
3015 fprintf(debugFP, "%d ", option);
3016 /* Whatever this is, we are already not doing
3017 it, because we never agree to do anything
3018 non-default, so according to the protocol
3019 rules, we don't reply. */
3024 if (appData.debugMode)
3025 fprintf(debugFP, "\n<IAC ");
3026 /* Doubled IAC; pass it through */
3030 if (appData.debugMode)
3031 fprintf(debugFP, "\n<%d ", (unsigned char) buf[i]);
3032 /* Drop all other telnet commands on the floor */
3035 if (oldi > next_out)
3036 SendToPlayer(&buf[next_out], oldi - next_out);
3042 /* OK, this at least will *usually* work */
3043 if (!loggedOn && looking_at(buf, &i, "ics%")) {
3047 if (loggedOn && !intfSet) {
3048 if (ics_type == ICS_ICC) {
3049 snprintf(str, MSG_SIZ,
3050 "/set-quietly interface %s\n/set-quietly style 12\n",
3052 if(appData.seekGraph && appData.autoRefresh) // [HGM] seekgraph
3053 strcat(str, "/set-2 51 1\n/set seek 1\n");
3054 } else if (ics_type == ICS_CHESSNET) {
3055 snprintf(str, MSG_SIZ, "/style 12\n");
3057 safeStrCpy(str, "alias $ @\n$set interface ", sizeof(str)/sizeof(str[0]));
3058 strcat(str, programVersion);
3059 strcat(str, "\n$iset startpos 1\n$iset ms 1\n");
3060 if(appData.seekGraph && appData.autoRefresh) // [HGM] seekgraph
3061 strcat(str, "$iset seekremove 1\n$set seek 1\n");
3063 strcat(str, "$iset nohighlight 1\n");
3065 strcat(str, "$iset lock 1\n$style 12\n");
3068 NotifyFrontendLogin();
3072 if (started == STARTED_COMMENT) {
3073 /* Accumulate characters in comment */
3074 parse[parse_pos++] = buf[i];
3075 if (buf[i] == '\n') {
3076 parse[parse_pos] = NULLCHAR;
3077 if(chattingPartner>=0) {
3079 snprintf(mess, MSG_SIZ, "%s%s", talker, parse);
3080 OutputChatMessage(chattingPartner, mess);
3081 if(collective == 1) { // broadcasted talk also goes to private chatbox of talker
3083 talker[strlen(talker+1)-1] = NULLCHAR; // strip closing delimiter
3084 for(p=0; p<MAX_CHAT; p++) if(!StrCaseCmp(talker+1, chatPartner[p])) {
3085 snprintf(mess, MSG_SIZ, "%s: %s", chatPartner[chattingPartner], parse);
3086 OutputChatMessage(p, mess);
3090 chattingPartner = -1;
3091 if(collective != 3) next_out = i+1; // [HGM] suppress printing in ICS window
3094 if(!suppressKibitz) // [HGM] kibitz
3095 AppendComment(forwardMostMove, StripHighlight(parse), TRUE);
3096 else { // [HGM kibitz: divert memorized engine kibitz to engine-output window
3097 int nrDigit = 0, nrAlph = 0, j;
3098 if(parse_pos > MSG_SIZ - 30) // defuse unreasonably long input
3099 { parse_pos = MSG_SIZ-30; parse[parse_pos - 1] = '\n'; }
3100 parse[parse_pos] = NULLCHAR;
3101 // try to be smart: if it does not look like search info, it should go to
3102 // ICS interaction window after all, not to engine-output window.
3103 for(j=0; j<parse_pos; j++) { // count letters and digits
3104 nrDigit += (parse[j] >= '0' && parse[j] <= '9');
3105 nrAlph += (parse[j] >= 'a' && parse[j] <= 'z');
3106 nrAlph += (parse[j] >= 'A' && parse[j] <= 'Z');
3108 if(nrAlph < 9*nrDigit) { // if more than 10% digit we assume search info
3109 int depth=0; float score;
3110 if(sscanf(parse, "!!! %f/%d", &score, &depth) == 2 && depth>0) {
3111 // [HGM] kibitz: save kibitzed opponent info for PGN and eval graph
3112 pvInfoList[forwardMostMove-1].depth = depth;
3113 pvInfoList[forwardMostMove-1].score = 100*score;
3115 OutputKibitz(suppressKibitz, parse);
3118 if(gameMode == IcsObserving) // restore original ICS messages
3119 /* TRANSLATORS: to 'kibitz' is to send a message to all players and the game observers */
3120 snprintf(tmp, MSG_SIZ, "%s kibitzes: %s", star_match[0], parse);
3122 /* TRANSLATORS: to 'kibitz' is to send a message to all players and the game observers */
3123 snprintf(tmp, MSG_SIZ, _("your opponent kibitzes: %s"), parse);
3124 SendToPlayer(tmp, strlen(tmp));
3126 next_out = i+1; // [HGM] suppress printing in ICS window
3128 started = STARTED_NONE;
3130 /* Don't match patterns against characters in comment */
3135 if (started == STARTED_CHATTER) {
3136 if (buf[i] != '\n') {
3137 /* Don't match patterns against characters in chatter */
3141 started = STARTED_NONE;
3142 if(suppressKibitz) next_out = i+1;
3145 /* Kludge to deal with rcmd protocol */
3146 if (firstTime && looking_at(buf, &i, "\001*")) {
3147 DisplayFatalError(&buf[1], 0, 1);
3153 if (!loggedOn && looking_at(buf, &i, "chessclub.com")) {
3156 if (appData.debugMode)
3157 fprintf(debugFP, "ics_type %d\n", ics_type);
3160 if (!loggedOn && looking_at(buf, &i, "freechess.org")) {
3161 ics_type = ICS_FICS;
3163 if (appData.debugMode)
3164 fprintf(debugFP, "ics_type %d\n", ics_type);
3167 if (!loggedOn && looking_at(buf, &i, "chess.net")) {
3168 ics_type = ICS_CHESSNET;
3170 if (appData.debugMode)
3171 fprintf(debugFP, "ics_type %d\n", ics_type);
3176 (looking_at(buf, &i, "\"*\" is *a registered name") ||
3177 looking_at(buf, &i, "Logging you in as \"*\"") ||
3178 looking_at(buf, &i, "will be \"*\""))) {
3179 safeStrCpy(ics_handle, star_match[0], sizeof(ics_handle)/sizeof(ics_handle[0]));
3183 if (loggedOn && !have_set_title && ics_handle[0] != NULLCHAR) {
3185 snprintf(buf, sizeof(buf), "%s@%s", ics_handle, appData.icsHost);
3186 DisplayIcsInteractionTitle(buf);
3187 have_set_title = TRUE;
3190 /* skip finger notes */
3191 if (started == STARTED_NONE &&
3192 ((buf[i] == ' ' && isdigit(buf[i+1])) ||
3193 (buf[i] == '1' && buf[i+1] == '0')) &&
3194 buf[i+2] == ':' && buf[i+3] == ' ') {
3195 started = STARTED_CHATTER;
3201 // [HGM] seekgraph: recognize sought lines and end-of-sought message
3202 if(appData.seekGraph) {
3203 if(soughtPending && MatchSoughtLine(buf+i)) {
3204 i = strstr(buf+i, "rated") - buf;
3205 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3206 next_out = leftover_start = i;
3207 started = STARTED_CHATTER;
3208 suppressKibitz = TRUE;
3211 if((gameMode == IcsIdle || gameMode == BeginningOfGame)
3212 && looking_at(buf, &i, "* ads displayed")) {
3213 soughtPending = FALSE;
3218 if(appData.autoRefresh) {
3219 if(looking_at(buf, &i, "* (*) seeking * * * * *\"play *\" to respond)\n")) {
3220 int s = (ics_type == ICS_ICC); // ICC format differs
3222 AddAd(star_match[0], star_match[1], atoi(star_match[2+s]), atoi(star_match[3+s]),
3223 star_match[4+s][0], star_match[5-3*s], atoi(star_match[7]), TRUE);
3224 looking_at(buf, &i, "*% "); // eat prompt
3225 if(oldi > 0 && buf[oldi-1] == '\n') oldi--; // suppress preceding LF, if any
3226 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3227 next_out = i; // suppress
3230 if(looking_at(buf, &i, "\nAds removed: *\n") || looking_at(buf, &i, "\031(51 * *\031)")) {
3231 char *p = star_match[0];
3233 if(seekGraphUp) RemoveSeekAd(atoi(p));
3234 while(*p && *p++ != ' '); // next
3236 looking_at(buf, &i, "*% "); // eat prompt
3237 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3244 /* skip formula vars */
3245 if (started == STARTED_NONE &&
3246 buf[i] == 'f' && isdigit(buf[i+1]) && buf[i+2] == ':') {
3247 started = STARTED_CHATTER;
3252 // [HGM] kibitz: try to recognize opponent engine-score kibitzes, to divert them to engine-output window
3253 if (appData.autoKibitz && started == STARTED_NONE &&
3254 !appData.icsEngineAnalyze && // [HGM] [DM] ICS analyze
3255 (gameMode == IcsPlayingWhite || gameMode == IcsPlayingBlack || gameMode == IcsObserving)) {
3256 if((looking_at(buf, &i, "\n* kibitzes: ") || looking_at(buf, &i, "\n* whispers: ") ||
3257 looking_at(buf, &i, "* kibitzes: ") || looking_at(buf, &i, "* whispers: ")) &&
3258 (StrStr(star_match[0], gameInfo.white) == star_match[0] ||
3259 StrStr(star_match[0], gameInfo.black) == star_match[0] )) { // kibitz of self or opponent
3260 suppressKibitz = TRUE;
3261 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3263 if((StrStr(star_match[0], gameInfo.white) == star_match[0]
3264 && (gameMode == IcsPlayingWhite)) ||
3265 (StrStr(star_match[0], gameInfo.black) == star_match[0]
3266 && (gameMode == IcsPlayingBlack)) ) // opponent kibitz
3267 started = STARTED_CHATTER; // own kibitz we simply discard
3269 started = STARTED_COMMENT; // make sure it will be collected in parse[]
3270 parse_pos = 0; parse[0] = NULLCHAR;
3271 savingComment = TRUE;
3272 suppressKibitz = gameMode != IcsObserving ? 2 :
3273 (StrStr(star_match[0], gameInfo.white) == NULL) + 1;
3277 if((looking_at(buf, &i, "\nkibitzed to *\n") || looking_at(buf, &i, "kibitzed to *\n") ||
3278 looking_at(buf, &i, "\n(kibitzed to *\n") || looking_at(buf, &i, "(kibitzed to *\n"))
3279 && atoi(star_match[0])) {
3280 // suppress the acknowledgements of our own autoKibitz
3282 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3283 if(p = strchr(star_match[0], ' ')) p[1] = NULLCHAR; // clip off "players)" on FICS
3284 SendToPlayer(star_match[0], strlen(star_match[0]));
3285 if(looking_at(buf, &i, "*% ")) // eat prompt
3286 suppressKibitz = FALSE;
3290 } // [HGM] kibitz: end of patch
3292 if(looking_at(buf, &i, "* rating adjustment: * --> *\n")) continue;
3294 // [HGM] chat: intercept tells by users for which we have an open chat window
3296 if(started == STARTED_NONE && (looking_at(buf, &i, "* tells you:") || looking_at(buf, &i, "* says:") ||
3297 looking_at(buf, &i, "* whispers:") ||
3298 looking_at(buf, &i, "* kibitzes:") ||
3299 looking_at(buf, &i, "* shouts:") ||
3300 looking_at(buf, &i, "* c-shouts:") ||
3301 looking_at(buf, &i, "--> * ") ||
3302 looking_at(buf, &i, "*(*):") && (sscanf(star_match[1], "%d", &channel),1) ||
3303 looking_at(buf, &i, "*(*)(*):") && (sscanf(star_match[2], "%d", &channel),1) ||
3304 looking_at(buf, &i, "*(*)(*)(*):") && (sscanf(star_match[3], "%d", &channel),1) ||
3305 looking_at(buf, &i, "*(*)(*)(*)(*):") && sscanf(star_match[4], "%d", &channel) == 1 )) {
3307 sscanf(star_match[0], "%[^(]", talker+1); // strip (C) or (U) off ICS handle
3308 chattingPartner = -1; collective = 0;
3310 if(channel >= 0) // channel broadcast; look if there is a chatbox for this channel
3311 for(p=0; p<MAX_CHAT; p++) {
3313 if(chatPartner[p][0] >= '0' && chatPartner[p][0] <= '9' && channel == atoi(chatPartner[p])) {
3314 talker[0] = '['; strcat(talker, "] ");
3315 Colorize((channel == 1 ? ColorChannel1 : ColorChannel), FALSE);
3316 chattingPartner = p; break;
3319 if(buf[i-3] == 'e') // kibitz; look if there is a KIBITZ chatbox
3320 for(p=0; p<MAX_CHAT; p++) {
3322 if(!strcmp("kibitzes", chatPartner[p])) {
3323 talker[0] = '['; strcat(talker, "] ");
3324 chattingPartner = p; break;
3327 if(buf[i-3] == 'r') // whisper; look if there is a WHISPER chatbox
3328 for(p=0; p<MAX_CHAT; p++) {
3330 if(!strcmp("whispers", chatPartner[p])) {
3331 talker[0] = '['; strcat(talker, "] ");
3332 chattingPartner = p; break;
3335 if(buf[i-3] == 't' || buf[oldi+2] == '>') {// shout, c-shout or it; look if there is a 'shouts' chatbox
3336 if(buf[i-8] == '-' && buf[i-3] == 't')
3337 for(p=0; p<MAX_CHAT; p++) { // c-shout; check if dedicatesd c-shout box exists
3339 if(!strcmp("c-shouts", chatPartner[p])) {
3340 talker[0] = '('; strcat(talker, ") "); Colorize(ColorSShout, FALSE);
3341 chattingPartner = p; break;
3344 if(chattingPartner < 0)
3345 for(p=0; p<MAX_CHAT; p++) {
3347 if(!strcmp("shouts", chatPartner[p])) {
3348 if(buf[oldi+2] == '>') { talker[0] = '<'; strcat(talker, "> "); Colorize(ColorShout, FALSE); }
3349 else if(buf[i-8] == '-') { talker[0] = '('; strcat(talker, ") "); Colorize(ColorSShout, FALSE); }
3350 else { talker[0] = '['; strcat(talker, "] "); Colorize(ColorShout, FALSE); }
3351 chattingPartner = p; break;
3355 if(chattingPartner<0) // if not, look if there is a chatbox for this indivdual
3356 for(p=0; p<MAX_CHAT; p++) if(!StrCaseCmp(talker+1, chatPartner[p])) {
3358 Colorize(ColorTell, FALSE);
3359 if(collective) safeStrCpy(talker, "broadcasts: ", MSG_SIZ);
3361 chattingPartner = p; break;
3363 if(chattingPartner<0) i = oldi, safeStrCpy(lastTalker, talker+1, MSG_SIZ); else {
3364 Colorize(curColor, TRUE); // undo the bogus colorations we just made to trigger the souds
3365 started = STARTED_COMMENT;
3366 parse_pos = 0; parse[0] = NULLCHAR;
3367 savingComment = 3 + chattingPartner; // counts as TRUE
3368 if(collective == 3) i = oldi; else {
3369 suppressKibitz = TRUE;
3370 if(oldi > 0 && buf[oldi-1] == '\n') oldi--;
3371 if (oldi > next_out) SendToPlayer(&buf[next_out], oldi - next_out);
3375 } // [HGM] chat: end of patch
3378 if (appData.zippyTalk || appData.zippyPlay) {
3379 /* [DM] Backup address for color zippy lines */
3381 if (loggedOn == TRUE)
3382 if (ZippyControl(buf, &backup) || ZippyConverse(buf, &backup) ||
3383 (appData.zippyPlay && ZippyMatch(buf, &backup)));
3385 } // [DM] 'else { ' deleted
3387 /* Regular tells and says */
3388 (tkind = 1, looking_at(buf, &i, "* tells you: ")) ||
3389 looking_at(buf, &i, "* (your partner) tells you: ") ||
3390 looking_at(buf, &i, "* says: ") ||
3391 /* Don't color "message" or "messages" output */
3392 (tkind = 5, looking_at(buf, &i, "*. * (*:*): ")) ||
3393 looking_at(buf, &i, "*. * at *:*: ") ||
3394 looking_at(buf, &i, "--* (*:*): ") ||
3395 /* Message notifications (same color as tells) */
3396 looking_at(buf, &i, "* has left a message ") ||
3397 looking_at(buf, &i, "* just sent you a message:\n") ||
3398 /* Whispers and kibitzes */
3399 (tkind = 2, looking_at(buf, &i, "* whispers: ")) ||
3400 looking_at(buf, &i, "* kibitzes: ") ||
3402 (tkind = 3, looking_at(buf, &i, "*(*: "))) {
3404 if (tkind == 1 && strchr(star_match[0], ':')) {
3405 /* Avoid "tells you:" spoofs in channels */
3408 if (star_match[0][0] == NULLCHAR ||
3409 strchr(star_match[0], ' ') ||
3410 (tkind == 3 && strchr(star_match[1], ' '))) {
3411 /* Reject bogus matches */
3414 if (appData.colorize) {
3415 if (oldi > next_out) {
3416 SendToPlayer(&buf[next_out], oldi - next_out);
3421 Colorize(ColorTell, FALSE);
3422 curColor = ColorTell;
3425 Colorize(ColorKibitz, FALSE);
3426 curColor = ColorKibitz;
3429 p = strrchr(star_match[1], '(');
3436 Colorize(ColorChannel1, FALSE);
3437 curColor = ColorChannel1;
3439 Colorize(ColorChannel, FALSE);
3440 curColor = ColorChannel;
3444 curColor = ColorNormal;
3448 if (started == STARTED_NONE && appData.autoComment &&
3449 (gameMode == IcsObserving ||
3450 gameMode == IcsPlayingWhite ||
3451 gameMode == IcsPlayingBlack)) {
3452 parse_pos = i - oldi;
3453 memcpy(parse, &buf[oldi], parse_pos);
3454 parse[parse_pos] = NULLCHAR;
3455 started = STARTED_COMMENT;
3456 savingComment = TRUE;
3457 } else if(collective != 3) {
3458 started = STARTED_CHATTER;
3459 savingComment = FALSE;
3466 if (looking_at(buf, &i, "* s-shouts: ") ||
3467 looking_at(buf, &i, "* c-shouts: ")) {
3468 if (appData.colorize) {
3469 if (oldi > next_out) {
3470 SendToPlayer(&buf[next_out], oldi - next_out);
3473 Colorize(ColorSShout, FALSE);
3474 curColor = ColorSShout;
3477 started = STARTED_CHATTER;
3481 if (looking_at(buf, &i, "--->")) {
3486 if (looking_at(buf, &i, "* shouts: ") ||
3487 looking_at(buf, &i, "--> ")) {
3488 if (appData.colorize) {
3489 if (oldi > next_out) {
3490 SendToPlayer(&buf[next_out], oldi - next_out);
3493 Colorize(ColorShout, FALSE);
3494 curColor = ColorShout;
3497 started = STARTED_CHATTER;
3501 if (looking_at( buf, &i, "Challenge:")) {
3502 if (appData.colorize) {
3503 if (oldi > next_out) {
3504 SendToPlayer(&buf[next_out], oldi - next_out);
3507 Colorize(ColorChallenge, FALSE);
3508 curColor = ColorChallenge;
3514 if (looking_at(buf, &i, "* offers you") ||
3515 looking_at(buf, &i, "* offers to be") ||
3516 looking_at(buf, &i, "* would like to") ||
3517 looking_at(buf, &i, "* requests to") ||
3518 looking_at(buf, &i, "Your opponent offers") ||
3519 looking_at(buf, &i, "Your opponent requests")) {
3521 if (appData.colorize) {
3522 if (oldi > next_out) {
3523 SendToPlayer(&buf[next_out], oldi - next_out);
3526 Colorize(ColorRequest, FALSE);
3527 curColor = ColorRequest;
3532 if (looking_at(buf, &i, "* (*) seeking")) {
3533 if (appData.colorize) {
3534 if (oldi > next_out) {
3535 SendToPlayer(&buf[next_out], oldi - next_out);
3538 Colorize(ColorSeek, FALSE);
3539 curColor = ColorSeek;
3544 if(i < backup) { i = backup; continue; } // [HGM] for if ZippyControl matches, but the colorie code doesn't
3546 if (looking_at(buf, &i, "\\ ")) {
3547 if (prevColor != ColorNormal) {
3548 if (oldi > next_out) {
3549 SendToPlayer(&buf[next_out], oldi - next_out);
3552 Colorize(prevColor, TRUE);
3553 curColor = prevColor;
3555 if (savingComment) {
3556 parse_pos = i - oldi;
3557 memcpy(parse, &buf[oldi], parse_pos);
3558 parse[parse_pos] = NULLCHAR;
3559 started = STARTED_COMMENT;
3560 if(savingComment >= 3) // [HGM] chat: continuation of line for chat box
3561 chattingPartner = savingComment - 3; // kludge to remember the box
3563 started = STARTED_CHATTER;
3568 if (looking_at(buf, &i, "Black Strength :") ||
3569 looking_at(buf, &i, "<<< style 10 board >>>") ||
3570 looking_at(buf, &i, "<10>") ||
3571 looking_at(buf, &i, "#@#")) {
3572 /* Wrong board style */
3574 SendToICS(ics_prefix);
3575 SendToICS("set style 12\n");
3576 SendToICS(ics_prefix);
3577 SendToICS("refresh\n");
3581 if (looking_at(buf, &i, "login:")) {
3582 if (!have_sent_ICS_logon) {
3584 have_sent_ICS_logon = 1;
3585 else // no init script was found
3586 have_sent_ICS_logon = (appData.autoCreateLogon ? 2 : 1); // flag that we should capture username + password
3587 } else { // we have sent (or created) the InitScript, but apparently the ICS rejected it
3588 have_sent_ICS_logon = (appData.autoCreateLogon ? 2 : 1); // request creation of a new script
3593 if (ics_getting_history != H_GETTING_MOVES /*smpos kludge*/ &&
3594 (looking_at(buf, &i, "\n<12> ") ||
3595 looking_at(buf, &i, "<12> "))) {
3597 if (oldi > next_out) {
3598 SendToPlayer(&buf[next_out], oldi - next_out);
3601 started = STARTED_BOARD;
3606 if ((started == STARTED_NONE && looking_at(buf, &i, "\n<b1> ")) ||
3607 looking_at(buf, &i, "<b1> ")) {
3608 if (oldi > next_out) {
3609 SendToPlayer(&buf[next_out], oldi - next_out);
3612 started = STARTED_HOLDINGS;
3617 if (looking_at(buf, &i, "* *vs. * *--- *")) {
3619 /* Header for a move list -- first line */
3621 switch (ics_getting_history) {
3625 case BeginningOfGame:
3626 /* User typed "moves" or "oldmoves" while we
3627 were idle. Pretend we asked for these
3628 moves and soak them up so user can step
3629 through them and/or save them.
3632 gameMode = IcsObserving;
3635 ics_getting_history = H_GOT_UNREQ_HEADER;
3637 case EditGame: /*?*/
3638 case EditPosition: /*?*/
3639 /* Should above feature work in these modes too? */
3640 /* For now it doesn't */
3641 ics_getting_history = H_GOT_UNWANTED_HEADER;
3644 ics_getting_history = H_GOT_UNWANTED_HEADER;
3649 /* Is this the right one? */
3650 if (gameInfo.white && gameInfo.black &&
3651 strcmp(gameInfo.white, star_match[0]) == 0 &&
3652 strcmp(gameInfo.black, star_match[2]) == 0) {
3654 ics_getting_history = H_GOT_REQ_HEADER;
3657 case H_GOT_REQ_HEADER:
3658 case H_GOT_UNREQ_HEADER:
3659 case H_GOT_UNWANTED_HEADER:
3660 case H_GETTING_MOVES:
3661 /* Should not happen */
3662 DisplayError(_("Error gathering move list: two headers"), 0);
3663 ics_getting_history = H_FALSE;
3667 /* Save player ratings into gameInfo if needed */
3668 if ((ics_getting_history == H_GOT_REQ_HEADER ||
3669 ics_getting_history == H_GOT_UNREQ_HEADER) &&
3670 (gameInfo.whiteRating == -1 ||
3671 gameInfo.blackRating == -1)) {
3673 gameInfo.whiteRating = string_to_rating(star_match[1]);
3674 gameInfo.blackRating = string_to_rating(star_match[3]);
3675 if (appData.debugMode)
3676 fprintf(debugFP, "Ratings from header: W %d, B %d\n",
3677 gameInfo.whiteRating, gameInfo.blackRating);
3682 if (looking_at(buf, &i,
3683 "* * match, initial time: * minute*, increment: * second")) {
3684 /* Header for a move list -- second line */
3685 /* Initial board will follow if this is a wild game */
3686 if (gameInfo.event != NULL) free(gameInfo.event);
3687 snprintf(str, MSG_SIZ, "ICS %s %s match", star_match[0], star_match[1]);
3688 gameInfo.event = StrSave(str);
3689 /* [HGM] we switched variant. Translate boards if needed. */
3690 VariantSwitch(boards[currentMove], StringToVariant(gameInfo.event));
3694 if (looking_at(buf, &i, "Move ")) {
3695 /* Beginning of a move list */
3696 switch (ics_getting_history) {
3698 /* Normally should not happen */
3699 /* Maybe user hit reset while we were parsing */
3702 /* Happens if we are ignoring a move list that is not
3703 * the one we just requested. Common if the user
3704 * tries to observe two games without turning off
3707 case H_GETTING_MOVES:
3708 /* Should not happen */
3709 DisplayError(_("Error gathering move list: nested"), 0);
3710 ics_getting_history = H_FALSE;
3712 case H_GOT_REQ_HEADER:
3713 ics_getting_history = H_GETTING_MOVES;
3714 started = STARTED_MOVES;
3716 if (oldi > next_out) {
3717 SendToPlayer(&buf[next_out], oldi - next_out);
3720 case H_GOT_UNREQ_HEADER:
3721 ics_getting_history = H_GETTING_MOVES;
3722 started = STARTED_MOVES_NOHIDE;
3725 case H_GOT_UNWANTED_HEADER:
3726 ics_getting_history = H_FALSE;
3732 if (looking_at(buf, &i, "% ") ||
3733 ((started == STARTED_MOVES || started == STARTED_MOVES_NOHIDE)
3734 && looking_at(buf, &i, "}*"))) { char *bookHit = NULL; // [HGM] book
3735 if(soughtPending && nrOfSeekAds) { // [HGM] seekgraph: on ICC sought-list has no termination line
3736 soughtPending = FALSE;
3740 if(suppressKibitz) next_out = i;
3741 savingComment = FALSE;
3745 case STARTED_MOVES_NOHIDE:
3746 memcpy(&parse[parse_pos], &buf[oldi], i - oldi);
3747 parse[parse_pos + i - oldi] = NULLCHAR;
3748 ParseGameHistory(parse);
3750 if (appData.zippyPlay && first.initDone) {
3751 FeedMovesToProgram(&first, forwardMostMove);
3752 if (gameMode == IcsPlayingWhite) {
3753 if (WhiteOnMove(forwardMostMove)) {
3754 if (first.sendTime) {
3755 if (first.useColors) {
3756 SendToProgram("black\n", &first);
3758 SendTimeRemaining(&first, TRUE);
3760 if (first.useColors) {
3761 SendToProgram("white\n", &first); // [HGM] book: made sending of "go\n" book dependent
3763 bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE); // [HGM] book: probe book for initial pos
3764 first.maybeThinking = TRUE;
3766 if (first.usePlayother) {
3767 if (first.sendTime) {
3768 SendTimeRemaining(&first, TRUE);
3770 SendToProgram("playother\n", &first);
3776 } else if (gameMode == IcsPlayingBlack) {
3777 if (!WhiteOnMove(forwardMostMove)) {
3778 if (first.sendTime) {
3779 if (first.useColors) {
3780 SendToProgram("white\n", &first);
3782 SendTimeRemaining(&first, FALSE);
3784 if (first.useColors) {
3785 SendToProgram("black\n", &first);
3787 bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE);
3788 first.maybeThinking = TRUE;
3790 if (first.usePlayother) {
3791 if (first.sendTime) {
3792 SendTimeRemaining(&first, FALSE);
3794 SendToProgram("playother\n", &first);
3803 if (gameMode == IcsObserving && ics_gamenum == -1) {
3804 /* Moves came from oldmoves or moves command
3805 while we weren't doing anything else.
3807 currentMove = forwardMostMove;
3808 ClearHighlights();/*!!could figure this out*/
3809 flipView = appData.flipView;
3810 DrawPosition(TRUE, boards[currentMove]);
3811 DisplayBothClocks();
3812 snprintf(str, MSG_SIZ, "%s %s %s",
3813 gameInfo.white, _("vs."), gameInfo.black);
3817 /* Moves were history of an active game */
3818 if (gameInfo.resultDetails != NULL) {
3819 free(gameInfo.resultDetails);
3820 gameInfo.resultDetails = NULL;
3823 HistorySet(parseList, backwardMostMove,
3824 forwardMostMove, currentMove-1);
3825 DisplayMove(currentMove - 1);
3826 if (started == STARTED_MOVES) next_out = i;
3827 started = STARTED_NONE;
3828 ics_getting_history = H_FALSE;
3831 case STARTED_OBSERVE:
3832 started = STARTED_NONE;
3833 SendToICS(ics_prefix);
3834 SendToICS("refresh\n");
3840 if(bookHit) { // [HGM] book: simulate book reply
3841 static char bookMove[MSG_SIZ]; // a bit generous?
3843 programStats.nodes = programStats.depth = programStats.time =
3844 programStats.score = programStats.got_only_move = 0;
3845 sprintf(programStats.movelist, "%s (xbook)", bookHit);
3847 safeStrCpy(bookMove, "move ", sizeof(bookMove)/sizeof(bookMove[0]));
3848 strcat(bookMove, bookHit);
3849 HandleMachineMove(bookMove, &first);
3854 if ((started == STARTED_MOVES || started == STARTED_BOARD ||
3855 started == STARTED_HOLDINGS ||
3856 started == STARTED_MOVES_NOHIDE) && i >= leftover_len) {
3857 /* Accumulate characters in move list or board */
3858 parse[parse_pos++] = buf[i];
3861 /* Start of game messages. Mostly we detect start of game
3862 when the first board image arrives. On some versions
3863 of the ICS, though, we need to do a "refresh" after starting
3864 to observe in order to get the current board right away. */
3865 if (looking_at(buf, &i, "Adding game * to observation list")) {
3866 started = STARTED_OBSERVE;
3870 /* Handle auto-observe */
3871 if (appData.autoObserve &&
3872 (gameMode == IcsIdle || gameMode == BeginningOfGame) &&
3873 looking_at(buf, &i, "Game notification: * (*) vs. * (*)")) {
3875 /* Choose the player that was highlighted, if any. */
3876 if (star_match[0][0] == '\033' ||
3877 star_match[1][0] != '\033') {
3878 player = star_match[0];
3880 player = star_match[2];
3882 snprintf(str, MSG_SIZ, "%sobserve %s\n",
3883 ics_prefix, StripHighlightAndTitle(player));
3886 /* Save ratings from notify string */
3887 safeStrCpy(player1Name, star_match[0], sizeof(player1Name)/sizeof(player1Name[0]));
3888 player1Rating = string_to_rating(star_match[1]);
3889 safeStrCpy(player2Name, star_match[2], sizeof(player2Name)/sizeof(player2Name[0]));
3890 player2Rating = string_to_rating(star_match[3]);