416 lines
12 KiB
C
416 lines
12 KiB
C
/******************************************************************************
|
|
* $Id$
|
|
*
|
|
* Copyright (c) 2005-2007 Transmission authors and contributors
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*****************************************************************************/
|
|
|
|
/***********************************************************************
|
|
* This file handles all outgoing messages
|
|
**********************************************************************/
|
|
|
|
#define PEER_MSG_CHOKE 0
|
|
#define PEER_MSG_UNCHOKE 1
|
|
#define PEER_MSG_INTERESTED 2
|
|
#define PEER_MSG_UNINTERESTED 3
|
|
#define PEER_MSG_HAVE 4
|
|
#define PEER_MSG_BITFIELD 5
|
|
#define PEER_MSG_REQUEST 6
|
|
#define PEER_MSG_PIECE 7
|
|
#define PEER_MSG_CANCEL 8
|
|
#define PEER_MSG_PORT 9
|
|
|
|
static uint8_t * messagesPending( tr_peer_t * peer, int * size )
|
|
{
|
|
if( peer->outBlockSending || peer->outMessagesPos < 1 )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
*size = MIN( peer->outMessagesPos, 1024 );
|
|
|
|
return peer->outMessages;
|
|
}
|
|
|
|
static void messagesSent( tr_peer_t * peer, int size )
|
|
{
|
|
peer->outMessagesPos -= size;
|
|
memmove( peer->outMessages, &peer->outMessages[size],
|
|
peer->outMessagesPos );
|
|
}
|
|
|
|
static uint8_t * blockPending( tr_torrent_t * tor, tr_peer_t * peer,
|
|
int * size )
|
|
{
|
|
if( !peer->outBlockLoaded )
|
|
{
|
|
uint8_t * p;
|
|
tr_request_t * r;
|
|
|
|
if( peer->amChoking || peer->outRequestCount < 1 )
|
|
{
|
|
/* No piece to send */
|
|
return NULL;
|
|
}
|
|
|
|
/* We need to load the block for the next request */
|
|
r = &peer->outRequests[0];
|
|
|
|
/* Sanity check */
|
|
if( !tr_cpPieceIsComplete( tor->completion, r->index ) )
|
|
{
|
|
/* We have been asked for something we don't have, buggy client?
|
|
Let's just drop this request */
|
|
tr_inf( "Block %d/%d/%d was requested but we don't have it",
|
|
r->index, r->begin, r->length );
|
|
(peer->outRequestCount)--;
|
|
memmove( &peer->outRequests[0], &peer->outRequests[1],
|
|
peer->outRequestCount * sizeof( tr_request_t ) );
|
|
return NULL;
|
|
}
|
|
|
|
p = (uint8_t *) peer->outBlock;
|
|
|
|
TR_HTONL( 9 + r->length, p );
|
|
p[4] = PEER_MSG_PIECE;
|
|
TR_HTONL( r->index, p + 5 );
|
|
TR_HTONL( r->begin, p + 9 );
|
|
|
|
tr_ioRead( tor->io, r->index, r->begin, r->length, &p[13] );
|
|
|
|
if( peer->outRequestCount < 1 )
|
|
{
|
|
/* We were choked during the read */
|
|
return NULL;
|
|
}
|
|
|
|
peer_dbg( "SEND piece %d/%d (%d bytes)",
|
|
r->index, r->begin, r->length );
|
|
|
|
peer->outBlockSize = 13 + r->length;
|
|
peer->outBlockLoaded = 1;
|
|
|
|
(peer->outRequestCount)--;
|
|
memmove( &peer->outRequests[0], &peer->outRequests[1],
|
|
peer->outRequestCount * sizeof( tr_request_t ) );
|
|
}
|
|
|
|
*size = MIN( 1024, peer->outBlockSize );
|
|
|
|
return (uint8_t *) peer->outBlock;
|
|
}
|
|
|
|
static void blockSent( tr_peer_t * peer, int size )
|
|
{
|
|
peer->outBlockSize -= size;
|
|
memmove( peer->outBlock, &peer->outBlock[size], peer->outBlockSize );
|
|
|
|
if( peer->outBlockSize > 0 )
|
|
{
|
|
/* We can't send messages until we are done sending the block */
|
|
peer->outBlockSending = 1;
|
|
}
|
|
else
|
|
{
|
|
/* Block fully sent */
|
|
peer->outBlockSending = 0;
|
|
peer->outBlockLoaded = 0;
|
|
}
|
|
}
|
|
|
|
static uint8_t * getMessagePointer( tr_peer_t * peer, int size, int id )
|
|
{
|
|
uint8_t * p;
|
|
int index = 0;
|
|
|
|
size += 4;
|
|
if( peer->azproto )
|
|
{
|
|
index = azmsgIdIndex( id );
|
|
assert( 0 <= index );
|
|
size += 4 + azmsgLen( index ) + 1;
|
|
}
|
|
else if( 0 <= id )
|
|
{
|
|
size++;
|
|
}
|
|
|
|
if( peer->outMessagesPos + size > peer->outMessagesSize )
|
|
{
|
|
peer->outMessagesSize = peer->outMessagesPos + size;
|
|
peer->outMessages = realloc( peer->outMessages,
|
|
peer->outMessagesSize );
|
|
}
|
|
|
|
p = &peer->outMessages[peer->outMessagesPos];
|
|
peer->outMessagesPos += size;
|
|
|
|
TR_HTONL( size - 4, p );
|
|
p += 4;
|
|
if( peer->azproto )
|
|
{
|
|
TR_HTONL( azmsgLen( index ), p );
|
|
memcpy( p + 4, azmsgStr( index ), azmsgLen( index ) );
|
|
p[ 4 + azmsgLen( index ) ] = AZ_EXT_VERSION;
|
|
p += 4 + azmsgLen( index ) + 1;
|
|
}
|
|
else if( 0 <= id )
|
|
{
|
|
*p = id;
|
|
p++;
|
|
}
|
|
|
|
return p;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendKeepAlive
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static void sendKeepAlive( tr_peer_t * peer )
|
|
{
|
|
getMessagePointer( peer, 0, AZ_MSG_BT_KEEP_ALIVE );
|
|
|
|
peer_dbg( "SEND keep-alive" );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* sendChoke
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static void sendChoke( tr_peer_t * peer, int yes )
|
|
{
|
|
int id;
|
|
|
|
id = ( yes ? PEER_MSG_CHOKE : PEER_MSG_UNCHOKE );
|
|
getMessagePointer( peer, 0, id );
|
|
|
|
peer->amChoking = yes;
|
|
|
|
if( !yes )
|
|
{
|
|
/* Drop older requests from the last time it was unchoked,
|
|
if any */
|
|
peer->outRequestCount = 0;
|
|
}
|
|
|
|
peer_dbg( "SEND %schoke", yes ? "" : "un" );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendInterest
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static void sendInterest( tr_peer_t * peer, int yes )
|
|
{
|
|
int id;
|
|
|
|
id = ( yes ? PEER_MSG_INTERESTED : PEER_MSG_UNINTERESTED );
|
|
getMessagePointer( peer, 0, id );
|
|
|
|
peer->amInterested = yes;
|
|
|
|
peer_dbg( "SEND %sinterested", yes ? "" : "un" );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendHave
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static void sendHave( tr_peer_t * peer, int piece )
|
|
{
|
|
uint8_t * p;
|
|
|
|
p = getMessagePointer( peer, 4, PEER_MSG_HAVE );
|
|
|
|
TR_HTONL( piece, p );
|
|
|
|
peer_dbg( "SEND have %d", piece );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendBitfield
|
|
***********************************************************************
|
|
* Builds a 'bitfield' message:
|
|
* - size = 5 + X (4 bytes)
|
|
* - id = 5 (1 byte)
|
|
* - bitfield (X bytes)
|
|
**********************************************************************/
|
|
static void sendBitfield( tr_torrent_t * tor, tr_peer_t * peer )
|
|
{
|
|
uint8_t * p;
|
|
tr_bitfield_t * bitfield;
|
|
|
|
bitfield = tr_cpPieceBitfield( tor->completion );
|
|
p = getMessagePointer( peer, bitfield->len, PEER_MSG_BITFIELD );
|
|
|
|
memcpy( p, bitfield->bits, bitfield->len );
|
|
|
|
peer_dbg( "SEND bitfield" );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendRequest
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static void sendRequest( tr_torrent_t * tor, tr_peer_t * peer, int block )
|
|
{
|
|
tr_info_t * inf = &tor->info;
|
|
tr_request_t * r;
|
|
uint8_t * p;
|
|
|
|
/* Get the piece the block is a part of, its position in the piece
|
|
and its size */
|
|
r = &peer->inRequests[peer->inRequestCount];
|
|
r->index = block / ( inf->pieceSize / tor->blockSize );
|
|
r->begin = ( block % ( inf->pieceSize / tor->blockSize ) ) *
|
|
tor->blockSize;
|
|
r->length = tr_blockSize( block );
|
|
(peer->inRequestCount)++;
|
|
|
|
/* Build the "ask" message */
|
|
p = getMessagePointer( peer, 12, PEER_MSG_REQUEST );
|
|
|
|
TR_HTONL( r->index, p );
|
|
TR_HTONL( r->begin, p + 4 );
|
|
TR_HTONL( r->length, p + 8 );
|
|
|
|
tr_cpDownloaderAdd( tor->completion, block );
|
|
|
|
peer_dbg( "SEND request %d/%d (%d bytes)",
|
|
r->index, r->begin, r->length );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendCancel
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static void sendCancel( tr_torrent_t * tor, int block )
|
|
{
|
|
int i, j;
|
|
uint8_t * p;
|
|
tr_peer_t * peer;
|
|
tr_request_t * r;
|
|
|
|
for( i = 0; i < tor->peerCount; i++ )
|
|
{
|
|
peer = tor->peers[i];
|
|
|
|
for( j = 1; j < peer->inRequestCount; j++ )
|
|
{
|
|
r = &peer->inRequests[j];
|
|
|
|
if( block != tr_block( r->index, r->begin ) )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
p = getMessagePointer( peer, 12, PEER_MSG_CANCEL );
|
|
|
|
/* Build the "cancel" message */
|
|
TR_HTONL( r->index, p );
|
|
TR_HTONL( r->begin, p + 4 );
|
|
TR_HTONL( r->length, p + 8 );
|
|
|
|
peer_dbg( "SEND cancel %d/%d (%d bytes)",
|
|
r->index, r->begin, r->length );
|
|
|
|
(peer->inRequestCount)--;
|
|
memmove( &peer->inRequests[j], &peer->inRequests[j+1],
|
|
( peer->inRequestCount - j ) * sizeof( tr_request_t ) );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendExtended
|
|
***********************************************************************
|
|
* Builds an extended message:
|
|
* - size = 6 + X (4 bytes)
|
|
* - id = 20 (1 byte)
|
|
* - eid = Y (1 byte)
|
|
* - data (X bytes)
|
|
**********************************************************************/
|
|
static int sendExtended( tr_torrent_t * tor, tr_peer_t * peer, int id )
|
|
{
|
|
uint8_t * p;
|
|
char * buf;
|
|
int len;
|
|
|
|
buf = NULL;
|
|
switch( id )
|
|
{
|
|
case EXTENDED_HANDSHAKE_ID:
|
|
buf = makeExtendedHandshake( tor, peer, &len );
|
|
break;
|
|
case EXTENDED_PEX_ID:
|
|
buf = makeUTPex( tor, peer, &len );
|
|
break;
|
|
default:
|
|
assert( 0 );
|
|
break;
|
|
}
|
|
if( NULL == buf )
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
/* add header and queue it to be sent */
|
|
p = getMessagePointer( peer, 1 + len, PEER_MSG_EXTENDED );
|
|
p[0] = id;
|
|
memcpy( p + 1, buf, len );
|
|
free( buf );
|
|
|
|
return 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* sendAZPex
|
|
***********************************************************************
|
|
*
|
|
**********************************************************************/
|
|
static int sendAZPex( tr_torrent_t * tor, tr_peer_t * peer )
|
|
{
|
|
uint8_t * p;
|
|
char * buf;
|
|
int len;
|
|
|
|
buf = makeAZPex( tor, peer, &len );
|
|
if( NULL == buf )
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
/* add header and queue it to be sent */
|
|
p = getMessagePointer( peer, len, AZ_MSG_AZ_PEER_EXCHANGE );
|
|
memcpy( p, buf, len );
|
|
free( buf );
|
|
|
|
return 0;
|
|
}
|