/* * This file Copyright (C) 2007 Charles Kerr * * This file is licensed by the GPL version 2. Works owned by the * Transmission project are granted a special exemption to clause 2(b) * so that the bulk of its code can remain under the MIT license. * This exemption does not extend to derived works not owned by * the Transmission project. * * $Id$ */ #include #include #include #include #include #include #include /* event.h needs this */ #include #include "transmission.h" #include "bencode.h" #include "completion.h" #include "inout.h" #include "list.h" #include "peer-io.h" #include "peer-mgr.h" #include "peer-mgr-private.h" #include "peer-msgs.h" #include "ratecontrol.h" #include "trevent.h" #include "utils.h" /** *** **/ #define MAX_ALLOWED_SET_COUNT 10 /* number of pieces generated for allow-fast, threshold for fast-allowing others */ enum { BT_CHOKE = 0, BT_UNCHOKE = 1, BT_INTERESTED = 2, BT_NOT_INTERESTED = 3, BT_HAVE = 4, BT_BITFIELD = 5, BT_REQUEST = 6, BT_PIECE = 7, BT_CANCEL = 8, BT_PORT = 9, BT_SUGGEST = 13, BT_HAVE_ALL = 14, BT_HAVE_NONE = 15, BT_REJECT = 16, BT_ALLOWED_FAST = 17, BT_LTEP = 20, LTEP_HANDSHAKE = 0, OUR_LTEP_PEX = 1, MAX_REQUEST_BYTE_COUNT = (16 * 1024), /* drop requests who want too much */ KEEPALIVE_INTERVAL_SECS = 90, /* idle seconds before we send a keepalive */ PEX_INTERVAL = (60 * 1000), /* msec between calls to sendPex() */ PEER_PULSE_INTERVAL = (50) , /* msec between calls to pulse() */ }; enum { AWAITING_BT_LENGTH, AWAITING_BT_MESSAGE, READING_BT_PIECE }; struct peer_request { uint32_t index; uint32_t offset; uint32_t length; time_t time_requested; }; static int peer_request_compare( const void * va, const void * vb ) { struct peer_request * a = (struct peer_request*) va; struct peer_request * b = (struct peer_request*) vb; if( a->index != b->index ) return a->index - b->index; if( a->offset != b->offset ) return a->offset - b->offset; if( a->length != b->length ) return a->length - b->length; return 0; } struct tr_peermsgs { tr_peer * info; tr_handle * handle; tr_torrent * torrent; tr_peerIo * io; tr_publisher_t * publisher; struct evbuffer * outMessages; /* buffer of all the non-piece messages */ struct evbuffer * inBlock; /* the block we're currently receiving */ tr_list * peerAskedFor; tr_list * clientAskedFor; tr_timer * pulseTimer; tr_timer * pexTimer; struct peer_request blockToUs; /* the block currntly being sent to us */ time_t lastReqAddedAt; time_t clientSentPexAt; time_t clientSentAnythingAt; unsigned int notListening : 1; unsigned int peerSupportsPex : 1; unsigned int clientSentLtepHandshake : 1; unsigned int peerSentLtepHandshake : 1; tr_bitfield * clientAllowedPieces; tr_bitfield * peerAllowedPieces; uint8_t state; uint8_t ut_pex_id; uint16_t pexCount; uint32_t incomingMessageLength; tr_pex * pex; }; /** *** **/ static void myDebug( const char * file, int line, const struct tr_peermsgs * msgs, const char * fmt, ... ) { FILE * fp = tr_getLog( ); if( fp != NULL ) { va_list args; const char * addr = tr_peerIoGetAddrStr( msgs->io ); struct evbuffer * buf = evbuffer_new( ); evbuffer_add_printf( buf, "[%s:%d] %s (%p) ", file, line, addr, msgs->io ); va_start( args, fmt ); evbuffer_add_vprintf( buf, fmt, args ); va_end( args ); fprintf( fp, "%s\n", EVBUFFER_DATA(buf) ); evbuffer_free( buf ); } } #define dbgmsg(handshake, fmt...) myDebug(__FILE__, __LINE__, handshake, ##fmt ) /** *** EVENTS **/ static const tr_peermsgs_event blankEvent = { 0, 0, 0, 0, 0.0f }; static void publish( tr_peermsgs * msgs, tr_peermsgs_event * e ) { tr_publisherPublish( msgs->publisher, msgs->info, e ); } static void fireGotError( tr_peermsgs * msgs ) { tr_peermsgs_event e = blankEvent; e.eventType = TR_PEERMSG_GOT_ERROR; publish( msgs, &e ); } static void fireNeedReq( tr_peermsgs * msgs ) { tr_peermsgs_event e = blankEvent; e.eventType = TR_PEERMSG_NEED_REQ; publish( msgs, &e ); } static void firePeerProgress( tr_peermsgs * msgs ) { tr_peermsgs_event e = blankEvent; e.eventType = TR_PEERMSG_PEER_PROGRESS; e.progress = msgs->info->progress; publish( msgs, &e ); } static void fireClientHave( tr_peermsgs * msgs, uint32_t pieceIndex ) { tr_peermsgs_event e = blankEvent; e.eventType = TR_PEERMSG_CLIENT_HAVE; e.pieceIndex = pieceIndex; publish( msgs, &e ); } static void fireGotBlock( tr_peermsgs * msgs, uint32_t pieceIndex, uint32_t offset, uint32_t length ) { tr_peermsgs_event e = blankEvent; e.eventType = TR_PEERMSG_CLIENT_BLOCK; e.pieceIndex = pieceIndex; e.offset = offset; e.length = length; publish( msgs, &e ); } /** *** INTEREST **/ static int isPieceInteresting( const tr_peermsgs * peer, int piece ) { const tr_torrent * torrent = peer->torrent; if( torrent->info.pieces[piece].dnd ) /* we don't want it */ return FALSE; if( tr_cpPieceIsComplete( torrent->completion, piece ) ) /* we already have it */ return FALSE; if( !tr_bitfieldHas( peer->info->have, piece ) ) /* peer doesn't have it */ return FALSE; if( tr_bitfieldHas( peer->info->banned, piece ) ) /* peer is banned for it */ return FALSE; return TRUE; } /* "interested" means we'll ask for piece data from the peer if they unchoke us */ static int isPeerInteresting( const tr_peermsgs * msgs ) { int i; const tr_torrent * torrent; const tr_bitfield * bitfield; const int clientIsSeed = tr_cpGetStatus( msgs->torrent->completion ) != TR_CP_INCOMPLETE; if( clientIsSeed ) return FALSE; torrent = msgs->torrent; bitfield = tr_cpPieceBitfield( torrent->completion ); if( !msgs->info->have ) /* We don't know what this peer has... what should this be? */ return TRUE; assert( bitfield->len == msgs->info->have->len ); for( i=0; iinfo.pieceCount; ++i ) if( isPieceInteresting( msgs, i ) ) return TRUE; return FALSE; } static void sendInterest( tr_peermsgs * msgs, int weAreInterested ) { assert( msgs != NULL ); assert( weAreInterested==0 || weAreInterested==1 ); msgs->info->clientIsInterested = weAreInterested; dbgmsg( msgs, ": sending an %s message", (weAreInterested ? "INTERESTED" : "NOT_INTERESTED") ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, weAreInterested ? BT_INTERESTED : BT_NOT_INTERESTED ); } static void updateInterest( tr_peermsgs * msgs ) { const int i = isPeerInteresting( msgs ); if( i != msgs->info->clientIsInterested ) sendInterest( msgs, i ); if( i ) fireNeedReq( msgs ); } void tr_peerMsgsSetChoke( tr_peermsgs * msgs, int choke ) { assert( msgs != NULL ); assert( msgs->info != NULL ); assert( choke==0 || choke==1 ); if( msgs->info->peerIsChoked != choke ) { msgs->info->peerIsChoked = choke; tr_list * walk; if( choke ) for( walk = msgs->peerAskedFor; walk != NULL; ) { tr_list * next = walk->next; /* We shouldn't reject a peer's fast allowed requests at choke */ struct peer_request *req = walk->data; if ( !tr_bitfieldHas( msgs->peerAllowedPieces, req->index ) ) { tr_list_remove_data( &msgs->peerAskedFor, req ); tr_free( req ); } walk = next; } dbgmsg( msgs, "sending a %s message", (choke ? "CHOKE" : "UNCHOKE") ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, choke ? BT_CHOKE : BT_UNCHOKE ); msgs->info->chokeChangedAt = time( NULL ); } } /** *** **/ void tr_peerMsgsCancel( tr_peermsgs * msgs, uint32_t pieceIndex, uint32_t offset, uint32_t length ) { tr_list * node; struct peer_request tmp; assert( msgs != NULL ); assert( length > 0 ); /* have we asked the peer for this piece? */ tmp.index = pieceIndex; tmp.offset = offset; tmp.length = length; node = tr_list_remove( &msgs->clientAskedFor, &tmp, peer_request_compare ); /* if so, send a cancel message */ if( node != NULL ) { tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) + 3*sizeof(uint32_t) ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, BT_CANCEL ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, pieceIndex ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, offset ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, length ); tr_free( node ); } } /** *** **/ void tr_peerMsgsHave( tr_peermsgs * msgs, uint32_t pieceIndex ) { dbgmsg( msgs, "w00t telling them we HAVE piece #%d", pieceIndex ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) + sizeof(uint32_t) ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, BT_HAVE ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, pieceIndex ); updateInterest( msgs ); } #if 0 static void sendFastSuggest( tr_peermsgs * msgs, uint32_t pieceIndex ) { dbgmsg( msgs, "w00t SUGGESTing them piece #%d", pieceIndex ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) + sizeof(uint32_t) ); tr_peerIoWriteUint8( msgs->io, msgs->outMessages, BT_SUGGEST ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, pieceIndex ); updateInterest( msgs ); } #endif static void sendFastHave( tr_peermsgs * msgs, int all) { dbgmsg( msgs, "w00t telling them we %s pieces", (all ? "HAVE_ALL" : "HAVE_NONE" ) ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) ); tr_peerIoWriteUint8( msgs->io, msgs->outMessages, ( all ? BT_HAVE_ALL : BT_HAVE_NONE ) ); updateInterest( msgs ); } static void sendFastReject( tr_peermsgs * msgs, uint32_t pieceIndex, uint32_t offset, uint32_t length ) { assert( msgs != NULL ); assert( length > 0 ); /* reject the request */ const uint32_t len = sizeof(uint8_t) + 3 * sizeof(uint32_t); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, len ); tr_peerIoWriteUint8( msgs->io, msgs->outMessages, BT_REJECT ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, pieceIndex ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, offset ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, length ); } static void sendFastAllowed( tr_peermsgs * msgs, uint32_t pieceIndex) { dbgmsg( msgs, "w00t telling them we ALLOW_FAST piece #%d", pieceIndex ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) + sizeof(uint32_t) ); tr_peerIoWriteUint8( msgs->io, msgs->outMessages, BT_ALLOWED_FAST ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, pieceIndex ); } static void sendFastAllowedSet( tr_peermsgs * msgs ) { int i = 0; while (i <= msgs->torrent->info.pieceCount ) { if ( tr_bitfieldHas( msgs->peerAllowedPieces, i) ) sendFastAllowed( msgs, i ); i++; } } /** *** **/ static int pulse( void * vmsgs ); static int reqIsValid( const tr_peermsgs * msgs, uint32_t index, uint32_t offset, uint32_t length ) { const tr_torrent * tor = msgs->torrent; if( index >= (uint32_t) tor->info.pieceCount ) return FALSE; if ( (int)offset >= tr_torPieceCountBytes( tor, (int)index ) ) return FALSE; if( length > MAX_REQUEST_BYTE_COUNT ) return FALSE; if( tr_pieceOffset( tor, index, offset, length ) > tor->info.totalSize ) return FALSE; return TRUE; } static int requestIsValid( const tr_peermsgs * msgs, struct peer_request * req ) { return reqIsValid( msgs, req->index, req->offset, req->length ); } int tr_peerMsgsAddRequest( tr_peermsgs * msgs, uint32_t index, uint32_t offset, uint32_t length ) { struct peer_request * req; int maxSize; assert( msgs != NULL ); assert( msgs->torrent != NULL ); assert( reqIsValid( msgs, index, offset, length ) ); if( msgs->info->clientIsChoked ) return TR_ADDREQ_CLIENT_CHOKED; if( !tr_bitfieldHas( msgs->info->have, index ) ) return TR_ADDREQ_MISSING; maxSize = MIN( 3 + (int)(tr_rcRate(msgs->info->rateToClient)/10), 100 ); if( tr_list_size( msgs->clientAskedFor) >= maxSize ) return TR_ADDREQ_FULL; dbgmsg( msgs, "w00t peer has a max request queue size of %d... adding request for piece %d, offset %d", maxSize, (int)index, (int)offset ); /* queue the request */ tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) + 3*sizeof(uint32_t) ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, BT_REQUEST ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, index ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, offset ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, length ); /* add it to our `requests sent' list */ req = tr_new( struct peer_request, 1 ); req->index = index; req->offset = offset; req->length = length; req->time_requested = msgs->lastReqAddedAt = time( NULL ); tr_list_append( &msgs->clientAskedFor, req ); pulse( msgs ); return TR_ADDREQ_OK; } /** *** **/ static void sendLtepHandshake( tr_peermsgs * msgs ) { benc_val_t val, *m; char * buf; int len; int pex; const char * v = TR_NAME " " USERAGENT_PREFIX; const int port = tr_getPublicPort( msgs->handle ); struct evbuffer * outbuf; if( msgs->clientSentLtepHandshake ) return; outbuf = evbuffer_new( ); dbgmsg( msgs, "sending an ltep handshake" ); msgs->clientSentLtepHandshake = 1; /* decide if we want to advertise pex support */ if( msgs->torrent->pexDisabled ) pex = 0; else if( msgs->peerSentLtepHandshake ) pex = msgs->peerSupportsPex ? 1 : 0; else pex = 1; tr_bencInit( &val, TYPE_DICT ); tr_bencDictReserve( &val, 4 ); tr_bencInitInt( tr_bencDictAdd( &val, "e" ), 1 ); m = tr_bencDictAdd( &val, "m" ); tr_bencInit( m, TYPE_DICT ); if( pex ) { tr_bencDictReserve( m, 1 ); tr_bencInitInt( tr_bencDictAdd( m, "ut_pex" ), OUR_LTEP_PEX ); } if( port > 0 ) tr_bencInitInt( tr_bencDictAdd( &val, "p" ), port ); tr_bencInitStr( tr_bencDictAdd( &val, "v" ), v, 0, 1 ); buf = tr_bencSaveMalloc( &val, &len ); tr_peerIoWriteUint32( msgs->io, outbuf, 2*sizeof(uint8_t) + len ); tr_peerIoWriteUint8 ( msgs->io, outbuf, BT_LTEP ); tr_peerIoWriteUint8 ( msgs->io, outbuf, LTEP_HANDSHAKE ); tr_peerIoWriteBytes ( msgs->io, outbuf, buf, len ); tr_peerIoWriteBuf( msgs->io, outbuf ); #if 0 dbgmsg( msgs, "here is the ltep handshake we sent:" ); tr_bencPrint( &val ); #endif /* cleanup */ tr_bencFree( &val ); tr_free( buf ); evbuffer_free( outbuf ); } static void parseLtepHandshake( tr_peermsgs * msgs, int len, struct evbuffer * inbuf ) { benc_val_t val, * sub; uint8_t * tmp = tr_new( uint8_t, len ); tr_peerIoReadBytes( msgs->io, inbuf, tmp, len ); msgs->peerSentLtepHandshake = 1; if( tr_bencLoad( tmp, len, &val, NULL ) || val.type!=TYPE_DICT ) { dbgmsg( msgs, "GET extended-handshake, couldn't get dictionary" ); tr_free( tmp ); return; } #if 0 dbgmsg( msgs, "here is the ltep handshake we read:" ); tr_bencPrint( &val ); #endif /* does the peer prefer encrypted connections? */ sub = tr_bencDictFind( &val, "e" ); if( tr_bencIsInt( sub ) ) msgs->info->encryption_preference = sub->val.i ? ENCRYPTION_PREFERENCE_YES : ENCRYPTION_PREFERENCE_NO; /* check supported messages for utorrent pex */ sub = tr_bencDictFind( &val, "m" ); if( tr_bencIsDict( sub ) ) { sub = tr_bencDictFind( sub, "ut_pex" ); if( tr_bencIsInt( sub ) ) { msgs->peerSupportsPex = 1; msgs->ut_pex_id = (uint8_t) sub->val.i; dbgmsg( msgs, "msgs->ut_pex is %d", (int)msgs->ut_pex_id ); } } #if 0 /* get peer's client name */ sub = tr_bencDictFind( &val, "v" ); if( tr_bencIsStr( sub ) ) { int i; tr_free( msgs->info->client ); fprintf( stderr, "dictionary says client is [%s]\n", sub->val.s.s ); msgs->info->client = tr_strndup( sub->val.s.s, sub->val.s.i ); for( i=0; ival.s.i; ++i ) { fprintf( stderr, "[%c] (%d)\n", sub->val.s.s[i], (int)sub->val.s.s[i] ); if( (int)msgs->info->client[i]==-75 ) msgs->info->client[i]='u'; } fprintf( stderr, "msgs->client is now [%s]\n", msgs->info->client ); } #endif /* get peer's listening port */ sub = tr_bencDictFind( &val, "p" ); if( tr_bencIsInt( sub ) ) { msgs->info->port = htons( (uint16_t)sub->val.i ); dbgmsg( msgs, "msgs->port is now %hu", msgs->info->port ); } tr_bencFree( &val ); tr_free( tmp ); } static void parseUtPex( tr_peermsgs * msgs, int msglen, struct evbuffer * inbuf ) { benc_val_t val, * sub; uint8_t * tmp; if( msgs->torrent->pexDisabled ) /* no sharing! */ return; tmp = tr_new( uint8_t, msglen ); tr_peerIoReadBytes( msgs->io, inbuf, tmp, msglen ); if( tr_bencLoad( tmp, msglen, &val, NULL ) || !tr_bencIsDict( &val ) ) { dbgmsg( msgs, "GET can't read extended-pex dictionary" ); tr_free( tmp ); return; } sub = tr_bencDictFind( &val, "added" ); if( tr_bencIsStr(sub) && ((sub->val.s.i % 6) == 0)) { const int n = sub->val.s.i / 6 ; dbgmsg( msgs, "got %d peers from uT pex", n ); tr_peerMgrAddPeers( msgs->handle->peerMgr, msgs->torrent->info.hash, TR_PEER_FROM_PEX, (uint8_t*)sub->val.s.s, n ); } tr_bencFree( &val ); tr_free( tmp ); } static void sendPex( tr_peermsgs * msgs ); static void parseLtep( tr_peermsgs * msgs, int msglen, struct evbuffer * inbuf ) { uint8_t ltep_msgid; tr_peerIoReadUint8( msgs->io, inbuf, <ep_msgid ); msglen--; if( ltep_msgid == LTEP_HANDSHAKE ) { dbgmsg( msgs, "got ltep handshake" ); parseLtepHandshake( msgs, msglen, inbuf ); sendLtepHandshake( msgs ); sendPex( msgs ); } else if( ltep_msgid == msgs->ut_pex_id ) { dbgmsg( msgs, "got ut pex" ); msgs->peerSupportsPex = 1; parseUtPex( msgs, msglen, inbuf ); } else { dbgmsg( msgs, "skipping unknown ltep message (%d)", (int)ltep_msgid ); evbuffer_drain( inbuf, msglen ); } } static int readBtLength( tr_peermsgs * msgs, struct evbuffer * inbuf ) { uint32_t len; const size_t needlen = sizeof(uint32_t); if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; tr_peerIoReadUint32( msgs->io, inbuf, &len ); if( len == 0 ) /* peer sent us a keepalive message */ dbgmsg( msgs, "peer sent us a keepalive message..." ); else { dbgmsg( msgs, "peer is sending us a message with %"PRIu64" bytes...", (uint64_t)len ); msgs->incomingMessageLength = len; msgs->state = AWAITING_BT_MESSAGE; } return READ_AGAIN; } static int readBtMessage( tr_peermsgs * msgs, struct evbuffer * inbuf ) { uint8_t id; uint32_t ui32; uint32_t msglen = msgs->incomingMessageLength; if( EVBUFFER_LENGTH(inbuf) < msglen ) return READ_MORE; tr_peerIoReadUint8( msgs->io, inbuf, &id ); msglen--; dbgmsg( msgs, "peer sent us a message... " "bt id number is %d, and remaining len is %d", (int)id, (int)msglen ); switch( id ) { case BT_CHOKE: dbgmsg( msgs, "w00t peer sent us a BT_CHOKE" ); assert( msglen == 0 ); msgs->info->clientIsChoked = 1; tr_list * walk; for( walk = msgs->peerAskedFor; walk != NULL; ) { tr_list * next = walk->next; /* We shouldn't reject a peer's fast allowed requests at choke */ struct peer_request *req = walk->data; if ( !tr_bitfieldHas( msgs->peerAllowedPieces, req->index ) ) { tr_list_remove_data( &msgs->peerAskedFor, req ); tr_free( req ); } walk = next; } tr_list_free( &msgs->clientAskedFor, tr_free ); break; case BT_UNCHOKE: dbgmsg( msgs, "w00t peer sent us a BT_UNCHOKE" ); assert( msglen == 0 ); msgs->info->clientIsChoked = 0; fireNeedReq( msgs ); break; case BT_INTERESTED: dbgmsg( msgs, "w00t peer sent us a BT_INTERESTED" ); assert( msglen == 0 ); msgs->info->peerIsInterested = 1; tr_peerMsgsSetChoke( msgs, 0 ); break; case BT_NOT_INTERESTED: dbgmsg( msgs, "w00t peer sent us a BT_NOT_INTERESTED" ); assert( msglen == 0 ); msgs->info->peerIsInterested = 0; break; case BT_HAVE: dbgmsg( msgs, "w00t peer sent us a BT_HAVE" ); assert( msglen == 4 ); tr_peerIoReadUint32( msgs->io, inbuf, &ui32 ); tr_bitfieldAdd( msgs->info->have, ui32 ); msgs->info->progress = tr_bitfieldCountTrueBits( msgs->info->have ) / (float)msgs->torrent->info.pieceCount; dbgmsg( msgs, "after the HAVE message, peer progress is %f", msgs->info->progress ); updateInterest( msgs ); firePeerProgress( msgs ); break; case BT_BITFIELD: { const int clientIsSeed = tr_cpGetStatus( msgs->torrent->completion ) != TR_CP_INCOMPLETE; dbgmsg( msgs, "w00t peer sent us a BT_BITFIELD" ); assert( msglen == msgs->info->have->len ); tr_peerIoReadBytes( msgs->io, inbuf, msgs->info->have->bits, msglen ); msgs->info->progress = tr_bitfieldCountTrueBits( msgs->info->have ) / (float)msgs->torrent->info.pieceCount; tr_peerMsgsSetChoke( msgs, !clientIsSeed || (msgs->info->progress<1.0) ); dbgmsg( msgs, "after the HAVE message, peer progress is %f", msgs->info->progress ); updateInterest( msgs ); fireNeedReq( msgs ); firePeerProgress( msgs ); break; } case BT_REQUEST: { struct peer_request * req; dbgmsg( msgs, "peer sent us a BT_REQUEST" ); assert( msglen == 12 ); req = tr_new( struct peer_request, 1 ); tr_peerIoReadUint32( msgs->io, inbuf, &req->index ); tr_peerIoReadUint32( msgs->io, inbuf, &req->offset ); tr_peerIoReadUint32( msgs->io, inbuf, &req->length ); if ( !requestIsValid( msgs, req ) ) { dbgmsg( msgs, "BT_REQUEST: invalid request, ignoring" ); tr_free( req ); break; } /* If we're not choking him -> continue If we're choking him it doesn't support FPE -> He's deaf, reCHOKE and bail... it support FPE If the asked piece is not allowed OR he's above our threshold OR we don't have the requested piece -> Reject Else Asked piece allowed AND he's below our threshold -> continue... */ if ( msgs->info->peerIsChoked ) { if ( !tr_peerIoSupportsFEXT( msgs->io ) ) { dbgmsg( msgs, "BT_REQUEST: peer is choked, ignoring" ); /* Didn't he get it? */ tr_peerMsgsSetChoke( msgs, 1 ); tr_free( req ); break; } else { if ( !tr_bitfieldHas( msgs->peerAllowedPieces, req->index ) || ( msgs->info->progress * (float)msgs->torrent->info.pieceCount) >= MAX_ALLOWED_SET_COUNT || !tr_cpPieceIsComplete( msgs->torrent->completion, req->index ) ) { dbgmsg( msgs, "BT_REQUEST: peer requests an un-fastallowed piece" ); sendFastReject( msgs, req->index, req->offset, req->length ); tr_free( req ); break; } dbgmsg( msgs, "BT_REQUEST: fast allowed piece, accepting request" ); } } tr_list_append( &msgs->peerAskedFor, req ); break; } case BT_CANCEL: { struct peer_request req; void * data; dbgmsg( msgs, "peer sent us a BT_CANCEL" ); assert( msglen == 12 ); tr_peerIoReadUint32( msgs->io, inbuf, &req.index ); tr_peerIoReadUint32( msgs->io, inbuf, &req.offset ); tr_peerIoReadUint32( msgs->io, inbuf, &req.length ); data = tr_list_remove( &msgs->peerAskedFor, &req, peer_request_compare ); tr_free( data ); break; } case BT_PIECE: { dbgmsg( msgs, "peer sent us a BT_PIECE" ); assert( msgs->blockToUs.length == 0 ); tr_peerIoReadUint32( msgs->io, inbuf, &msgs->blockToUs.index ); tr_peerIoReadUint32( msgs->io, inbuf, &msgs->blockToUs.offset ); msgs->blockToUs.length = msglen - 8; assert( EVBUFFER_LENGTH(msgs->inBlock) == 0 ); //evbuffer_drain( msgs->inBlock, ~0 ); msgs->state = msgs->blockToUs.length ? READING_BT_PIECE : AWAITING_BT_LENGTH; return READ_AGAIN; break; } case BT_PORT: { dbgmsg( msgs, "peer sent us a BT_PORT" ); assert( msglen == 2 ); tr_peerIoReadUint16( msgs->io, inbuf, &msgs->info->port ); break; } case BT_SUGGEST: { /* tiennou TODO */ break; } case BT_HAVE_ALL: { assert( msglen == 0 ); dbgmsg( msgs, "peer sent us a BT_HAVE_ALL" ); memset( msgs->info->have->bits, 1, msgs->info->have->len ); msgs->info->progress = tr_bitfieldCountTrueBits( msgs->info->have ) / (float)msgs->torrent->info.pieceCount; dbgmsg( msgs, "after the HAVE_ALL message, peer progress is %f", msgs->info->progress ); updateInterest( msgs ); firePeerProgress( msgs ); break; } case BT_HAVE_NONE: { assert( msglen == 0 ); dbgmsg( msgs, "peer sent us a BT_HAVE_NONE" ); memset( msgs->info->have->bits, 1, msgs->info->have->len ); msgs->info->progress = tr_bitfieldCountTrueBits( msgs->info->have ) / (float)msgs->torrent->info.pieceCount; dbgmsg( msgs, "after the HAVE_NONE message, peer progress is %f", msgs->info->progress ); updateInterest( msgs ); firePeerProgress( msgs ); break; } case BT_REJECT: { struct peer_request req; tr_list * node; assert( msglen == 12 ); dbgmsg( msgs, "peer sent us a BT_REJECT" ); tr_peerIoReadUint32( msgs->io, inbuf, &req.index ); tr_peerIoReadUint32( msgs->io, inbuf, &req.offset ); tr_peerIoReadUint32( msgs->io, inbuf, &req.length ); node = tr_list_find( msgs->peerAskedFor, &req, peer_request_compare ); if( node != NULL ) { void * data = node->data; tr_list_remove_data( &msgs->peerAskedFor, data ); tr_free( data ); dbgmsg( msgs, "found the req that peer has rejected... cancelled." ); } break; } case BT_ALLOWED_FAST: { assert( msglen == 4 ); dbgmsg( msgs, "peer sent us a BT_ALLOWED_FAST" ); tr_peerIoReadUint32( msgs->io, inbuf, &ui32 ); tr_bitfieldAdd( msgs->clientAllowedPieces, ui32 ); break; } case BT_LTEP: dbgmsg( msgs, "peer sent us a BT_LTEP" ); parseLtep( msgs, msglen, inbuf ); break; default: dbgmsg( msgs, "peer sent us an UNKNOWN: %d", (int)id ); tr_peerIoDrain( msgs->io, inbuf, msglen ); assert( 0 ); } msgs->incomingMessageLength = -1; msgs->state = AWAITING_BT_LENGTH; return READ_AGAIN; } static void clientGotBytes( tr_peermsgs * msgs, uint32_t byteCount ) { tr_torrent * tor = msgs->torrent; tor->activityDate = tr_date( ); tor->downloadedCur += byteCount; msgs->info->pieceDataActivityDate = time( NULL ); tr_rcTransferred( msgs->info->rateToClient, byteCount ); tr_rcTransferred( tor->download, byteCount ); tr_rcTransferred( tor->handle->download, byteCount ); } static void peerGotBytes( tr_peermsgs * msgs, uint32_t byteCount ) { tr_torrent * tor = msgs->torrent; tor->activityDate = tr_date( ); tor->uploadedCur += byteCount; msgs->info->pieceDataActivityDate = time( NULL ); tr_rcTransferred( msgs->info->rateToPeer, byteCount ); tr_rcTransferred( tor->upload, byteCount ); tr_rcTransferred( tor->handle->upload, byteCount ); } static int canDownload( const tr_peermsgs * msgs ) { tr_torrent * tor = msgs->torrent; if( tor->downloadLimitMode == TR_SPEEDLIMIT_GLOBAL ) return !tor->handle->useDownloadLimit || tr_rcCanTransfer( tor->handle->download ); if( tor->downloadLimitMode == TR_SPEEDLIMIT_SINGLE ) return tr_rcCanTransfer( tor->download ); return TRUE; } static void reassignBytesToCorrupt( tr_peermsgs * msgs, uint32_t byteCount ) { tr_torrent * tor = msgs->torrent; /* increment the `corrupt' field */ tor->corruptCur += byteCount; /* decrement the `downloaded' field */ if( tor->downloadedCur >= byteCount ) tor->downloadedCur -= byteCount; else tor->downloadedCur = 0; } static void gotBadPiece( tr_peermsgs * msgs, uint32_t pieceIndex ) { const uint32_t byteCount = tr_torPieceCountBytes( msgs->torrent, (int)pieceIndex ); reassignBytesToCorrupt( msgs, byteCount ); } static void gotUnwantedBlock( tr_peermsgs * msgs, uint32_t index UNUSED, uint32_t offset UNUSED, uint32_t length ) { reassignBytesToCorrupt( msgs, length ); } static void addUsToBlamefield( tr_peermsgs * msgs, uint32_t index ) { if( !msgs->info->blame ) msgs->info->blame = tr_bitfieldNew( msgs->torrent->info.pieceCount ); tr_bitfieldAdd( msgs->info->blame, index ); } static void gotBlock( tr_peermsgs * msgs, struct evbuffer * inbuf, uint32_t index, uint32_t offset, uint32_t length ) { tr_torrent * tor = msgs->torrent; const int block = _tr_block( tor, index, offset ); struct peer_request key, *req; /** *** Remove the block from our `we asked for this' list **/ key.index = index; key.offset = offset; key.length = length; req = (struct peer_request*) tr_list_remove( &msgs->clientAskedFor, &key, peer_request_compare ); if( req == NULL ) { gotUnwantedBlock( msgs, index, offset, length ); dbgmsg( msgs, "we didn't ask for this message..." ); return; } dbgmsg( msgs, "w00t peer sent us a block. turnaround time was %d seconds", (int)(time(NULL) - req->time_requested) ); tr_free( req ); dbgmsg( msgs, "peer has %d more blocks we've asked for", tr_list_size(msgs->clientAskedFor)); /** *** Error checks **/ if( tr_cpBlockIsComplete( tor->completion, block ) ) { dbgmsg( msgs, "have this block already..." ); tr_dbg( "have this block already..." ); gotUnwantedBlock( msgs, index, offset, length ); return; } if( (int)length != tr_torBlockCountBytes( tor, block ) ) { dbgmsg( msgs, "block is the wrong length..." ); tr_dbg( "block is the wrong length..." ); gotUnwantedBlock( msgs, index, offset, length ); return; } /** *** Write the block **/ if( tr_ioWrite( tor, index, offset, length, EVBUFFER_DATA( inbuf ))) { return; } tr_cpBlockAdd( tor->completion, block ); addUsToBlamefield( msgs, index ); fireGotBlock( msgs, index, offset, length ); fireNeedReq( msgs ); /** *** Handle if this was the last block in the piece **/ if( tr_cpPieceIsComplete( tor->completion, index ) ) { if( tr_ioHash( tor, index ) ) { gotBadPiece( msgs, index ); return; } fireClientHave( msgs, index ); } } static ReadState readBtPiece( tr_peermsgs * msgs, struct evbuffer * inbuf ) { uint32_t inlen; uint8_t * tmp; assert( msgs != NULL ); assert( msgs->blockToUs.length > 0 ); assert( inbuf != NULL ); assert( EVBUFFER_LENGTH( inbuf ) > 0 ); /* read from the inbuf into our block buffer */ inlen = MIN( EVBUFFER_LENGTH(inbuf), msgs->blockToUs.length ); tmp = tr_new( uint8_t, inlen ); tr_peerIoReadBytes( msgs->io, inbuf, tmp, inlen ); evbuffer_add( msgs->inBlock, tmp, inlen ); /* update our tables accordingly */ assert( inlen >= msgs->blockToUs.length ); msgs->blockToUs.length -= inlen; msgs->info->peerSentPieceDataAt = time( NULL ); clientGotBytes( msgs, inlen ); /* if this was the entire block, save it */ if( !msgs->blockToUs.length ) { dbgmsg( msgs, "w00t -- got block index %u, offset %u", msgs->blockToUs.index, msgs->blockToUs.offset ); assert( (int)EVBUFFER_LENGTH( msgs->inBlock ) == tr_torBlockCountBytes( msgs->torrent, _tr_block(msgs->torrent,msgs->blockToUs.index, msgs->blockToUs.offset) ) ); gotBlock( msgs, msgs->inBlock, msgs->blockToUs.index, msgs->blockToUs.offset, EVBUFFER_LENGTH( msgs->inBlock ) ); evbuffer_drain( msgs->inBlock, ~0 ); msgs->state = AWAITING_BT_LENGTH; } /* cleanup */ tr_free( tmp ); return READ_AGAIN; } static ReadState canRead( struct bufferevent * evin, void * vmsgs ) { ReadState ret; tr_peermsgs * msgs = (tr_peermsgs *) vmsgs; struct evbuffer * inbuf = EVBUFFER_INPUT ( evin ); if( !canDownload( msgs ) ) { msgs->notListening = 1; tr_peerIoSetIOMode ( msgs->io, 0, EV_READ ); ret = READ_DONE; } else switch( msgs->state ) { case AWAITING_BT_LENGTH: ret = readBtLength ( msgs, inbuf ); break; case AWAITING_BT_MESSAGE: ret = readBtMessage ( msgs, inbuf ); break; case READING_BT_PIECE: ret = readBtPiece ( msgs, inbuf ); break; default: assert( 0 ); } return ret; } static void sendKeepalive( tr_peermsgs * msgs ) { dbgmsg( msgs, "sending a keepalive message" ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, 0 ); } /** *** **/ static int canWrite( const tr_peermsgs * msgs ) { /* don't let our outbuffer get too large */ if( tr_peerIoWriteBytesWaiting( msgs->io ) > 8192 ) return FALSE; return TRUE; } static int canUpload( const tr_peermsgs * msgs ) { const tr_torrent * tor = msgs->torrent; if( !canWrite( msgs ) ) return FALSE; if( tor->uploadLimitMode == TR_SPEEDLIMIT_GLOBAL ) return !tor->handle->useUploadLimit || tr_rcCanTransfer( tor->handle->upload ); if( tor->uploadLimitMode == TR_SPEEDLIMIT_SINGLE ) return tr_rcCanTransfer( tor->upload ); return TRUE; } static int pulse( void * vmsgs ) { const time_t now = time( NULL ); tr_peermsgs * msgs = (tr_peermsgs *) vmsgs; size_t len; /* if we froze out a downloaded block because of speed limits, start listening to the peer again */ if( msgs->notListening && canDownload( msgs ) ) { msgs->notListening = 0; tr_peerIoSetIOMode ( msgs->io, EV_READ, 0 ); } if( !canWrite( msgs ) ) { } else if(( len = EVBUFFER_LENGTH( msgs->outMessages ) )) { tr_peerIoWriteBuf( msgs->io, msgs->outMessages ); msgs->clientSentAnythingAt = now; } else if(( msgs->peerAskedFor )) { if( canUpload( msgs ) ) { struct peer_request * r = tr_list_pop_front( &msgs->peerAskedFor ); uint8_t * tmp = tr_new( uint8_t, r->length ); const uint32_t msglen = sizeof(uint8_t) + 2*sizeof(uint32_t) + r->length; struct evbuffer * out = evbuffer_new( ); assert( requestIsValid( msgs, r ) ); tr_peerIoWriteUint32( msgs->io, out, msglen ); tr_peerIoWriteUint8 ( msgs->io, out, BT_PIECE ); tr_peerIoWriteUint32( msgs->io, out, r->index ); tr_peerIoWriteUint32( msgs->io, out, r->offset ); tr_peerIoWriteBuf( msgs->io, out ); tr_ioRead( msgs->torrent, r->index, r->offset, r->length, tmp ); tr_peerIoWrite( msgs->io, tmp, r->length ); peerGotBytes( msgs, r->length ); dbgmsg( msgs, "putting req into out queue: index %d, offset %d, length %d ... %d blocks left in our queue", (int)r->index, (int)r->offset, (int)r->length, tr_list_size(msgs->peerAskedFor) ); tr_free( r ); tr_free( tmp ); evbuffer_free( out ); } } else if( ( now - msgs->clientSentAnythingAt ) > KEEPALIVE_INTERVAL_SECS ) { sendKeepalive( msgs ); } return TRUE; /* loop forever */ } static void didWrite( struct bufferevent * evin UNUSED, void * vmsgs ) { pulse( vmsgs ); } static void gotError( struct bufferevent * evbuf UNUSED, short what UNUSED, void * vpeer ) { fireGotError( vpeer ); } static void sendBitfield( tr_peermsgs * msgs ) { const tr_bitfield * bitfield = tr_cpPieceBitfield( msgs->torrent->completion ); dbgmsg( msgs, "sending peer a bitfield message" ); tr_peerIoWriteUint32( msgs->io, msgs->outMessages, sizeof(uint8_t) + bitfield->len ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, BT_BITFIELD ); tr_peerIoWriteBytes ( msgs->io, msgs->outMessages, bitfield->bits, bitfield->len ); } /** *** **/ /* some peers give us error messages if we send more than this many peers in a single pex message */ #define MAX_PEX_DIFFS 200 typedef struct { tr_pex * added; tr_pex * dropped; tr_pex * elements; int addedCount; int droppedCount; int elementCount; int diffCount; } PexDiffs; static void pexAddedCb( void * vpex, void * userData ) { PexDiffs * diffs = (PexDiffs *) userData; tr_pex * pex = (tr_pex *) vpex; if( diffs->diffCount < MAX_PEX_DIFFS ) { diffs->diffCount++; diffs->added[diffs->addedCount++] = *pex; diffs->elements[diffs->elementCount++] = *pex; } } static void pexRemovedCb( void * vpex, void * userData ) { PexDiffs * diffs = (PexDiffs *) userData; tr_pex * pex = (tr_pex *) vpex; if( diffs->diffCount < MAX_PEX_DIFFS ) { diffs->diffCount++; diffs->dropped[diffs->droppedCount++] = *pex; } } static void pexElementCb( void * vpex, void * userData ) { PexDiffs * diffs = (PexDiffs *) userData; tr_pex * pex = (tr_pex *) vpex; if( diffs->diffCount < MAX_PEX_DIFFS ) { diffs->diffCount++; diffs->elements[diffs->elementCount++] = *pex; } } static void sendPex( tr_peermsgs * msgs ) { if( msgs->peerSupportsPex && !msgs->torrent->pexDisabled ) { int i; tr_pex * newPex = NULL; const int newCount = tr_peerMgrGetPeers( msgs->handle->peerMgr, msgs->torrent->info.hash, &newPex ); PexDiffs diffs; benc_val_t val, *added, *dropped, *flags; uint8_t *tmp, *walk; char * benc; int bencLen; /* build the diffs */ diffs.added = tr_new( tr_pex, newCount ); diffs.addedCount = 0; diffs.dropped = tr_new( tr_pex, msgs->pexCount ); diffs.droppedCount = 0; diffs.elements = tr_new( tr_pex, newCount + msgs->pexCount ); diffs.elementCount = 0; diffs.diffCount = 0; tr_set_compare( msgs->pex, msgs->pexCount, newPex, newCount, tr_pexCompare, sizeof(tr_pex), pexRemovedCb, pexAddedCb, pexElementCb, &diffs ); dbgmsg( msgs, "pex: old peer count %d, new peer count %d, added %d, removed %d", msgs->pexCount, newCount, diffs.addedCount, diffs.droppedCount ); /* update peer */ tr_free( msgs->pex ); msgs->pex = diffs.elements; msgs->pexCount = diffs.elementCount; /* build the pex payload */ tr_bencInit( &val, TYPE_DICT ); tr_bencDictReserve( &val, 3 ); /* "added" */ added = tr_bencDictAdd( &val, "added" ); tmp = walk = tr_new( uint8_t, diffs.addedCount * 6 ); for( i=0; i> -->> flag is %d", (int)diffs.added[i].flags ); *walk++ = diffs.added[i].flags; } assert( ( walk - tmp ) == diffs.addedCount ); tr_bencInitStr( flags, tmp, walk-tmp, FALSE ); /* "dropped" */ dropped = tr_bencDictAdd( &val, "dropped" ); tmp = walk = tr_new( uint8_t, diffs.droppedCount * 6 ); for( i=0; iio, msgs->outMessages, 2*sizeof(uint8_t) + bencLen ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, BT_LTEP ); tr_peerIoWriteUint8 ( msgs->io, msgs->outMessages, OUR_LTEP_PEX ); tr_peerIoWriteBytes ( msgs->io, msgs->outMessages, benc, bencLen ); /* cleanup */ tr_free( benc ); tr_bencFree( &val ); tr_free( diffs.added ); tr_free( diffs.dropped ); tr_free( newPex ); msgs->clientSentPexAt = time( NULL ); } } static int pexPulse( void * vpeer ) { sendPex( vpeer ); return TRUE; } /** *** **/ tr_peermsgs* tr_peerMsgsNew( struct tr_torrent * torrent, struct tr_peer * info ) { tr_peermsgs * msgs; assert( info != NULL ); assert( info->io != NULL ); msgs = tr_new0( tr_peermsgs, 1 ); msgs->publisher = tr_publisherNew( ); msgs->info = info; msgs->handle = torrent->handle; msgs->torrent = torrent; msgs->io = info->io; msgs->info->clientIsChoked = 1; msgs->info->peerIsChoked = 1; msgs->info->clientIsInterested = 0; msgs->info->peerIsInterested = 0; msgs->info->have = tr_bitfieldNew( torrent->info.pieceCount ); msgs->pulseTimer = tr_timerNew( msgs->handle, pulse, msgs, PEER_PULSE_INTERVAL ); msgs->pexTimer = tr_timerNew( msgs->handle, pexPulse, msgs, PEX_INTERVAL ); msgs->outMessages = evbuffer_new( ); msgs->inBlock = evbuffer_new( ); msgs->peerAllowedPieces = NULL; msgs->clientAllowedPieces = NULL; if ( tr_peerIoSupportsFEXT( msgs->io ) ) { /* This peer is fastpeer-enabled, generate its allowed set * (before registering our callbacks) */ if ( !msgs->peerAllowedPieces ) { const struct in_addr *peerAddr = tr_peerIoGetAddress( msgs->io, NULL ); msgs->peerAllowedPieces = tr_peerMgrGenerateAllowedSet( MAX_ALLOWED_SET_COUNT, msgs->torrent->info.pieceCount, msgs->torrent->info.hash, peerAddr ); } msgs->clientAllowedPieces = tr_bitfieldNew( msgs->torrent->info.pieceCount ); } tr_peerIoSetIOFuncs( msgs->io, canRead, didWrite, gotError, msgs ); tr_peerIoSetIOMode( msgs->io, EV_READ|EV_WRITE, 0 ); /** *** If we initiated this connection, *** we may need to send LTEP/AZMP handshakes. *** Otherwise we'll wait for the peer to send theirs first. **/ if( !tr_peerIoIsIncoming( msgs->io ) ) { if ( tr_peerIoSupportsLTEP( msgs->io ) ) { sendLtepHandshake( msgs ); } else if ( tr_peerIoSupportsAZMP( msgs->io ) ) { dbgmsg( msgs, "FIXME: need to send AZMP handshake" ); } else { /* no-op */ } } if ( tr_peerIoSupportsFEXT( msgs->io ) ) { /* This peer is fastpeer-enabled, send it have-all or have-none if appropriate */ float completion = tr_cpPercentComplete( msgs->torrent->completion ); if ( completion == 0.0f ) { sendFastHave( msgs, 0 ); } else if ( completion == 1.0f ) { sendFastHave( msgs, 1 ); } else { sendBitfield( msgs ); } uint32_t peerProgress = msgs->torrent->info.pieceCount * msgs->info->progress; if ( peerProgress < MAX_ALLOWED_SET_COUNT ) sendFastAllowedSet( msgs ); } else { sendBitfield( msgs ); } return msgs; } void tr_peerMsgsFree( tr_peermsgs* msgs ) { if( msgs != NULL ) { tr_timerFree( &msgs->pulseTimer ); tr_timerFree( &msgs->pexTimer ); tr_publisherFree( &msgs->publisher ); tr_list_free( &msgs->clientAskedFor, tr_free ); tr_list_free( &msgs->peerAskedFor, tr_free ); evbuffer_free( msgs->outMessages ); evbuffer_free( msgs->inBlock ); tr_free( msgs->pex ); msgs->pexCount = 0; tr_free( msgs ); } } tr_publisher_tag tr_peerMsgsSubscribe( tr_peermsgs * peer, tr_delivery_func func, void * userData ) { return tr_publisherSubscribe( peer->publisher, func, userData ); } void tr_peerMsgsUnsubscribe( tr_peermsgs * peer, tr_publisher_tag tag ) { tr_publisherUnsubscribe( peer->publisher, tag ); }