2013-01-16 15:46:33 +01:00
|
|
|
/* -*- 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 <netdb.h>
|
|
|
|
#include <errno.h>
|
2013-01-23 16:43:58 +01:00
|
|
|
#include <stdbool.h>
|
2013-01-16 15:46:33 +01:00
|
|
|
|
|
|
|
#include "relaycon.h"
|
|
|
|
#include "comtypes.h"
|
|
|
|
|
|
|
|
typedef struct _RelayConStorage {
|
|
|
|
int socket;
|
|
|
|
RelayConnProcs procs;
|
|
|
|
void* procsClosure;
|
2013-01-19 23:37:49 +01:00
|
|
|
struct sockaddr_in saddr;
|
2013-07-20 20:57:10 +02:00
|
|
|
uint32_t nextID;
|
2013-01-16 15:46:33 +01:00
|
|
|
} RelayConStorage;
|
|
|
|
|
2013-01-23 16:43:58 +01:00
|
|
|
typedef struct _MsgHeader {
|
|
|
|
XWRelayReg cmd;
|
|
|
|
uint32_t packetID;
|
|
|
|
} MsgHeader;
|
|
|
|
|
2013-01-16 15:46:33 +01:00
|
|
|
static RelayConStorage* getStorage( LaunchParams* params );
|
|
|
|
static XP_U32 hostNameToIP( const XP_UCHAR* name );
|
|
|
|
static void relaycon_receive( void* closure, int socket );
|
2013-01-19 23:37:49 +01:00
|
|
|
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 );
|
2013-01-23 16:43:58 +01:00
|
|
|
static void getNetString( const XP_U8** ptr, XP_U16 len, XP_UCHAR* buf );
|
|
|
|
static XP_U16 getNetShort( const XP_U8** ptr );
|
2013-01-29 16:42:10 +01:00
|
|
|
static XP_U32 getNetLong( const XP_U8** ptr );
|
2013-07-20 20:57:10 +02:00
|
|
|
static int writeHeader( RelayConStorage* storage, XP_U8* dest, XWRelayReg cmd );
|
2013-01-23 16:43:58 +01:00
|
|
|
static bool readHeader( const XP_U8** buf, MsgHeader* header );
|
2013-01-29 16:42:10 +01:00
|
|
|
static size_t writeDevID( XP_U8* buf, size_t len, const XP_UCHAR* str );
|
|
|
|
|
2013-01-16 15:46:33 +01:00
|
|
|
|
|
|
|
void
|
|
|
|
relaycon_init( LaunchParams* params, const RelayConnProcs* procs,
|
2013-01-19 23:37:49 +01:00
|
|
|
void* procsClosure, const char* host, int port )
|
2013-01-16 15:46:33 +01:00
|
|
|
{
|
|
|
|
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 );
|
2013-01-24 16:49:49 +01:00
|
|
|
(*procs->socketChanged)( procsClosure, storage->socket, -1,
|
|
|
|
relaycon_receive, params );
|
2013-01-16 15:46:33 +01:00
|
|
|
|
2013-01-19 23:37:49 +01:00
|
|
|
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 );
|
2013-01-29 16:42:10 +01:00
|
|
|
XP_ASSERT( !!devID || typ == ID_TYPE_ANON );
|
2013-07-20 20:57:10 +02:00
|
|
|
indx += writeHeader( storage, tmpbuf, XWPDEV_REG );
|
2013-01-19 23:37:49 +01:00
|
|
|
tmpbuf[indx++] = typ;
|
2013-01-29 16:42:10 +01:00
|
|
|
indx += writeDevID( &tmpbuf[indx], sizeof(tmpbuf) - indx, devID );
|
2013-01-19 23:37:49 +01:00
|
|
|
|
|
|
|
sendIt( storage, tmpbuf, indx );
|
2013-01-16 15:46:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
XP_S16
|
|
|
|
relaycon_send( LaunchParams* params, const XP_U8* buf, XP_U16 buflen,
|
2013-01-19 23:37:49 +01:00
|
|
|
XP_U32 gameToken, const CommsAddrRec* XP_UNUSED(addrRec) )
|
2013-01-16 15:46:33 +01:00
|
|
|
{
|
2013-01-25 04:20:35 +01:00
|
|
|
XP_ASSERT( 0 != gameToken );
|
2013-01-16 15:46:33 +01:00
|
|
|
ssize_t nSent = -1;
|
|
|
|
RelayConStorage* storage = getStorage( params );
|
|
|
|
|
2013-01-23 16:43:58 +01:00
|
|
|
XP_U8 tmpbuf[1 + 4 + 1 + sizeof(gameToken) + buflen];
|
|
|
|
int indx = 0;
|
2013-07-20 20:57:10 +02:00
|
|
|
indx += writeHeader( storage, tmpbuf, XWPDEV_MSG );
|
2013-01-16 15:46:33 +01:00
|
|
|
XP_U32 inNBO = htonl(gameToken);
|
2013-01-23 16:43:58 +01:00
|
|
|
XP_MEMCPY( &tmpbuf[indx], &inNBO, sizeof(inNBO) );
|
|
|
|
indx += sizeof(inNBO);
|
|
|
|
XP_MEMCPY( &tmpbuf[indx], buf, buflen );
|
|
|
|
indx += buflen;
|
|
|
|
nSent = sendIt( storage, tmpbuf, indx );
|
2013-01-19 23:37:49 +01:00
|
|
|
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 );
|
2013-01-25 04:20:35 +01:00
|
|
|
XP_ASSERT( 0 != gameToken );
|
2013-01-19 23:37:49 +01:00
|
|
|
XP_U16 indx = 0;
|
|
|
|
ssize_t nSent = -1;
|
|
|
|
RelayConStorage* storage = getStorage( params );
|
|
|
|
|
|
|
|
XP_U16 idLen = XP_STRLEN( relayID );
|
2013-01-23 16:43:58 +01:00
|
|
|
XP_U8 tmpbuf[1 + 4 + 1 +
|
2013-01-19 23:37:49 +01:00
|
|
|
1 + idLen +
|
|
|
|
sizeof(gameToken) + buflen];
|
2013-07-20 20:57:10 +02:00
|
|
|
indx += writeHeader( storage, tmpbuf, XWPDEV_MSGNOCONN );
|
2013-01-19 23:37:49 +01:00
|
|
|
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;
|
2013-01-16 15:46:33 +01:00
|
|
|
}
|
|
|
|
LOG_RETURNF( "%d", nSent );
|
|
|
|
return nSent;
|
|
|
|
}
|
|
|
|
|
2013-01-19 23:37:49 +01:00
|
|
|
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;
|
2013-07-20 20:57:10 +02:00
|
|
|
indx += writeHeader( storage, tmpbuf, XWPDEV_RQSTMSGS );
|
2013-01-19 23:37:49 +01:00
|
|
|
indx += addStrWithLength( &tmpbuf[indx], tmpbuf + sizeof(tmpbuf), devID );
|
|
|
|
|
|
|
|
sendIt( storage, tmpbuf, indx );
|
|
|
|
}
|
|
|
|
|
2013-01-29 16:42:10 +01:00
|
|
|
void
|
|
|
|
relaycon_deleted( LaunchParams* params, const XP_UCHAR* devID,
|
|
|
|
XP_U32 gameToken )
|
|
|
|
{
|
|
|
|
LOG_FUNC();
|
|
|
|
RelayConStorage* storage = getStorage( params );
|
|
|
|
XP_U8 tmpbuf[128];
|
|
|
|
int indx = 0;
|
2013-07-20 20:57:10 +02:00
|
|
|
indx += writeHeader( storage, tmpbuf, XWPDEV_DELGAME );
|
2013-01-29 16:42:10 +01:00
|
|
|
indx += writeDevID( &tmpbuf[indx], sizeof(tmpbuf) - indx, devID );
|
|
|
|
gameToken = htonl( gameToken );
|
|
|
|
memcpy( &tmpbuf[indx], &gameToken, sizeof(gameToken) );
|
|
|
|
indx += sizeof( gameToken );
|
|
|
|
|
|
|
|
sendIt( storage, tmpbuf, indx );
|
|
|
|
}
|
|
|
|
|
2013-01-23 16:43:58 +01:00
|
|
|
static void
|
|
|
|
sendAckIf( RelayConStorage* storage, const MsgHeader* header )
|
|
|
|
{
|
|
|
|
if ( header->cmd != XWPDEV_ACK ) {
|
|
|
|
XP_U8 tmpbuf[16];
|
2013-07-20 20:57:10 +02:00
|
|
|
int indx = writeHeader( storage, tmpbuf, XWPDEV_ACK );
|
2013-01-23 16:43:58 +01:00
|
|
|
uint32_t msgID = htonl( header->packetID );
|
|
|
|
memcpy( &tmpbuf[indx], &msgID, sizeof(msgID) );
|
|
|
|
indx += sizeof(msgID);
|
|
|
|
sendIt( storage, tmpbuf, indx );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-16 15:46:33 +01:00
|
|
|
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 );
|
2013-07-18 16:07:25 +02:00
|
|
|
|
|
|
|
gchar* b64 = g_base64_encode( (const guchar*)buf,
|
|
|
|
((0 <= nRead)? nRead : 0) );
|
|
|
|
XP_LOGF( "%s: read %d bytes ('%s')", __func__, nRead, b64 );
|
|
|
|
g_free( b64 );
|
2013-01-16 15:46:33 +01:00
|
|
|
if ( 0 <= nRead ) {
|
2013-01-23 16:43:58 +01:00
|
|
|
const XP_U8* ptr = buf;
|
2013-01-16 15:46:33 +01:00
|
|
|
const XP_U8* end = buf + nRead;
|
2013-01-23 16:43:58 +01:00
|
|
|
MsgHeader header;
|
|
|
|
if ( readHeader( &ptr, &header ) ) {
|
|
|
|
sendAckIf( storage, &header );
|
|
|
|
switch( header.cmd ) {
|
|
|
|
case XWPDEV_REGRSP: {
|
|
|
|
XP_U16 len = getNetShort( &ptr );
|
|
|
|
XP_UCHAR devID[len+1];
|
|
|
|
getNetString( &ptr, len, devID );
|
2013-07-29 16:27:30 +02:00
|
|
|
XP_U16 maxInterval = getNetShort( &ptr );
|
|
|
|
XP_LOGF( "%s: maxInterval=%d", __func__, maxInterval );
|
2013-08-02 17:01:16 +02:00
|
|
|
(*storage->procs.devIDChanged)( storage->procsClosure, devID,
|
|
|
|
maxInterval );
|
2013-01-23 16:43:58 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case XWPDEV_MSG:
|
|
|
|
(*storage->procs.msgReceived)( storage->procsClosure,
|
|
|
|
ptr, end - ptr );
|
|
|
|
break;
|
|
|
|
case XWPDEV_BADREG:
|
2013-08-02 17:01:16 +02:00
|
|
|
(*storage->procs.devIDChanged)( storage->procsClosure, NULL, 0 );
|
2013-01-23 16:43:58 +01:00
|
|
|
break;
|
|
|
|
case XWPDEV_HAVEMSGS: {
|
2013-01-25 04:20:35 +01:00
|
|
|
(*storage->procs.msgNoticeReceived)( storage->procsClosure );
|
2013-01-23 16:43:58 +01:00
|
|
|
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;
|
|
|
|
}
|
2013-01-29 16:42:10 +01:00
|
|
|
case XWPDEV_ACK: {
|
|
|
|
XP_U32 packetID = getNetLong( &ptr );
|
2013-07-17 15:52:35 +02:00
|
|
|
XP_USE( packetID );
|
2013-01-29 16:42:10 +01:00
|
|
|
XP_LOGF( "got ack for packetID %ld", packetID );
|
|
|
|
break;
|
|
|
|
}
|
2013-01-23 16:43:58 +01:00
|
|
|
default:
|
|
|
|
XP_LOGF( "%s: Unexpected cmd %d", __func__, header.cmd );
|
|
|
|
XP_ASSERT( 0 );
|
|
|
|
}
|
2013-01-16 15:46:33 +01:00
|
|
|
}
|
|
|
|
} 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 )
|
|
|
|
{
|
2013-08-08 06:09:48 +02:00
|
|
|
XP_ASSERT( params->useUdp );
|
2013-01-16 15:46:33 +01:00
|
|
|
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;
|
|
|
|
}
|
2013-01-19 23:37:49 +01:00
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
sendIt( RelayConStorage* storage, const XP_U8* msgbuf, XP_U16 len )
|
|
|
|
{
|
2013-01-20 19:04:24 +01:00
|
|
|
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;
|
2013-01-19 23:37:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
addStrWithLength( XP_U8* buf, XP_U8* end, const XP_UCHAR* str )
|
|
|
|
{
|
2013-01-29 16:42:10 +01:00
|
|
|
XP_U16 len = !!str? XP_STRLEN( str ) : 0;
|
2013-01-19 23:37:49 +01:00
|
|
|
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);
|
|
|
|
}
|
2013-01-20 22:07:01 +01:00
|
|
|
|
2013-01-29 16:42:10 +01:00
|
|
|
static size_t
|
|
|
|
writeDevID( XP_U8* buf, size_t len, const XP_UCHAR* str )
|
|
|
|
{
|
|
|
|
return addStrWithLength( buf, buf + len, str );
|
|
|
|
}
|
|
|
|
|
2013-01-20 22:07:01 +01:00
|
|
|
static XP_U16
|
2013-01-23 16:43:58 +01:00
|
|
|
getNetShort( const XP_U8** ptr )
|
2013-01-20 22:07:01 +01:00
|
|
|
{
|
|
|
|
XP_U16 result;
|
|
|
|
memcpy( &result, *ptr, sizeof(result) );
|
|
|
|
*ptr += sizeof(result);
|
|
|
|
return ntohs( result );
|
|
|
|
}
|
|
|
|
|
2013-01-29 16:42:10 +01:00
|
|
|
static XP_U32
|
|
|
|
getNetLong( const XP_U8** ptr )
|
|
|
|
{
|
|
|
|
XP_U32 result;
|
|
|
|
memcpy( &result, *ptr, sizeof(result) );
|
|
|
|
*ptr += sizeof(result);
|
|
|
|
return ntohl( result );
|
|
|
|
}
|
|
|
|
|
2013-01-20 22:07:01 +01:00
|
|
|
static void
|
2013-01-23 16:43:58 +01:00
|
|
|
getNetString( const XP_U8** ptr, XP_U16 len, XP_UCHAR* buf )
|
2013-01-20 22:07:01 +01:00
|
|
|
{
|
|
|
|
memcpy( buf, *ptr, len );
|
|
|
|
*ptr += len;
|
|
|
|
buf[len] = '\0';
|
|
|
|
}
|
2013-01-23 16:43:58 +01:00
|
|
|
|
|
|
|
static int
|
2013-07-20 20:57:10 +02:00
|
|
|
writeHeader( RelayConStorage* storage, XP_U8* dest, XWRelayReg cmd )
|
2013-01-23 16:43:58 +01:00
|
|
|
{
|
|
|
|
int indx = 0;
|
|
|
|
dest[indx++] = XWPDEV_PROTO_VERSION;
|
2013-07-20 20:57:10 +02:00
|
|
|
uint32_t packetNum = storage->nextID++;
|
|
|
|
packetNum = htonl(packetNum);
|
2013-01-23 16:43:58 +01:00
|
|
|
memcpy( &dest[indx], &packetNum, sizeof(packetNum) );
|
|
|
|
indx += sizeof(packetNum);
|
|
|
|
dest[indx++] = cmd;
|
|
|
|
return indx;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
readHeader( const XP_U8** buf, MsgHeader* header )
|
|
|
|
{
|
|
|
|
const XP_U8* ptr = *buf;
|
|
|
|
bool ok = XWPDEV_PROTO_VERSION == *ptr++;
|
|
|
|
assert( ok );
|
|
|
|
uint32_t packetID;
|
|
|
|
memcpy( &packetID, ptr, sizeof(packetID) );
|
|
|
|
ptr += sizeof(packetID);
|
|
|
|
header->packetID = ntohl( packetID );
|
|
|
|
XP_LOGF( "%s: got packet %d", __func__, header->packetID );
|
|
|
|
header->cmd = *ptr++;
|
|
|
|
*buf = ptr;
|
|
|
|
return ok;
|
|
|
|
}
|
2013-07-12 05:01:17 +02:00
|
|
|
|
|
|
|
/* Utilities */
|
|
|
|
#define TOKEN_MULT 1000000
|
|
|
|
XP_U32
|
|
|
|
makeClientToken( sqlite3_int64 rowid, XP_U16 seed )
|
|
|
|
{
|
|
|
|
XP_ASSERT( rowid < 0x0000FFFF );
|
|
|
|
XP_U32 result = rowid;
|
|
|
|
result *= TOKEN_MULT; /* so visible when displayed as base-10 */
|
|
|
|
result += seed;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rowidFromToken( XP_U32 clientToken, sqlite3_int64* rowid, XP_U16* seed )
|
|
|
|
{
|
|
|
|
*rowid = clientToken / TOKEN_MULT;
|
|
|
|
*seed = clientToken % TOKEN_MULT;
|
|
|
|
}
|