/* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */ /* * Copyright 2013 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. */ #include #include #include "relaycon.h" #include "comtypes.h" typedef struct _RelayConStorage { int socket; RelayConnProcs procs; void* procsClosure; struct sockaddr_in saddr; } RelayConStorage; static RelayConStorage* getStorage( LaunchParams* params ); static XP_U32 hostNameToIP( const XP_UCHAR* name ); static void relaycon_receive( void* closure, int socket ); static ssize_t sendIt( RelayConStorage* storage, const XP_U8* msgbuf, XP_U16 len ); static size_t addStrWithLength( XP_U8* buf, XP_U8* end, const XP_UCHAR* str ); static void getNetString( XP_U8** ptr, XP_U16 len, XP_UCHAR* buf ); static XP_U16 getNetShort( XP_U8** ptr ); void relaycon_init( LaunchParams* params, const RelayConnProcs* procs, void* procsClosure, const char* host, int port ) { XP_ASSERT( !params->relayConStorage ); RelayConStorage* storage = getStorage( params ); XP_MEMCPY( &storage->procs, procs, sizeof(storage->procs) ); storage->procsClosure = procsClosure; storage->socket = socket( AF_INET, SOCK_DGRAM, IPPROTO_UDP ); (*params->socketChanged)( params->socketChangedClosure, storage->socket, -1, relaycon_receive, params ); XP_MEMSET( &storage->saddr, 0, sizeof(storage->saddr) ); storage->saddr.sin_family = PF_INET; storage->saddr.sin_addr.s_addr = htonl( hostNameToIP(host) ); storage->saddr.sin_port = htons(port); } void relaycon_reg( LaunchParams* params, const XP_UCHAR* devID, DevIDType typ ) { LOG_FUNC(); XP_U8 tmpbuf[32]; int indx = 0; RelayConStorage* storage = getStorage( params ); XP_ASSERT( !!devID ); XP_U16 idLen = XP_STRLEN( devID ); XP_U16 lenNBO = XP_HTONS( idLen ); tmpbuf[indx++] = XWPDEV_PROTO_VERSION; tmpbuf[indx++] = XWPDEV_REG; tmpbuf[indx++] = typ; XP_MEMCPY( &tmpbuf[indx], &lenNBO, sizeof(lenNBO) ); indx += sizeof(lenNBO); XP_MEMCPY( &tmpbuf[indx], devID, idLen ); indx += idLen; sendIt( storage, tmpbuf, indx ); } XP_S16 relaycon_send( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, XP_U32 gameToken, const CommsAddrRec* XP_UNUSED(addrRec) ) { ssize_t nSent = -1; RelayConStorage* storage = getStorage( params ); XP_U8 tmpbuf[1 + 1 + sizeof(gameToken) + buflen]; tmpbuf[0] = XWPDEV_PROTO_VERSION; tmpbuf[1] = XWPDEV_MSG; XP_U32 inNBO = htonl(gameToken); XP_MEMCPY( &tmpbuf[2], &inNBO, sizeof(inNBO) ); XP_MEMCPY( &tmpbuf[1 + 1 + sizeof(gameToken)], buf, buflen ); nSent = sendIt( storage, tmpbuf, sizeof(tmpbuf) ); if ( nSent > buflen ) { nSent = buflen; } LOG_RETURNF( "%d", nSent ); return nSent; } XP_S16 relaycon_sendnoconn( LaunchParams* params, const XP_U8* buf, XP_U16 buflen, const XP_UCHAR* relayID, XP_U32 gameToken ) { XP_LOGF( "%s(relayID=%s)", __func__, relayID ); XP_U16 indx = 0; ssize_t nSent = -1; RelayConStorage* storage = getStorage( params ); XP_U16 idLen = XP_STRLEN( relayID ); XP_U8 tmpbuf[1 + 1 + 1 + idLen + sizeof(gameToken) + buflen]; tmpbuf[indx++] = XWPDEV_PROTO_VERSION; tmpbuf[indx++] = XWPDEV_MSGNOCONN; gameToken = htonl( gameToken ); XP_MEMCPY( &tmpbuf[indx], &gameToken, sizeof(gameToken) ); indx += sizeof(gameToken); XP_MEMCPY( &tmpbuf[indx], relayID, idLen ); indx += idLen; tmpbuf[indx++] = '\n'; XP_MEMCPY( &tmpbuf[indx], buf, buflen ); nSent = sendIt( storage, tmpbuf, sizeof(tmpbuf) ); if ( nSent > buflen ) { nSent = buflen; } LOG_RETURNF( "%d", nSent ); return nSent; } void relaycon_requestMsgs( LaunchParams* params, const XP_UCHAR* devID ) { XP_LOGF( "%s(devID=%s)", __func__, devID ); RelayConStorage* storage = getStorage( params ); XP_U8 tmpbuf[128]; int indx = 0; tmpbuf[indx++] = XWPDEV_PROTO_VERSION; tmpbuf[indx++] = XWPDEV_RQSTMSGS; indx += addStrWithLength( &tmpbuf[indx], tmpbuf + sizeof(tmpbuf), devID ); sendIt( storage, tmpbuf, indx ); } static void relaycon_receive( void* closure, int socket ) { LaunchParams* params = (LaunchParams*)closure; XP_ASSERT( !!params->relayConStorage ); RelayConStorage* storage = getStorage( params ); XP_U8 buf[512]; struct sockaddr_in from; socklen_t fromlen = sizeof(from); XP_LOGF( "%s: calling recvfrom on socket %d", __func__, socket ); ssize_t nRead = recvfrom( socket, buf, sizeof(buf), 0, /* flags */ (struct sockaddr*)&from, &fromlen ); XP_LOGF( "%s: read %d bytes", __func__, nRead ); if ( 0 <= nRead ) { XP_U8* ptr = buf; const XP_U8* end = buf + nRead; XP_ASSERT( XWPDEV_PROTO_VERSION == *ptr++ ); XWRelayReg cmd = *ptr++; switch( cmd ) { case XWPDEV_REGRSP: { XP_U16 len = getNetShort( &ptr ); XP_UCHAR devID[len+1]; getNetString( &ptr, len, devID ); (*storage->procs.devIDChanged)( storage->procsClosure, devID ); } break; case XWPDEV_MSG: (*storage->procs.msgReceived)( storage->procsClosure, ptr, end - ptr ); break; case XWPDEV_BADREG: (*storage->procs.devIDChanged)( storage->procsClosure, NULL ); break; case XWPDEV_HAVEMSGS: { XP_U32 gameToken; XP_MEMCPY( &gameToken, ptr, sizeof(gameToken) ); ptr += sizeof( gameToken ); (*storage->procs.msgNoticeReceived)( storage->procsClosure, ntohl(gameToken) ); break; } case XWPDEV_ALERT: { XP_U16 len = getNetShort( &ptr ); XP_UCHAR buf[len+1]; getNetString( &ptr, len, buf ); (*storage->procs.msgErrorMsg)( storage->procsClosure, buf ); break; } default: XP_LOGF( "%s: Unexpected cmd %d", __func__, cmd ); XP_ASSERT( 0 ); } } else { XP_LOGF( "%s: error reading udp socket: %d (%s)", __func__, errno, strerror(errno) ); } } void relaycon_cleanup( LaunchParams* params ) { XP_FREEP( params->mpool, ¶ms->relayConStorage ); } static RelayConStorage* getStorage( LaunchParams* params ) { RelayConStorage* storage = (RelayConStorage*)params->relayConStorage; if ( NULL == storage ) { storage = XP_CALLOC( params->mpool, sizeof(*storage) ); storage->socket = -1; params->relayConStorage = storage; } return storage; } static XP_U32 hostNameToIP( const XP_UCHAR* name ) { XP_U32 ip; struct hostent* host; XP_LOGF( "%s: looking up %s", __func__, name ); host = gethostbyname( name ); if ( NULL == host ) { XP_WARNF( "gethostbyname returned NULL\n" ); } else { XP_MEMCPY( &ip, host->h_addr_list[0], sizeof(ip) ); ip = ntohl(ip); } XP_LOGF( "%s found %lx for %s", __func__, ip, name ); return ip; } static ssize_t sendIt( RelayConStorage* storage, const XP_U8* msgbuf, XP_U16 len ) { ssize_t nSent = sendto( storage->socket, msgbuf, len, 0, /* flags */ (struct sockaddr*)&storage->saddr, sizeof(storage->saddr) ); XP_LOGF( "%s()=>%d", __func__, nSent ); return nSent; } static size_t addStrWithLength( XP_U8* buf, XP_U8* end, const XP_UCHAR* str ) { XP_U16 len = XP_STRLEN( str ); if ( buf + len + sizeof(len) <= end ) { XP_U16 lenNBO = htons( len ); XP_MEMCPY( buf, &lenNBO, sizeof(lenNBO) ); buf += sizeof(lenNBO); XP_MEMCPY( buf, str, len ); } return len + sizeof(len); } static XP_U16 getNetShort( XP_U8** ptr ) { XP_U16 result; memcpy( &result, *ptr, sizeof(result) ); *ptr += sizeof(result); return ntohs( result ); } static void getNetString( XP_U8** ptr, XP_U16 len, XP_UCHAR* buf ) { memcpy( buf, *ptr, len ); *ptr += len; buf[len] = '\0'; }