mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-28 07:58:08 +01:00
Add files forgotten with last checkin; add additional parameter to
addListener so listeners can be removed on reset.
This commit is contained in:
parent
e1d5632a7b
commit
a560c9f8a8
6 changed files with 387 additions and 16 deletions
|
@ -694,7 +694,8 @@ curses_socket_changed( void* closure, int oldSock, int newSock,
|
|||
} /* curses_socket_changed */
|
||||
|
||||
static void
|
||||
curses_socket_acceptor( int listener, Acceptor func, CommonGlobals* cGlobals )
|
||||
curses_socket_acceptor( int listener, Acceptor func, CommonGlobals* cGlobals,
|
||||
void** storage )
|
||||
{
|
||||
CursesAppGlobals* globals = (CursesAppGlobals*)cGlobals;
|
||||
XP_ASSERT( !cGlobals->acceptor || (func == cGlobals->acceptor) );
|
||||
|
|
|
@ -1721,27 +1721,48 @@ acceptorInput( GIOChannel* source, GIOCondition condition, gpointer data )
|
|||
}
|
||||
|
||||
return keepSource;
|
||||
}
|
||||
} /* acceptorInput */
|
||||
|
||||
static void
|
||||
gtk_socket_acceptor( int listener, Acceptor func, CommonGlobals* globals )
|
||||
gtk_socket_acceptor( int listener, Acceptor func, CommonGlobals* globals,
|
||||
void** storage )
|
||||
{
|
||||
SockInfo* info = (SockInfo*)*storage;
|
||||
GIOChannel* channel;
|
||||
guint result;
|
||||
guint watch;
|
||||
|
||||
LOG_FUNC();
|
||||
|
||||
XP_ASSERT( !globals->acceptor || (func == globals->acceptor) );
|
||||
globals->acceptor = func;
|
||||
if ( listener == -1 ) {
|
||||
XP_ASSERT( !!globals->acceptor );
|
||||
globals->acceptor = NULL;
|
||||
XP_ASSERT( !!info );
|
||||
int oldSock = info->socket;
|
||||
g_source_remove( info->watch );
|
||||
g_io_channel_unref( info->channel );
|
||||
XP_FREE( globals->params->util->mpool, info );
|
||||
*storage = NULL;
|
||||
XP_LOGF( "Removed listener %d from gtk's list of listened-to sockets", oldSock );
|
||||
} else {
|
||||
XP_ASSERT( !globals->acceptor || (func == globals->acceptor) );
|
||||
globals->acceptor = func;
|
||||
|
||||
channel = g_io_channel_unix_new( listener );
|
||||
g_io_channel_set_close_on_unref( channel, TRUE );
|
||||
result = g_io_add_watch( channel,
|
||||
G_IO_IN | G_IO_HUP | G_IO_ERR | G_IO_PRI,
|
||||
acceptorInput, globals );
|
||||
g_io_channel_unref( channel ); /* only main loop holds it now */
|
||||
XP_LOGF( "%s: g_io_add_watch(%d) => %d", __FUNCTION__, listener, result );
|
||||
}
|
||||
channel = g_io_channel_unix_new( listener );
|
||||
g_io_channel_set_close_on_unref( channel, TRUE );
|
||||
watch = g_io_add_watch( channel,
|
||||
G_IO_IN | G_IO_HUP | G_IO_ERR | G_IO_PRI,
|
||||
acceptorInput, globals );
|
||||
g_io_channel_unref( channel ); /* only main loop holds it now */
|
||||
XP_LOGF( "%s: g_io_add_watch(%d) => %d", __FUNCTION__, listener, watch );
|
||||
|
||||
XP_ASSERT( NULL == info );
|
||||
info = XP_MALLOC( globals->params->util->mpool, sizeof(*info) );
|
||||
info->channel = channel;
|
||||
info->watch = watch;
|
||||
info->socket = listener;
|
||||
*storage = info;
|
||||
}
|
||||
} /* gtk_socket_acceptor */
|
||||
|
||||
static void
|
||||
sendOnClose( XWStreamCtxt* stream, void* closure )
|
||||
|
|
|
@ -56,6 +56,7 @@ typedef struct LinBtStuff {
|
|||
union {
|
||||
struct {
|
||||
int listener; /* socket */
|
||||
void* listenerStorage;
|
||||
XP_Bool threadDie;
|
||||
sdp_session_t* session;
|
||||
} master;
|
||||
|
@ -347,7 +348,8 @@ lbt_listenerSetup( CommonGlobals* globals )
|
|||
|
||||
lbt_register( btStuff, htobs( XW_PSM ), rc_channel );
|
||||
|
||||
(*globals->addAcceptor)( listener, lbt_accept, globals );
|
||||
(*globals->addAcceptor)( listener, lbt_accept, globals,
|
||||
&btStuff->u.master.listenerStorage );
|
||||
} /* lbt_listenerSetup */
|
||||
|
||||
void
|
||||
|
@ -392,6 +394,9 @@ linux_bt_close( CommonGlobals* globals )
|
|||
if ( !!btStuff ) {
|
||||
if ( btStuff->amMaster ) {
|
||||
XP_LOGF( "%s: closing listener socket %d", __func__, btStuff->u.master.listener );
|
||||
/* Remove from main event loop */
|
||||
(*globals->addAcceptor)( -1, NULL, globals,
|
||||
&btStuff->u.master.listenerStorage );
|
||||
close( btStuff->u.master.listener );
|
||||
btStuff->u.master.listener = -1;
|
||||
|
||||
|
|
304
xwords4/linux/linuxudp.c
Normal file
304
xwords4/linux/linuxudp.c
Normal file
|
@ -0,0 +1,304 @@
|
|||
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; compile-command: "make MEMDEBUG=TRUE";-*- */
|
||||
/*
|
||||
* Copyright 2007 by Eric House (xwords@eehouse.org). All rights
|
||||
* 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.
|
||||
*/
|
||||
|
||||
#ifdef XWFEATURE_IP_DIRECT
|
||||
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <errno.h>
|
||||
#include <netdb.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/socket.h>
|
||||
#include <bluetooth/bluetooth.h>
|
||||
#if defined BT_USE_L2CAP
|
||||
# include <bluetooth/l2cap.h>
|
||||
#elif defined BT_USE_RFCOMM
|
||||
# include <bluetooth/rfcomm.h>
|
||||
#endif
|
||||
#include <bluetooth/sdp.h>
|
||||
#include <bluetooth/sdp_lib.h>
|
||||
|
||||
#include "linuxudp.h"
|
||||
#include "comms.h"
|
||||
#include "strutils.h"
|
||||
|
||||
/* Conecting: Expectation is that the client initiates the connection to a
|
||||
* known port on the server and that the server uses return addresses to reach
|
||||
* the client. This works for games started from scratch. But when we start
|
||||
* from saved games it doesn't: the server knows whatever ephemeral port the
|
||||
* client was on before, but that's all. The client needs to ping the server
|
||||
* immediately, perhaps in a way analogous to btStartup.
|
||||
*/
|
||||
|
||||
|
||||
typedef struct LinUDPStuff {
|
||||
CommonGlobals* globals;
|
||||
CommsAddrRec addr;
|
||||
XP_Bool isServer;
|
||||
void* storage;
|
||||
int knownSocks[MAX_NUM_PLAYERS];
|
||||
int nKnownSocks;
|
||||
int socket; /* host opens to receive; guest opens to send */
|
||||
} LinUDPStuff;
|
||||
|
||||
void
|
||||
linux_udp_open( CommonGlobals* globals, const CommsAddrRec* newAddr )
|
||||
{
|
||||
LOG_FUNC();
|
||||
LinUDPStuff* stuff = globals->udpStuff;
|
||||
if ( !stuff ) {
|
||||
struct sockaddr_in saddr;
|
||||
int err;
|
||||
|
||||
stuff = XP_MALLOC( globals->params->util->mpool, sizeof(*stuff) );
|
||||
XP_MEMSET( stuff, 0, sizeof(*stuff) );
|
||||
XP_MEMCPY( &stuff->addr, newAddr, sizeof(stuff->addr) );
|
||||
|
||||
globals->udpStuff = stuff;
|
||||
stuff->globals = globals;
|
||||
stuff->socket = -1;
|
||||
|
||||
stuff->isServer = comms_getIsServer( globals->game.comms );
|
||||
|
||||
if ( stuff->isServer ) {
|
||||
int listenSock;
|
||||
listenSock = socket( AF_INET, SOCK_DGRAM, IPPROTO_UDP );
|
||||
saddr.sin_family = PF_INET;
|
||||
saddr.sin_addr.s_addr = htonl(INADDR_ANY);
|
||||
saddr.sin_port = htons(stuff->addr.u.ip.port_ip);
|
||||
XP_LOGF( "binding listen socket" );
|
||||
err = bind( listenSock, (struct sockaddr *)&saddr, sizeof(saddr) );
|
||||
if ( 0 != err ) {
|
||||
XP_LOGF( "bind()=>%s", strerror(errno) );
|
||||
listenSock = -1;
|
||||
}
|
||||
stuff->socket = listenSock;
|
||||
(*globals->socketChanged)( globals->socketChangedClosure, -1,
|
||||
listenSock, &stuff->storage );
|
||||
}
|
||||
|
||||
}
|
||||
LOG_RETURN_VOID();
|
||||
} /* linux_udp_open */
|
||||
|
||||
void
|
||||
linux_udp_reset( CommonGlobals* globals )
|
||||
{
|
||||
CommsAddrRec addr;
|
||||
LinUDPStuff* stuff = globals->udpStuff;
|
||||
LOG_FUNC();
|
||||
XP_ASSERT( stuff );
|
||||
if ( !!stuff ) {
|
||||
XP_MEMCPY( &addr, &stuff->addr, sizeof(addr) );
|
||||
linux_udp_close( globals );
|
||||
}
|
||||
sleep( 3 );
|
||||
linux_udp_open( globals, &addr );
|
||||
LOG_RETURN_VOID();
|
||||
}
|
||||
|
||||
void
|
||||
linux_udp_close( CommonGlobals* globals )
|
||||
{
|
||||
LinUDPStuff* stuff = globals->udpStuff;
|
||||
LOG_FUNC();
|
||||
if ( !!stuff ) {
|
||||
if ( stuff->socket != -1 ) {
|
||||
XP_LOGF( "closed socket %d", stuff->socket );
|
||||
(*globals->socketChanged)( globals->socketChangedClosure, stuff->socket, -1,
|
||||
&stuff->storage );
|
||||
close( stuff->socket );
|
||||
stuff->socket = -1;
|
||||
} else {
|
||||
XP_LOGF( "no socket to close" );
|
||||
}
|
||||
XP_FREE( globals->params->util->mpool, stuff );
|
||||
globals->udpStuff = NULL;
|
||||
}
|
||||
LOG_RETURN_VOID();
|
||||
}
|
||||
|
||||
static XP_U32
|
||||
addrForHost( const CommsAddrRec* addr )
|
||||
{
|
||||
struct hostent* host;
|
||||
XP_U32 ip = addr->u.ip.ipAddr_ip;
|
||||
|
||||
if ( 0L == ip ) {
|
||||
host = gethostbyname( addr->u.ip.hostName_ip );
|
||||
if ( NULL == host ) {
|
||||
XP_WARNF( "gethostbyname returned -1\n" );
|
||||
} else {
|
||||
XP_MEMCPY( &ip, host->h_addr_list[0], sizeof(ip) );
|
||||
ip = ntohl(ip);
|
||||
}
|
||||
XP_LOGF( "%s found %lx for %s", __func__, ip, addr->u.ip.hostName_ip );
|
||||
}
|
||||
return ip;
|
||||
} /* addrForHost */
|
||||
|
||||
static void
|
||||
addressToServer( struct sockaddr_in* to, const CommsAddrRec* addr )
|
||||
{
|
||||
to->sin_family = PF_INET;
|
||||
to->sin_addr.s_addr = htonl( addrForHost(addr) );
|
||||
to->sin_port = htons(addr->u.ip.port_ip);
|
||||
}
|
||||
|
||||
static void
|
||||
remember( LinUDPStuff* stuff, int sock )
|
||||
{
|
||||
XP_U16 i;
|
||||
XP_Bool known;
|
||||
for ( i = 0, known = XP_FALSE;
|
||||
!known && i < sizeof(stuff->knownSocks)/sizeof(stuff->knownSocks[0]);
|
||||
++i ) {
|
||||
if ( stuff->knownSocks[i] == sock ) {
|
||||
known = XP_TRUE;
|
||||
}
|
||||
}
|
||||
|
||||
if ( !known ) {
|
||||
XP_ASSERT( stuff->nKnownSocks
|
||||
< sizeof(stuff->knownSocks)/sizeof(stuff->knownSocks[0])-1 );
|
||||
XP_LOGF( "%s recording %d", __func__, sock );
|
||||
stuff->knownSocks[stuff->nKnownSocks++] = sock;
|
||||
}
|
||||
}
|
||||
|
||||
static XP_Bool
|
||||
remembered( const LinUDPStuff* stuff, int sock )
|
||||
{
|
||||
XP_Bool known = XP_FALSE;
|
||||
XP_U16 i;
|
||||
for ( i = 0; i < stuff->nKnownSocks; ++i ) {
|
||||
XP_ASSERT( i < sizeof(stuff->knownSocks)/sizeof(stuff->knownSocks[0]) );
|
||||
if ( stuff->knownSocks[i] == sock ) {
|
||||
known = XP_TRUE;
|
||||
}
|
||||
}
|
||||
LOG_RETURNF( "%d", (int)known );
|
||||
return known;
|
||||
}
|
||||
|
||||
static XP_Bool
|
||||
addressToClient( LinUDPStuff* stuff, struct sockaddr_in* to, const CommsAddrRec* addr )
|
||||
{
|
||||
int port = addr->u.ip.port_ip;
|
||||
XP_Bool known = remembered( stuff, port );
|
||||
|
||||
if ( known ) {
|
||||
to->sin_family = PF_INET;
|
||||
to->sin_addr.s_addr = htonl(addr->u.ip.ipAddr_ip);
|
||||
to->sin_port = htons(port);
|
||||
}
|
||||
return known;
|
||||
}
|
||||
|
||||
XP_S16
|
||||
linux_udp_send( const XP_U8* buf, XP_U16 buflen, const CommsAddrRec* addrp,
|
||||
CommonGlobals* globals )
|
||||
{
|
||||
LinUDPStuff* stuff = globals->udpStuff;
|
||||
ssize_t nSent = -1;
|
||||
|
||||
LOG_FUNC();
|
||||
if ( NULL != stuff ) {
|
||||
XP_Bool haveAddress = XP_TRUE;
|
||||
CommsAddrRec addr;
|
||||
if ( !addrp ) {
|
||||
comms_getAddr( globals->game.comms, &addr );
|
||||
addrp = &addr;
|
||||
}
|
||||
|
||||
struct sockaddr_in to;
|
||||
XP_MEMSET( &to, 0, sizeof(to) );
|
||||
|
||||
if ( stuff->isServer ) {
|
||||
if ( !addressToClient( stuff, &to, addrp ) ) {
|
||||
haveAddress = XP_FALSE;
|
||||
}
|
||||
} else {
|
||||
if ( stuff->socket == -1 ) {
|
||||
stuff->socket = socket( AF_INET, SOCK_DGRAM, IPPROTO_UDP );
|
||||
XP_LOGF( "%s: client made socket = %d", __func__, stuff->socket );
|
||||
(*stuff->globals->socketChanged)( stuff->globals->socketChangedClosure,
|
||||
-1, stuff->socket, &stuff->storage );
|
||||
}
|
||||
addressToServer( &to, addrp );
|
||||
}
|
||||
|
||||
if ( haveAddress ) {
|
||||
XP_LOGF( "calling sendto: sock=%d; port=%d; ipaddr=%lx", stuff->socket,
|
||||
ntohs(to.sin_port), to.sin_addr.s_addr );
|
||||
nSent = sendto( stuff->socket, buf, buflen, 0,
|
||||
(struct sockaddr*)&to, sizeof(to) );
|
||||
if ( nSent != buflen ) {
|
||||
XP_LOGF( "sendto->%s", strerror(errno) );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
LOG_RETURNF( "%d", nSent );
|
||||
return nSent;
|
||||
} /* linux_udp_send */
|
||||
|
||||
XP_S16
|
||||
linux_udp_receive( int sock, XP_U8* buf, XP_U16 buflen, CommsAddrRec* addr,
|
||||
CommonGlobals* globals )
|
||||
{
|
||||
struct sockaddr_in from;
|
||||
socklen_t fromlen = buflen;
|
||||
|
||||
XP_LOGF( "%s: calling recvfrom on socket %d", __func__, sock );
|
||||
|
||||
ssize_t nRead = recvfrom( sock, buf, buflen, 0, /* flags */
|
||||
(struct sockaddr*)&from, &fromlen );
|
||||
XP_ASSERT( nRead > 0 );
|
||||
XP_LOGF( "%s read %d bytes", __func__, nRead );
|
||||
|
||||
if ( nRead > 0 ) {
|
||||
int port;
|
||||
XP_MEMSET( addr, 0, sizeof(*addr) );
|
||||
addr->conType = COMMS_CONN_IP_DIRECT;
|
||||
port = ntohs(from.sin_port);
|
||||
addr->u.ip.port_ip = port;
|
||||
addr->u.ip.ipAddr_ip = ntohl(from.sin_addr.s_addr);
|
||||
|
||||
if ( globals->udpStuff->isServer ) {
|
||||
remember( globals->udpStuff, port );
|
||||
}
|
||||
}
|
||||
|
||||
return nRead;
|
||||
}
|
||||
|
||||
void
|
||||
linux_udp_socketclosed( CommonGlobals* globals, int sock )
|
||||
{
|
||||
LinUDPStuff* stuff = globals->udpStuff;
|
||||
LOG_FUNC();
|
||||
XP_ASSERT( !!stuff );
|
||||
XP_ASSERT( stuff->socket == sock );
|
||||
stuff->socket = -1;
|
||||
LOG_RETURN_VOID();
|
||||
}
|
||||
|
||||
#endif /* XWFEATURE_IP_DIRECT */
|
40
xwords4/linux/linuxudp.h
Normal file
40
xwords4/linux/linuxudp.h
Normal file
|
@ -0,0 +1,40 @@
|
|||
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; compile-command: "make MEMDEBUG=TRUE";-*- */
|
||||
/*
|
||||
* Copyright 2007 by Eric House (xwords@eehouse.org). All rights
|
||||
* 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.
|
||||
*/
|
||||
#ifndef _LINUXUDP_H_
|
||||
#define _LINUXUD_H_
|
||||
|
||||
#ifdef XWFEATURE_IP_DIRECT
|
||||
|
||||
#include "main.h"
|
||||
|
||||
void linux_udp_open( CommonGlobals* globals, const CommsAddrRec* newAddr );
|
||||
void linux_udp_reset( CommonGlobals* globals );
|
||||
void linux_udp_close( CommonGlobals* globals );
|
||||
|
||||
XP_S16 linux_udp_send( const XP_U8* buf, XP_U16 buflen,
|
||||
const CommsAddrRec* addrRec,
|
||||
CommonGlobals* globals );
|
||||
XP_S16 linux_udp_receive( int sock, XP_U8* buf, XP_U16 buflen, CommsAddrRec* addr,
|
||||
CommonGlobals* globals );
|
||||
|
||||
void linux_udp_socketclosed( CommonGlobals* globals, int sock );
|
||||
|
||||
#endif /* XWFEATURE_IP_DIRECT */
|
||||
#endif /* #ifndef _LINUXUD_H_ */
|
|
@ -99,7 +99,7 @@ typedef void (*SocketChangedFunc)(void* closure, int oldsock, int newsock,
|
|||
void** storage );
|
||||
typedef XP_Bool (*Acceptor)( int sock, void* ctxt );
|
||||
typedef void (*AddAcceptorFunc)(int listener, Acceptor func,
|
||||
CommonGlobals* globals );
|
||||
CommonGlobals* globals, void** storage );
|
||||
|
||||
struct CommonGlobals {
|
||||
LaunchParams* params;
|
||||
|
|
Loading…
Add table
Reference in a new issue