2011-10-19 18:34:26 -07:00
|
|
|
/* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */
|
2008-06-30 03:39:27 +00:00
|
|
|
/*
|
2011-10-19 18:34:26 -07:00
|
|
|
* Copyright 2000-2011 by Eric House (xwords@eehouse.org). All rights
|
2008-06-30 03:39:27 +00:00
|
|
|
* reserved.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2011-04-10 13:04:05 -07:00
|
|
|
#include <stdbool.h>
|
2011-04-13 06:45:22 -07:00
|
|
|
#include <ctype.h> /* BAD: use glib to support utf8 */
|
2011-11-28 18:12:56 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
# include <execinfo.h> /* for backtrace */
|
|
|
|
#endif
|
2008-06-30 03:39:27 +00:00
|
|
|
|
2014-08-26 06:55:26 -07:00
|
|
|
#include "movestak.h"
|
2008-06-30 03:39:27 +00:00
|
|
|
#include "linuxutl.h"
|
2011-04-10 13:04:05 -07:00
|
|
|
#include "main.h"
|
2013-01-07 20:40:46 -08:00
|
|
|
#include "linuxdict.h"
|
2013-01-29 07:42:10 -08:00
|
|
|
#include "linuxmain.h"
|
2013-09-14 21:06:14 -07:00
|
|
|
#include "gamesdb.h"
|
2008-06-30 03:39:27 +00:00
|
|
|
#include "LocalizedStrIncludes.h"
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
void
|
|
|
|
linux_debugf( const char* format, ... )
|
|
|
|
{
|
2017-10-19 21:20:14 -07:00
|
|
|
char buf[1024*8];
|
2008-06-30 03:39:27 +00:00
|
|
|
va_list ap;
|
|
|
|
struct tm* timp;
|
|
|
|
struct timeval tv;
|
|
|
|
struct timezone tz;
|
|
|
|
|
|
|
|
gettimeofday( &tv, &tz );
|
|
|
|
timp = localtime( &tv.tv_sec );
|
|
|
|
|
2017-10-28 16:13:11 -07:00
|
|
|
size_t len = snprintf( buf, sizeof(buf), "<%d:%lx>%.2d:%.2d:%.2d:", getpid(),
|
2017-10-27 05:57:40 -07:00
|
|
|
pthread_self(), timp->tm_hour, timp->tm_min, timp->tm_sec );
|
2017-10-19 21:20:14 -07:00
|
|
|
XP_ASSERT( len < sizeof(buf) );
|
2008-06-30 03:39:27 +00:00
|
|
|
|
|
|
|
va_start(ap, format);
|
2017-10-19 21:20:14 -07:00
|
|
|
len = vsprintf(buf+strlen(buf), format, ap);
|
2008-06-30 03:39:27 +00:00
|
|
|
va_end(ap);
|
2017-10-19 21:20:14 -07:00
|
|
|
|
|
|
|
if ( len >= sizeof(buf) ) {
|
|
|
|
buf[sizeof(buf)-1] = '\0';
|
|
|
|
}
|
2008-06-30 03:39:27 +00:00
|
|
|
|
2008-11-06 03:16:09 +00:00
|
|
|
fprintf( stderr, "%s\n", buf );
|
2008-06-30 03:39:27 +00:00
|
|
|
}
|
2011-11-28 18:12:56 -08:00
|
|
|
|
|
|
|
void
|
|
|
|
linux_backtrace( void )
|
|
|
|
{
|
2019-01-28 16:31:41 -08:00
|
|
|
VDECL( void*, buffer, 128 );
|
2011-11-28 18:12:56 -08:00
|
|
|
int nFound = backtrace( buffer, VSIZE(buffer) );
|
|
|
|
XP_ASSERT( nFound < VSIZE(buffer) );
|
|
|
|
char** traces = backtrace_symbols( buffer, nFound );
|
|
|
|
|
|
|
|
XP_U16 ii;
|
|
|
|
for ( ii = 0; ii < nFound; ++ii ) {
|
|
|
|
XP_LOGF( "trace[%.2d]: %s", ii, traces[ii] );
|
|
|
|
}
|
|
|
|
free( traces );
|
|
|
|
}
|
2008-06-30 03:39:27 +00:00
|
|
|
#endif
|
|
|
|
|
2011-04-07 18:07:45 -07:00
|
|
|
#ifndef MEM_DEBUG
|
|
|
|
void
|
|
|
|
linux_freep( void** ptrp )
|
|
|
|
{
|
|
|
|
if ( !!*ptrp ) {
|
|
|
|
free( *ptrp );
|
|
|
|
*ptrp = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
static DictionaryCtxt*
|
2009-09-20 21:51:29 +00:00
|
|
|
linux_util_makeEmptyDict( XW_UtilCtxt* XP_UNUSED_DBG(uctx) )
|
2008-06-30 03:39:27 +00:00
|
|
|
{
|
2009-01-05 01:59:04 +00:00
|
|
|
XP_DEBUGF( "linux_util_makeEmptyDict called" );
|
2012-09-03 21:33:46 -07:00
|
|
|
return linux_dictionary_make( MPPARM(uctx->mpool) NULL, NULL, XP_FALSE );
|
2008-06-30 03:39:27 +00:00
|
|
|
} /* linux_util_makeEmptyDict */
|
|
|
|
|
|
|
|
#define EM BONUS_NONE
|
|
|
|
#define DL BONUS_DOUBLE_LETTER
|
|
|
|
#define DW BONUS_DOUBLE_WORD
|
|
|
|
#define TL BONUS_TRIPLE_LETTER
|
|
|
|
#define TW BONUS_TRIPLE_WORD
|
|
|
|
|
2019-01-28 16:31:41 -08:00
|
|
|
#define SC_BOARD_SIZE 36
|
|
|
|
#define SEVENTEEN_BOARD_SIZE 45
|
|
|
|
|
2011-11-16 19:01:11 -08:00
|
|
|
static XWBonusType*
|
|
|
|
bonusesFor( XP_U16 boardSize, XP_U16* len )
|
|
|
|
{
|
2019-01-28 16:31:41 -08:00
|
|
|
static XWBonusType scrabbleBoard[SC_BOARD_SIZE] = {
|
2011-11-16 19:01:11 -08:00
|
|
|
TW,//EM,EM,DL,EM,EM,EM,TW,
|
|
|
|
EM,DW,//EM,EM,EM,TL,EM,EM,
|
|
|
|
|
|
|
|
EM,EM,DW,//EM,EM,EM,DL,EM,
|
|
|
|
DL,EM,EM,DW,//EM,EM,EM,DL,
|
|
|
|
|
|
|
|
EM,EM,EM,EM,DW,//EM,EM,EM,
|
|
|
|
EM,TL,EM,EM,EM,TL,//EM,EM,
|
|
|
|
|
|
|
|
EM,EM,DL,EM,EM,EM,DL,//EM,
|
|
|
|
TW,EM,EM,DL,EM,EM,EM,DW,
|
|
|
|
}; /* scrabbleBoard */
|
|
|
|
|
2019-01-28 16:31:41 -08:00
|
|
|
static XWBonusType seventeen[SEVENTEEN_BOARD_SIZE] = {
|
2011-11-16 19:01:11 -08:00
|
|
|
TW,//EM,EM,DL,EM,EM,EM,TW,
|
|
|
|
EM,DW,//EM,EM,EM,TL,EM,EM,
|
|
|
|
|
|
|
|
EM,EM,DW,//EM,EM,EM,DL,EM,
|
|
|
|
DL,EM,EM,DW,//EM,EM,EM,DL,
|
|
|
|
|
|
|
|
EM,EM,EM,EM,DW,//EM,EM,EM,
|
|
|
|
EM,TL,EM,EM,EM,TL,//EM,EM,
|
|
|
|
|
|
|
|
EM,EM,DL,EM,EM,EM,DL,//EM,
|
|
|
|
TW,EM,EM,DL,EM,EM,EM,DW,
|
|
|
|
TW,EM,EM,DL,EM,EM,EM,DW,DW,
|
|
|
|
}; /* scrabbleBoard */
|
|
|
|
|
|
|
|
XWBonusType* result = NULL;
|
|
|
|
if ( boardSize == 15 ) {
|
|
|
|
result = scrabbleBoard;
|
2019-01-28 16:31:41 -08:00
|
|
|
*len = SC_BOARD_SIZE;
|
2011-11-16 19:01:11 -08:00
|
|
|
} else if ( boardSize == 17 ) {
|
|
|
|
result = seventeen;
|
2019-01-28 16:31:41 -08:00
|
|
|
*len = SEVENTEEN_BOARD_SIZE;
|
2011-11-16 19:01:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef STREAM_VERS_BIGBOARD
|
|
|
|
void
|
|
|
|
setSquareBonuses( const CommonGlobals* cGlobals )
|
|
|
|
{
|
|
|
|
XP_U16 nBonuses;
|
|
|
|
XWBonusType* bonuses =
|
2013-07-09 18:10:41 -07:00
|
|
|
bonusesFor( cGlobals->gi->boardSize, &nBonuses );
|
2011-11-16 19:01:11 -08:00
|
|
|
if ( !!bonuses ) {
|
|
|
|
model_setSquareBonuses( cGlobals->game.model, bonuses, nBonuses );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-04-10 13:04:05 -07:00
|
|
|
static XWBonusType*
|
|
|
|
parseBonusFile( XP_U16 nCols, const char* bonusFile )
|
|
|
|
{
|
|
|
|
XWBonusType* result = NULL;
|
|
|
|
FILE* file = fopen( bonusFile, "r" );
|
|
|
|
if ( !!file ) {
|
|
|
|
XP_U16 row = 0;
|
|
|
|
XP_U16 col;
|
|
|
|
result = malloc( nCols * nCols * sizeof(*result) );
|
|
|
|
char line[1024];
|
|
|
|
while ( line == fgets( line, sizeof(line), file ) && row < nCols ) {
|
|
|
|
bool inComment = false;
|
|
|
|
char* ch;
|
|
|
|
XWBonusType bonus;
|
|
|
|
col = 0;
|
|
|
|
for ( ch = line; '\0' != *ch; ++ch ) {
|
|
|
|
switch( *ch ) {
|
|
|
|
case '#': /* comment; line is done */
|
|
|
|
inComment = true;
|
|
|
|
break;
|
|
|
|
case '+':
|
|
|
|
bonus = BONUS_DOUBLE_LETTER;
|
|
|
|
break;
|
|
|
|
case '*':
|
|
|
|
bonus = BONUS_DOUBLE_WORD;
|
|
|
|
break;
|
|
|
|
case '^':
|
|
|
|
bonus = BONUS_TRIPLE_LETTER;
|
|
|
|
break;
|
|
|
|
case '!':
|
|
|
|
bonus = BONUS_TRIPLE_WORD;
|
|
|
|
break;
|
|
|
|
case '.':
|
|
|
|
case ' ':
|
|
|
|
bonus = BONUS_NONE;
|
|
|
|
break;
|
|
|
|
case '\n':
|
|
|
|
case '\a':
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if ( !inComment ) {
|
|
|
|
fprintf( stderr, "unexpected char '%c' in %s\n", *ch, bonusFile );
|
|
|
|
exit( 1 );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ( !inComment && col < nCols ) {
|
|
|
|
result[(row * nCols) + col] = bonus;
|
|
|
|
++col;
|
2011-04-10 13:13:45 -07:00
|
|
|
/* Let's just allow anything to follow the 15 letters we
|
|
|
|
care about, e.g. comments */
|
|
|
|
if ( col >= nCols ) {
|
|
|
|
inComment = true;
|
|
|
|
}
|
2011-04-10 13:04:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( col > 0 && row < nCols - 1) {
|
|
|
|
++row;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose( file );
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
static XWBonusType
|
2011-11-16 19:01:11 -08:00
|
|
|
linux_util_getSquareBonus( XW_UtilCtxt* uc, XP_U16 nCols,
|
2008-06-30 03:39:27 +00:00
|
|
|
XP_U16 col, XP_U16 row )
|
|
|
|
{
|
2011-04-10 13:04:05 -07:00
|
|
|
static XWBonusType* parsedFile = NULL;
|
2011-11-08 21:11:06 -08:00
|
|
|
XWBonusType result = EM;
|
|
|
|
|
2011-04-10 13:04:05 -07:00
|
|
|
CommonGlobals* cGlobals = (CommonGlobals*)uc->closure;
|
|
|
|
if ( NULL == parsedFile && NULL != cGlobals->params->bonusFile ) {
|
|
|
|
if ( !parsedFile ) {
|
|
|
|
parsedFile = parseBonusFile( nCols, cGlobals->params->bonusFile );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( NULL != parsedFile ) {
|
2011-11-08 21:11:06 -08:00
|
|
|
result = parsedFile[(row*nCols) + col];
|
2011-04-10 13:04:05 -07:00
|
|
|
} else {
|
2011-11-16 19:01:11 -08:00
|
|
|
XP_U16 nEntries;
|
|
|
|
XWBonusType* scrabbleBoard = bonusesFor( 15, &nEntries );
|
2011-04-10 13:04:05 -07:00
|
|
|
|
2011-11-16 19:01:11 -08:00
|
|
|
XP_U16 index, ii;
|
2011-11-14 18:21:41 -08:00
|
|
|
if ( col > (nCols/2) ) col = nCols - 1 - col;
|
|
|
|
if ( row > (nCols/2) ) row = nCols - 1 - row;
|
2011-11-08 21:11:06 -08:00
|
|
|
if ( col > row ) {
|
|
|
|
XP_U16 tmp = col;
|
|
|
|
col = row;
|
|
|
|
row = tmp;
|
|
|
|
}
|
|
|
|
index = col;
|
|
|
|
for ( ii = 1; ii <= row; ++ii ) {
|
|
|
|
index += ii;
|
|
|
|
}
|
|
|
|
|
2011-11-16 19:01:11 -08:00
|
|
|
if ( index < nEntries) {
|
|
|
|
result = scrabbleBoard[index];
|
2011-04-10 13:04:05 -07:00
|
|
|
}
|
2008-06-30 03:39:27 +00:00
|
|
|
}
|
2011-11-08 21:11:06 -08:00
|
|
|
return result;
|
2008-06-30 03:39:27 +00:00
|
|
|
} /* linux_util_getSquareBonus */
|
|
|
|
|
2018-07-05 07:58:50 -07:00
|
|
|
static XW_DUtilCtxt*
|
|
|
|
linux_util_getDevUtilCtxt( XW_UtilCtxt* uc )
|
2012-10-30 07:01:47 -07:00
|
|
|
{
|
|
|
|
CommonGlobals* cGlobals = (CommonGlobals*)uc->closure;
|
2018-07-05 07:58:50 -07:00
|
|
|
return cGlobals->params->dutil;
|
2012-11-03 10:58:01 -07:00
|
|
|
}
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
void
|
|
|
|
linux_util_vt_init( MPFORMAL XW_UtilCtxt* util )
|
|
|
|
{
|
2013-01-07 06:10:44 -08:00
|
|
|
#ifdef MEM_DEBUG
|
|
|
|
util->mpool = mpool;
|
|
|
|
#endif
|
2008-06-30 03:39:27 +00:00
|
|
|
util->vtable = XP_MALLOC( mpool, sizeof(UtilVtable) );
|
|
|
|
|
|
|
|
util->vtable->m_util_makeEmptyDict = linux_util_makeEmptyDict;
|
|
|
|
util->vtable->m_util_getSquareBonus = linux_util_getSquareBonus;
|
2018-07-05 07:58:50 -07:00
|
|
|
|
|
|
|
util->vtable->m_util_getDevUtilCtxt = linux_util_getDevUtilCtxt;
|
2008-06-30 03:39:27 +00:00
|
|
|
}
|
|
|
|
|
2008-08-30 13:27:27 +00:00
|
|
|
void
|
|
|
|
linux_util_vt_destroy( XW_UtilCtxt* util )
|
|
|
|
{
|
|
|
|
XP_FREE( util->mpool, util->vtable );
|
|
|
|
}
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
const XP_UCHAR*
|
|
|
|
linux_getErrString( UtilErrID id, XP_Bool* silent )
|
|
|
|
{
|
|
|
|
*silent = XP_FALSE;
|
|
|
|
const char* message = NULL;
|
|
|
|
|
2012-01-11 18:43:51 -08:00
|
|
|
switch( (int)id ) {
|
2008-06-30 03:39:27 +00:00
|
|
|
case ERR_TILES_NOT_IN_LINE:
|
|
|
|
message = "All tiles played must be in a line.";
|
|
|
|
break;
|
|
|
|
case ERR_NO_EMPTIES_IN_TURN:
|
|
|
|
message = "Empty squares cannot separate tiles played.";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_TOO_FEW_TILES_LEFT_TO_TRADE:
|
|
|
|
message = "Too few tiles left to trade.";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_TWO_TILES_FIRST_MOVE:
|
|
|
|
message = "Must play two or more pieces on the first move.";
|
|
|
|
break;
|
|
|
|
case ERR_TILES_MUST_CONTACT:
|
|
|
|
message = "New pieces must contact others already in place (or "
|
|
|
|
"the middle square on the first move).";
|
|
|
|
break;
|
|
|
|
case ERR_NOT_YOUR_TURN:
|
|
|
|
message = "You can't do that; it's not your turn!";
|
|
|
|
break;
|
|
|
|
case ERR_NO_PEEK_ROBOT_TILES:
|
|
|
|
message = "No peeking at the robot's tiles!";
|
|
|
|
break;
|
|
|
|
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
|
|
case ERR_NO_PEEK_REMOTE_TILES:
|
|
|
|
message = "No peeking at remote players' tiles!";
|
|
|
|
break;
|
|
|
|
case ERR_REG_UNEXPECTED_USER:
|
|
|
|
message = "Refused attempt to register unexpected user[s].";
|
|
|
|
break;
|
|
|
|
case ERR_SERVER_DICT_WINS:
|
|
|
|
message = "Conflict between Host and Guest dictionaries; Host wins.";
|
|
|
|
XP_WARNF( "GTK may have problems here." );
|
|
|
|
break;
|
|
|
|
case ERR_REG_SERVER_SANS_REMOTE:
|
|
|
|
message = "At least one player must be marked remote for a game "
|
|
|
|
"started as Host.";
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2011-03-07 06:29:40 -08:00
|
|
|
case ERR_NO_EMPTY_TRADE:
|
|
|
|
message = "No tiles selected; trade cancelled.";
|
|
|
|
break;
|
2018-12-29 12:15:28 -08:00
|
|
|
|
|
|
|
case ERR_TOO_MANY_TRADE:
|
|
|
|
message = "More tiles selected than remain in pool.";
|
|
|
|
break;
|
2011-03-07 06:29:40 -08:00
|
|
|
|
2015-07-11 16:24:21 -07:00
|
|
|
case ERR_NO_HINT_FOUND:
|
|
|
|
message = "Unable to suggest any moves.";
|
|
|
|
break;
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
case ERR_CANT_UNDO_TILEASSIGN:
|
|
|
|
message = "Tile assignment can't be undone.";
|
|
|
|
break;
|
|
|
|
|
2008-08-30 13:27:27 +00:00
|
|
|
case ERR_CANT_HINT_WHILE_DISABLED:
|
|
|
|
message = "The hint feature is disabled for this game. Enable "
|
|
|
|
"it for a new game using the Preferences dialog.";
|
|
|
|
break;
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
/* case INFO_REMOTE_CONNECTED: */
|
|
|
|
/* message = "Another device has joined the game"; */
|
|
|
|
/* break; */
|
|
|
|
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_LOST_OTHER:
|
|
|
|
*silent = XP_TRUE;
|
|
|
|
message = "XWRELAY_ERROR_LOST_OTHER";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_TIMEOUT:
|
|
|
|
message = "The relay timed you out; other players "
|
|
|
|
"have left or never showed up.";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_HEART_YOU:
|
|
|
|
message = "You were disconnected from relay because it didn't "
|
|
|
|
"hear from you in too long.";
|
|
|
|
break;
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_HEART_OTHER:
|
|
|
|
/* *silent = XP_TRUE; */
|
|
|
|
message = "The relay has lost contact with a device in this game.";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_OLDFLAGS:
|
|
|
|
message = "You need to upgrade your copy of Crosswords.";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_SHUTDOWN:
|
|
|
|
message = "Relay disconnected you to shut down (and probably reboot).";
|
|
|
|
break;
|
|
|
|
|
2009-11-22 19:25:49 +00:00
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_BADPROTO:
|
|
|
|
message = "XWRELAY_ERROR_BADPROTO";
|
|
|
|
break;
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_RELAYBUSY:
|
|
|
|
message = "XWRELAY_ERROR_RELAYBUSY";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_OTHER_DISCON:
|
2013-07-10 23:14:48 -07:00
|
|
|
*silent = XP_TRUE; /* happens all the time, and shouldn't matter */
|
2009-11-22 19:25:49 +00:00
|
|
|
message = "XWRELAY_ERROR_OTHER_DISCON";
|
|
|
|
break;
|
|
|
|
|
2009-12-04 08:18:49 +00:00
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_NO_ROOM:
|
|
|
|
message = "No such room. Has the host connected yet to reserve it?";
|
|
|
|
break;
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_DUP_ROOM:
|
|
|
|
message = "That room is reserved by another host. Rename your room, "
|
|
|
|
"become a guest, or try again in a few minutes.";
|
|
|
|
break;
|
|
|
|
|
2009-12-14 04:06:26 +00:00
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_TOO_MANY:
|
|
|
|
message = "You tried to supply more players than the host expected.";
|
|
|
|
break;
|
|
|
|
|
2012-09-09 15:49:17 -07:00
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_DELETED:
|
|
|
|
message = "Game deleted .";
|
|
|
|
break;
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_NORECONN:
|
|
|
|
message = "Cannot reconnect.";
|
|
|
|
break;
|
|
|
|
case ERR_RELAY_BASE + XWRELAY_ERROR_DEADGAME:
|
|
|
|
message = "Game is listed as dead on relay.";
|
|
|
|
break;
|
|
|
|
|
2008-06-30 03:39:27 +00:00
|
|
|
default:
|
|
|
|
XP_LOGF( "no code for error: %d", id );
|
|
|
|
message = "<unrecognized error code reported>";
|
|
|
|
}
|
|
|
|
|
|
|
|
return (XP_UCHAR*)message;
|
|
|
|
} /* linux_getErrString */
|
2011-04-13 06:45:22 -07:00
|
|
|
|
2014-08-26 06:55:26 -07:00
|
|
|
void
|
|
|
|
formatLMI( const LastMoveInfo* lmi, XP_UCHAR* buf, XP_U16 len )
|
|
|
|
{
|
|
|
|
const XP_UCHAR* name = lmi->name;
|
|
|
|
switch( lmi->moveType ) {
|
2016-02-18 07:45:41 -08:00
|
|
|
case ASSIGN_TYPE:
|
|
|
|
XP_SNPRINTF( buf, len, "Tiles assigned to %s", name );
|
|
|
|
break;
|
2014-08-26 06:55:26 -07:00
|
|
|
case MOVE_TYPE:
|
|
|
|
if ( 0 == lmi->nTiles ) {
|
|
|
|
XP_SNPRINTF( buf, len, "%s passed", name );
|
|
|
|
} else {
|
|
|
|
XP_SNPRINTF( buf, len, "%s played %s for %d points", name,
|
|
|
|
lmi->word, lmi->score );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TRADE_TYPE:
|
|
|
|
XP_SNPRINTF( buf, len, "%s traded %d tiles",
|
|
|
|
name, lmi->nTiles );
|
|
|
|
break;
|
|
|
|
case PHONY_TYPE:
|
|
|
|
XP_SNPRINTF( buf, len, "%s lost a turn", name );
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-19 18:34:26 -07:00
|
|
|
void
|
2017-02-17 08:10:09 -08:00
|
|
|
formatConfirmTrade( CommonGlobals* cGlobals, const XP_UCHAR** tiles,
|
|
|
|
XP_U16 nTiles )
|
2011-10-19 18:34:26 -07:00
|
|
|
{
|
|
|
|
int offset = 0;
|
2017-02-17 08:10:09 -08:00
|
|
|
char tileBuf[128];
|
2011-10-19 18:34:26 -07:00
|
|
|
int ii;
|
|
|
|
|
|
|
|
XP_ASSERT( nTiles > 0 );
|
|
|
|
for ( ii = 0; ii < nTiles; ++ii ) {
|
|
|
|
offset += snprintf( &tileBuf[offset], sizeof(tileBuf) - offset,
|
|
|
|
"%s, ", tiles[ii] );
|
|
|
|
XP_ASSERT( offset < sizeof(tileBuf) );
|
|
|
|
}
|
|
|
|
tileBuf[offset-2] = '\0';
|
|
|
|
|
2019-01-28 16:31:41 -08:00
|
|
|
snprintf( cGlobals->question, QUESTION_LEN,
|
2011-10-19 18:34:26 -07:00
|
|
|
"Are you sure you want to trade the selected tiles (%s)?",
|
|
|
|
tileBuf );
|
|
|
|
}
|
|
|
|
|
2011-11-17 19:56:36 -08:00
|
|
|
typedef struct _MsgRec {
|
|
|
|
XP_U8* msg;
|
2011-11-21 17:57:38 -08:00
|
|
|
XP_U16 msglen;
|
2011-11-17 19:56:36 -08:00
|
|
|
} MsgRec;
|
|
|
|
|
|
|
|
void
|
|
|
|
initNoConnStorage( CommonGlobals* cGlobals )
|
|
|
|
{
|
|
|
|
XP_ASSERT( NULL == cGlobals->noConnMsgs );
|
2011-11-21 17:57:38 -08:00
|
|
|
cGlobals->noConnMsgs = g_hash_table_new( g_str_hash, g_str_equal );
|
2011-11-17 19:56:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
XP_Bool
|
2011-11-21 17:57:38 -08:00
|
|
|
storeNoConnMsg( CommonGlobals* cGlobals, const XP_U8* msg, XP_U16 msglen,
|
2011-11-17 19:56:36 -08:00
|
|
|
const XP_UCHAR* relayID )
|
|
|
|
{
|
2011-11-21 17:57:38 -08:00
|
|
|
XP_ASSERT( 0 < msglen );
|
|
|
|
XP_Bool inUse = NULL != cGlobals->noConnMsgs;
|
2011-11-17 19:56:36 -08:00
|
|
|
if ( inUse ) {
|
2011-11-21 17:57:38 -08:00
|
|
|
GSList* list = g_hash_table_lookup( cGlobals->noConnMsgs, relayID );
|
|
|
|
gboolean missing = NULL == list;
|
|
|
|
|
2011-11-17 19:56:36 -08:00
|
|
|
MsgRec* msgrec = g_malloc( sizeof(*msgrec) );
|
2011-11-21 17:57:38 -08:00
|
|
|
msgrec->msg = g_malloc( msglen );
|
|
|
|
XP_MEMCPY( msgrec->msg, msg, msglen );
|
|
|
|
msgrec->msglen = msglen;
|
|
|
|
list = g_slist_append( list, msgrec );
|
|
|
|
if ( missing ) {
|
|
|
|
gchar* key = g_strdup( relayID ); /* will leak */
|
|
|
|
g_hash_table_insert( cGlobals->noConnMsgs, key, list );
|
|
|
|
}
|
2011-11-17 19:56:36 -08:00
|
|
|
}
|
|
|
|
return inUse;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
writeNoConnMsgs( CommonGlobals* cGlobals, int fd )
|
|
|
|
{
|
2011-11-21 17:57:38 -08:00
|
|
|
GHashTable* hash = cGlobals->noConnMsgs;
|
|
|
|
GList* keys = g_hash_table_get_keys( hash );
|
|
|
|
GList* iter;
|
|
|
|
for ( iter = keys; !!iter; iter = iter->next ) {
|
|
|
|
XP_UCHAR* relayID = (XP_UCHAR*)iter->data;
|
|
|
|
GSList* list = (GSList*)g_hash_table_lookup( hash, relayID );
|
|
|
|
guint nMsgs = g_slist_length( list );
|
|
|
|
XP_ASSERT( 0 < nMsgs );
|
2011-11-17 19:56:36 -08:00
|
|
|
|
2018-06-27 23:37:28 -07:00
|
|
|
XWStreamCtxt* stream = mem_stream_make_raw( MPPARM(cGlobals->util->mpool)
|
|
|
|
cGlobals->params->vtMgr );
|
2011-11-17 19:56:36 -08:00
|
|
|
stream_putU16( stream, 1 ); /* number of relayIDs */
|
|
|
|
stream_catString( stream, relayID );
|
|
|
|
stream_putU8( stream, '\n' );
|
|
|
|
stream_putU16( stream, nMsgs );
|
|
|
|
|
|
|
|
int ii;
|
|
|
|
for ( ii = 0; ii < nMsgs; ++ii ) {
|
2011-11-21 17:57:38 -08:00
|
|
|
MsgRec* rec = g_slist_nth_data( list, ii );
|
|
|
|
stream_putU16( stream, rec->msglen );
|
|
|
|
stream_putBytes( stream, rec->msg, rec->msglen );
|
2011-11-17 19:56:36 -08:00
|
|
|
|
|
|
|
g_free( rec->msg );
|
|
|
|
g_free( rec );
|
|
|
|
}
|
2011-11-21 17:57:38 -08:00
|
|
|
g_slist_free( list );
|
2011-11-17 19:56:36 -08:00
|
|
|
|
|
|
|
XP_U16 siz = stream_getSize( stream );
|
|
|
|
/* XP_U8 buf[siz]; */
|
|
|
|
/* stream_getBytes( stream, buf, siz ); */
|
|
|
|
XP_U16 tmp = XP_HTONS( siz );
|
2012-10-07 11:43:50 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
ssize_t nwritten =
|
|
|
|
#endif
|
|
|
|
write( fd, &tmp, sizeof(tmp) );
|
2011-11-17 19:56:36 -08:00
|
|
|
XP_ASSERT( nwritten == sizeof(tmp) );
|
2012-10-07 11:43:50 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
nwritten =
|
|
|
|
#endif
|
|
|
|
write( fd, stream_getPtr( stream ), siz );
|
2011-11-17 19:56:36 -08:00
|
|
|
XP_ASSERT( nwritten == siz );
|
|
|
|
stream_destroy( stream );
|
|
|
|
}
|
2011-11-21 17:57:38 -08:00
|
|
|
g_list_free( keys );
|
|
|
|
g_hash_table_unref( hash );
|
|
|
|
cGlobals->noConnMsgs = NULL;
|
2011-11-17 19:56:36 -08:00
|
|
|
} /* writeNoConnMsgs */
|
|
|
|
|
2011-04-13 06:45:22 -07:00
|
|
|
#ifdef TEXT_MODEL
|
|
|
|
/* This is broken for UTF-8, even Spanish */
|
|
|
|
void
|
|
|
|
linux_lowerstr( XP_UCHAR* str )
|
|
|
|
{
|
|
|
|
while ( *str ) {
|
|
|
|
*str = tolower( *str );
|
|
|
|
++str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|