/* * 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 /* UCHAR_MAX */ #include #include #include #include /* event.h needs this */ #include #include "transmission.h" #include "bencode.h" #include "crypto.h" #include "handshake.h" #include "peer-io.h" #include "utils.h" /* enable LibTransmission extension protocol */ #define ENABLE_LTEP /* enable Azureus messaging protocol */ //#define ENABLE_AZMP /*** **** ***/ #define HANDSHAKE_NAME "\023BitTorrent protocol" enum { /* BitTorrent Handshake Constants */ HANDSHAKE_NAME_LEN = 20, HANDSHAKE_FLAGS_LEN = 8, HANDSHAKE_SIZE = 68, PEER_ID_LEN = 20, /* Extension Negotiation Constants */ HANDSHAKE_EXTPREF_LTEP_FORCE = 0x0, HANDSHAKE_EXTPREF_LTEP_PREFER = 0x1, HANDSHAKE_EXTPREF_AZMP_PREFER = 0x2, HANDSHAKE_EXTPREF_AZMP_FORCE = 0x3, /* Encryption Constants */ PadA_MAXLEN = 512, PadB_MAXLEN = 512, PadC_MAXLEN = 512, PadD_MAXLEN = 512, VC_LENGTH = 8, KEY_LEN = 96 }; #ifdef ENABLE_LTEP #define HANDSHAKE_HAS_EXTMSGS( bits ) ( ( (bits)[5] & 0x10 ) ? 1 : 0 ) #define HANDSHAKE_SET_EXTMSGS( bits ) ( ( (bits)[5] |= 0x10 ) ? 1 : 0 ) #else #define HANDSHAKE_HAS_EXTMSGS( bits ) ( 0 ) #define HANDSHAKE_SET_EXTMSGS( bits ) ( (void)0 ) #endif #ifdef ENABLE_AZMP #define HANDSHAKE_HAS_AZPROTO( bits ) ( ( (bits)[0] & 0x80 ) ? 1 : 0 ) #define HANDSHAKE_SET_AZPROTO( bits ) ( ( (bits)[0] |= 0x80 ) ? 1 : 0 ) #else #define HANDSHAKE_HAS_AZPROTO( bits ) ( 0 ) #define HANDSHAKE_SET_AZPROTO( bits ) ( (void)0 ) #endif /* http://www.azureuswiki.com/index.php/Extension_negotiation_protocol these macros are to be used if both extended messaging and the azureus protocol is supported, they indicate which protocol is preferred */ #define HANDSHAKE_GET_EXTPREF( reserved ) ( (reserved)[5] & 0x03 ) #define HANDSHAKE_SET_EXTPREF( reserved, val ) ( (reserved)[5] |= 0x03 & (val) ) extern const char* getPeerId( void ) ; struct tr_handshake { unsigned int peerSupportsEncryption : 1; unsigned int havePeerID : 1; unsigned int haveSentBitTorrentHandshake : 1; unsigned int shunUnencryptedPeers : 1; tr_peerIo * io; tr_crypto * crypto; struct tr_handle * handle; uint8_t myPublicKey[KEY_LEN]; uint8_t mySecret[KEY_LEN]; uint8_t state; uint16_t pad_c_len; uint16_t pad_d_len; uint16_t ia_len; uint32_t crypto_select; uint8_t myReq1[SHA_DIGEST_LENGTH]; uint8_t peer_id[PEER_ID_LEN]; handshakeDoneCB doneCB; void * doneUserData; }; /** *** **/ enum { /* incoming */ AWAITING_HANDSHAKE, AWAITING_YA, AWAITING_PAD_A, AWAITING_CRYPTO_PROVIDE, AWAITING_PAD_C, AWAITING_IA, /* outgoing */ AWAITING_YB, AWAITING_VC, AWAITING_CRYPTO_SELECT, AWAITING_PAD_D, }; /** *** **/ static void myDebug( const char * file, int line, const tr_handshake * handshake, const char * fmt, ... ) { va_list args; const char * addr = tr_peerIoGetAddrStr( handshake->io ); struct evbuffer * buf = evbuffer_new( ); evbuffer_add_printf( buf, "[%s:%d] %s (%p) ", file, line, addr, handshake ); va_start( args, fmt ); evbuffer_add_vprintf( buf, fmt, args ); va_end( args ); fprintf( stderr, "%s\n", EVBUFFER_DATA(buf) ); evbuffer_free( buf ); } #define dbgmsg(handshake, fmt...) myDebug(__FILE__, __LINE__, handshake, ##fmt ) static const char* getStateName( short state ) { const char * str = "f00!"; switch( state ) { case AWAITING_HANDSHAKE: str = "awaiting handshake"; break; case AWAITING_YA: str = "awaiting ya"; break; //case SENDING_YB: str = "sending yb"; break; case AWAITING_PAD_A: str = "awaiting pad a"; break; case AWAITING_CRYPTO_PROVIDE: str = "awaiting crypto_provide"; break; case AWAITING_PAD_C: str = "awaiting pad c"; break; case AWAITING_IA: str = "awaiting ia"; break; //case SENDING_YA: str = "sending ya"; break; case AWAITING_YB: str = "awaiting yb"; break; //case SENDING_CRYPTO_PROVIDE: str = "sending crypto provide"; break; case AWAITING_VC: str = "awaiting vc"; break; case AWAITING_CRYPTO_SELECT: str = "awaiting crypto select"; break; case AWAITING_PAD_D: str = "awaiting pad d"; break; //case SENDING_NONE: str = "sending plaintext handshake"; break; } return str; } static void setState( tr_handshake * handshake, short state ) { dbgmsg( handshake, "setting to state [%s]", getStateName(state) ); handshake->state = state; } static void setReadState( tr_handshake * handshake, int state ) { setState( handshake, state ); //tr_peerIoSetIOMode( handshake->io, EV_READ, EV_WRITE ); } static uint8_t * buildHandshakeMessage( tr_handshake * handshake, int extensionPreference, int * setme_len ) { uint8_t * buf = tr_new0( uint8_t, HANDSHAKE_SIZE ); uint8_t * walk = buf; const uint8_t * torrentHash = tr_cryptoGetTorrentHash( handshake->crypto ); memcpy( walk, HANDSHAKE_NAME, HANDSHAKE_NAME_LEN ); walk += HANDSHAKE_NAME_LEN; memset( walk, 0, HANDSHAKE_FLAGS_LEN ); switch( extensionPreference ) { case HANDSHAKE_EXTPREF_LTEP_FORCE: HANDSHAKE_SET_EXTMSGS( walk ); HANDSHAKE_SET_EXTPREF( walk, extensionPreference ); break; case HANDSHAKE_EXTPREF_LTEP_PREFER: HANDSHAKE_SET_EXTMSGS( walk ); HANDSHAKE_SET_AZPROTO( walk ); HANDSHAKE_SET_EXTPREF( walk, extensionPreference ); break; case HANDSHAKE_EXTPREF_AZMP_PREFER: HANDSHAKE_SET_EXTMSGS( walk ); HANDSHAKE_SET_AZPROTO( walk ); HANDSHAKE_SET_EXTPREF( walk, extensionPreference ); break; case HANDSHAKE_EXTPREF_AZMP_FORCE: HANDSHAKE_SET_AZPROTO( walk ); HANDSHAKE_SET_EXTPREF( walk, extensionPreference ); break; } walk += HANDSHAKE_FLAGS_LEN; memcpy( walk, torrentHash, SHA_DIGEST_LENGTH ); walk += SHA_DIGEST_LENGTH; memcpy( walk, getPeerId(), TR_ID_LEN ); walk += TR_ID_LEN; assert( walk-buf == HANDSHAKE_SIZE ); *setme_len = walk - buf; return buf; } /*** **** **** OUTGOING CONNECTIONS **** ***/ /* 1 A->B: Diffie Hellman Ya, PadA */ static void sendYa( tr_handshake * handshake ) { int i; int len; const uint8_t * public_key; struct evbuffer * outbuf = evbuffer_new( ); uint8_t pad_a[PadA_MAXLEN]; /* add our public key (Ya) */ public_key = tr_cryptoGetMyPublicKey( handshake->crypto, &len ); assert( len == KEY_LEN ); assert( public_key != NULL ); evbuffer_add( outbuf, public_key, len ); /* add some bullshit padding */ len = tr_rand( PadA_MAXLEN ); for( i=0; iio, outbuf ); /* cleanup */ evbuffer_free( outbuf ); } static int readYb( tr_handshake * handshake, struct evbuffer * inbuf ) { int isEncrypted; const uint8_t * secret; uint8_t yb[KEY_LEN]; struct evbuffer * outbuf; size_t needlen = HANDSHAKE_NAME_LEN; if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; isEncrypted = memcmp( EVBUFFER_DATA(inbuf), HANDSHAKE_NAME, HANDSHAKE_NAME_LEN ); if( isEncrypted ) { needlen = KEY_LEN; if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; } dbgmsg( handshake, "got a %s handshake", (isEncrypted ? "encrypted" : "plaintext") ); handshake->peerSupportsEncryption = isEncrypted; tr_peerIoSetEncryption( handshake->io, isEncrypted ? PEER_ENCRYPTION_RC4 : PEER_ENCRYPTION_NONE ); if( !isEncrypted ) { setState( handshake, AWAITING_HANDSHAKE ); return READ_AGAIN; } /* compute the secret */ evbuffer_remove( inbuf, yb, KEY_LEN ); secret = tr_cryptoComputeSecret( handshake->crypto, yb ); memcpy( handshake->mySecret, secret, KEY_LEN ); /* now send these: HASH('req1', S), HASH('req2', SKEY) xor HASH('req3', S), * ENCRYPT(VC, crypto_provide, len(PadC), PadC, len(IA)), ENCRYPT(IA) */ outbuf = evbuffer_new( ); /* HASH('req1', S) */ { uint8_t req1[SHA_DIGEST_LENGTH]; tr_sha1( req1, "req1", 4, secret, KEY_LEN, NULL ); evbuffer_add( outbuf, req1, SHA_DIGEST_LENGTH ); } /* HASH('req2', SKEY) xor HASH('req3', S) */ { int i; uint8_t req2[SHA_DIGEST_LENGTH]; uint8_t req3[SHA_DIGEST_LENGTH]; uint8_t buf[SHA_DIGEST_LENGTH]; tr_sha1( req2, "req2", 4, tr_cryptoGetTorrentHash(handshake->crypto), SHA_DIGEST_LENGTH, NULL ); tr_sha1( req3, "req3", 4, secret, KEY_LEN, NULL ); for( i=0; icrypto ); /* vc */ tr_cryptoEncrypt( handshake->crypto, VC_LENGTH, vc, vc ); evbuffer_add( outbuf, vc, VC_LENGTH ); /* crypto_provide */ crypto_provide = 0; crypto_provide |= (1<<0); /* encryption */ if( !handshake->shunUnencryptedPeers ) /* plaintext */ crypto_provide |= (1<<1); assert( 1<=crypto_provide && crypto_provide<=3 ); crypto_provide = htonl( crypto_provide ); tr_cryptoEncrypt( handshake->crypto, sizeof(crypto_provide), &crypto_provide, &crypto_provide ); evbuffer_add( outbuf, &crypto_provide, sizeof(crypto_provide) ); /* len(padc) */ i = len = tr_rand( 512 ); i = htons( i ); tr_cryptoEncrypt( handshake->crypto, sizeof(i), &i, &i ); evbuffer_add( outbuf, &i, sizeof(i) ); /* padc */ for( i=0; icrypto, len, pad, pad ); evbuffer_add( outbuf, pad, len ); } /* ENCRYPT len(IA)), ENCRYPT(IA) */ { uint16_t i; int msgSize; uint8_t * msg = buildHandshakeMessage( handshake, HANDSHAKE_EXTPREF_LTEP_PREFER, &msgSize ); i = htons( msgSize ); tr_cryptoEncrypt( handshake->crypto, sizeof(uint16_t), &i, &i ); evbuffer_add( outbuf, &i, sizeof(uint16_t) ); tr_cryptoEncrypt( handshake->crypto, msgSize, msg, msg ); evbuffer_add( outbuf, msg, HANDSHAKE_SIZE ); handshake->haveSentBitTorrentHandshake = 1; tr_free( msg ); } /* send it */ tr_cryptoDecryptInit( handshake->crypto ); setReadState( handshake, AWAITING_VC ); tr_peerIoWriteBuf( handshake->io, outbuf ); /* cleanup */ evbuffer_free( outbuf ); return READ_DONE; } static int readVC( tr_handshake * handshake, struct evbuffer * inbuf ) { const uint8_t key[VC_LENGTH] = { 0, 0, 0, 0, 0, 0, 0, 0 }; const int key_len = VC_LENGTH; uint8_t tmp[VC_LENGTH]; /* note: this works w/o having to `unwind' the buffer if * we read too much, but it is pretty brute-force. * it would be nice to make this cleaner. */ for( ;; ) { if( EVBUFFER_LENGTH(inbuf) < VC_LENGTH ) { dbgmsg( handshake, "not enough bytes... returning read_more" ); return READ_MORE; } memcpy( tmp, EVBUFFER_DATA(inbuf), key_len ); tr_cryptoDecryptInit( handshake->crypto ); tr_cryptoDecrypt( handshake->crypto, key_len, tmp, tmp ); if( !memcmp( tmp, key, key_len ) ) break; evbuffer_drain( inbuf, 1 ); } dbgmsg( handshake, "got it!" ); evbuffer_drain( inbuf, key_len ); setState( handshake, AWAITING_CRYPTO_SELECT ); return READ_AGAIN; } static int readCryptoSelect( tr_handshake * handshake, struct evbuffer * inbuf ) { uint32_t crypto_select; uint16_t pad_d_len; const size_t needlen = sizeof(uint32_t) + sizeof(uint16_t); if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; tr_peerIoReadUint32( handshake->io, inbuf, &crypto_select ); assert( crypto_select==1 || crypto_select==2 ); handshake->crypto_select = crypto_select; dbgmsg( handshake, "crypto select is %d", (int)crypto_select ); tr_peerIoReadUint16( handshake->io, inbuf, &pad_d_len ); dbgmsg( handshake, "pad_d_len is %d", (int)pad_d_len ); assert( pad_d_len <= 512 ); handshake->pad_d_len = pad_d_len; setState( handshake, AWAITING_PAD_D ); return READ_AGAIN; } static int readPadD( tr_handshake * handshake, struct evbuffer * inbuf ) { const size_t needlen = handshake->pad_d_len; uint8_t * tmp; dbgmsg( handshake, "pad d: need %d, got %d", (int)needlen, (int)EVBUFFER_LENGTH(inbuf) ); if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; tmp = tr_new( uint8_t, needlen ); tr_peerIoReadBytes( handshake->io, inbuf, tmp, needlen ); tr_free( tmp ); tr_peerIoSetEncryption( handshake->io, handshake->crypto_select ); setState( handshake, AWAITING_HANDSHAKE ); return READ_AGAIN; } /*** **** **** INCOMING CONNECTIONS **** ***/ static void tr_handshakeDone( tr_handshake * handshake, int isConnected ); static int readHandshake( tr_handshake * handshake, struct evbuffer * inbuf ) { int i; uint8_t ltep = 0; uint8_t azmp = 0; uint8_t pstrlen; uint8_t * pstr; uint8_t reserved[HANDSHAKE_FLAGS_LEN]; uint8_t hash[SHA_DIGEST_LENGTH]; dbgmsg( handshake, "payload: need %d, got %d\n", (int)HANDSHAKE_SIZE, (int)EVBUFFER_LENGTH(inbuf) ); if( EVBUFFER_LENGTH(inbuf) < HANDSHAKE_SIZE ) return READ_MORE; pstrlen = EVBUFFER_DATA(inbuf)[0]; /* peek, don't read. We may be handing inbuf to AWAITING_YA */ if( pstrlen == 19 ) /* unencrypted */ { tr_peerIoSetEncryption( handshake->io, PEER_ENCRYPTION_NONE ); } else /* encrypted or corrupt */ { tr_peerIoSetEncryption( handshake->io, PEER_ENCRYPTION_RC4 ); if( tr_peerIoIsIncoming( handshake->io ) ) { dbgmsg( handshake, "I think peer is sending us an encrypted handshake..." ); setState( handshake, AWAITING_YA ); return READ_AGAIN; } tr_cryptoDecrypt( handshake->crypto, 1, &pstrlen, &pstrlen ); if( pstrlen != 19 ) { dbgmsg( handshake, "I think peer has sent us a corrupt handshake..." ); tr_handshakeDone( handshake, FALSE ); return READ_DONE; } } evbuffer_drain( inbuf, 1 ); /* pstr (BitTorrent) */ pstr = tr_new( uint8_t, pstrlen+1 ); tr_peerIoReadBytes( handshake->io, inbuf, pstr, pstrlen ); pstr[pstrlen] = '\0'; if( strcmp( (char*)pstr, "BitTorrent protocol" ) ) { tr_free( pstr ); tr_handshakeDone( handshake, FALSE ); return READ_DONE; } tr_free( pstr ); /* reserved bytes */ tr_peerIoReadBytes( handshake->io, inbuf, reserved, sizeof(reserved) ); /* torrent hash */ tr_peerIoReadBytes( handshake->io, inbuf, hash, sizeof(hash) ); if( tr_peerIoIsIncoming( handshake->io ) ) { if( !tr_torrentExists( handshake->handle, hash ) ) { dbgmsg( handshake, "peer is trying to connect to us for a torrent we don't have." ); tr_handshakeDone( handshake, FALSE ); return READ_DONE; } else { assert( !tr_peerIoHasTorrentHash( handshake->io ) ); tr_peerIoSetTorrentHash( handshake->io, hash ); } } else /* outgoing */ { assert( tr_torrentExists( handshake->handle, hash ) ); assert( tr_peerIoHasTorrentHash( handshake->io ) ); if( memcmp( hash, tr_peerIoGetTorrentHash(handshake->io), SHA_DIGEST_LENGTH ) ) { dbgmsg( handshake, "peer returned the wrong hash. wtf?" ); tr_handshakeDone( handshake, FALSE ); return READ_DONE; } } /* peer id */ tr_peerIoReadBytes( handshake->io, inbuf, handshake->peer_id, sizeof(handshake->peer_id) ); tr_peerIoSetPeersId( handshake->io, handshake->peer_id ); handshake->havePeerID = TRUE; dbgmsg( handshake, "peer-id is [%*.*s]", PEER_ID_LEN, PEER_ID_LEN, handshake->peer_id ); if( !memcmp( handshake->peer_id, getPeerId(), PEER_ID_LEN ) ) { dbgmsg( handshake, "streuth! we've connected to ourselves." ); tr_handshakeDone( handshake, FALSE ); return READ_DONE; } /** *** Extension negotiation **/ ltep = HANDSHAKE_HAS_EXTMSGS( reserved ); azmp = HANDSHAKE_HAS_AZPROTO( reserved ); if( ltep && azmp ) { switch( HANDSHAKE_GET_EXTPREF( reserved ) ) { case HANDSHAKE_EXTPREF_LTEP_FORCE: case HANDSHAKE_EXTPREF_LTEP_PREFER: azmp = 0; break; case HANDSHAKE_EXTPREF_AZMP_FORCE: case HANDSHAKE_EXTPREF_AZMP_PREFER: ltep = 0; break; } } assert( !ltep || !azmp ); if( ltep ) { i = LT_EXTENSIONS_LTEP; dbgmsg(handshake,"using ltep" ); } else if( azmp ) { i = LT_EXTENSIONS_AZMP; dbgmsg(handshake,"using azmp" ); } else { i = LT_EXTENSIONS_NONE; dbgmsg(handshake,"using no extensions" ); } tr_peerIoSetExtension( handshake->io, i ); /** *** If this is an incoming message, then we need to send a response handshake **/ if( !handshake->haveSentBitTorrentHandshake ) { const int ext = ltep ? HANDSHAKE_EXTPREF_LTEP_FORCE : HANDSHAKE_EXTPREF_AZMP_FORCE; int msgSize; uint8_t * msg = buildHandshakeMessage( handshake, ext, &msgSize ); tr_peerIoWrite( handshake->io, msg, msgSize ); tr_free( msg ); handshake->haveSentBitTorrentHandshake = 1; } /* we've completed the BT handshake... pass the work on to peer-msgs */ tr_handshakeDone( handshake, TRUE ); return READ_DONE; } static int readYa( tr_handshake * handshake, struct evbuffer * inbuf ) { uint8_t ya[KEY_LEN]; uint8_t *walk, outbuf[KEY_LEN + PadB_MAXLEN]; const uint8_t *myKey, *secret; int len; if( EVBUFFER_LENGTH( inbuf ) < KEY_LEN ) return READ_MORE; /* read the incoming peer's public key */ evbuffer_remove( inbuf, ya, KEY_LEN ); secret = tr_cryptoComputeSecret( handshake->crypto, ya ); memcpy( handshake->mySecret, secret, KEY_LEN ); tr_sha1( handshake->myReq1, "req1", 4, secret, KEY_LEN, NULL ); /* send our public key to the peer */ walk = outbuf; myKey = tr_cryptoGetMyPublicKey( handshake->crypto, &len ); memcpy( walk, myKey, len ); len = tr_rand( PadB_MAXLEN ); while( len-- ) *walk++ = tr_rand( UCHAR_MAX ); setReadState( handshake, AWAITING_PAD_A ); tr_peerIoWrite( handshake->io, outbuf, walk-outbuf ); return READ_DONE; } static int readPadA( tr_handshake * handshake, struct evbuffer * inbuf ) { uint8_t * pch; /** *** Resynchronizing on HASH('req1',S) **/ pch = memchr( EVBUFFER_DATA(inbuf), handshake->myReq1[0], EVBUFFER_LENGTH(inbuf) ); if( pch == NULL ) { evbuffer_drain( inbuf, EVBUFFER_LENGTH(inbuf) ); return READ_MORE; } evbuffer_drain( inbuf, pch-EVBUFFER_DATA(inbuf) ); if( EVBUFFER_LENGTH(inbuf) < SHA_DIGEST_LENGTH ) return READ_MORE; if( memcmp( EVBUFFER_DATA(inbuf), handshake->myReq1, SHA_DIGEST_LENGTH ) ) { evbuffer_drain( inbuf, 1 ); return READ_AGAIN; } setState( handshake, AWAITING_CRYPTO_PROVIDE ); return READ_AGAIN; } static int readCryptoProvide( tr_handshake * handshake, struct evbuffer * inbuf ) { /* HASH('req2', SKEY) xor HASH('req3', S), ENCRYPT(VC, crypto_provide, len(PadC)) */ int i; uint8_t vc_in[VC_LENGTH]; uint8_t req2[SHA_DIGEST_LENGTH]; uint8_t req3[SHA_DIGEST_LENGTH]; uint8_t obfuscatedTorrentHash[SHA_DIGEST_LENGTH]; uint16_t padc_len = 0; uint32_t crypto_provide = 0; const size_t needlen = SHA_DIGEST_LENGTH + VC_LENGTH + sizeof(crypto_provide) + sizeof(padc_len); tr_torrent * tor = NULL; if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; /* TODO: confirm they sent HASH('req1',S) here? */ evbuffer_drain( inbuf, SHA_DIGEST_LENGTH ); /* This next piece is HASH('req2', SKEY) xor HASH('req3', S) ... * we can get the first half of that (the obufscatedTorrentHash) * by building the latter and xor'ing it with what the peer sent us */ dbgmsg( handshake, "reading obfuscated torrent hash...\n" ); evbuffer_remove( inbuf, req2, SHA_DIGEST_LENGTH ); tr_sha1( req3, "req3", 4, handshake->mySecret, KEY_LEN, NULL ); for( i=0; ihandle, obfuscatedTorrentHash ); assert( tor != NULL ); dbgmsg( handshake, "found the torrent; it's [%s]\n", tor->info.name ); tr_peerIoSetTorrentHash( handshake->io, tor->info.hash ); /* next part: ENCRYPT(VC, crypto_provide, len(PadC), */ tr_cryptoDecryptInit( handshake->crypto ); tr_peerIoReadBytes( handshake->io, inbuf, vc_in, VC_LENGTH ); tr_peerIoReadUint32( handshake->io, inbuf, &crypto_provide ); dbgmsg( handshake, "crypto_provide is %d\n", (int)crypto_provide ); tr_peerIoReadUint16( handshake->io, inbuf, &padc_len ); dbgmsg( handshake, "padc is %d\n", (int)padc_len ); handshake->pad_c_len = padc_len; setState( handshake, AWAITING_PAD_C ); return READ_AGAIN; } static int readPadC( tr_handshake * handshake, struct evbuffer * inbuf ) { uint16_t ia_len; const size_t needlen = handshake->pad_c_len + sizeof(uint16_t); if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; evbuffer_drain( inbuf, needlen ); tr_peerIoReadUint16( handshake->io, inbuf, &ia_len ); dbgmsg( handshake, "ia_len is %d", (int)ia_len ); handshake->ia_len = ia_len; setState( handshake, AWAITING_IA ); return READ_AGAIN; } static int readIA( tr_handshake * handshake, struct evbuffer * inbuf ) { const size_t needlen = handshake->ia_len; if( EVBUFFER_LENGTH(inbuf) < needlen ) return READ_MORE; return readHandshake( handshake, inbuf ); } /*** **** **** **** ***/ static ReadState canRead( struct bufferevent * evin, void * arg ) { tr_handshake * handshake = (tr_handshake *) arg; struct evbuffer * inbuf = EVBUFFER_INPUT ( evin ); ReadState ret; dbgmsg( handshake, "handling canRead; state is [%s]\n", getStateName(handshake->state) ); switch( handshake->state ) { case AWAITING_HANDSHAKE: ret = readHandshake ( handshake, inbuf ); break; case AWAITING_YA: ret = readYa ( handshake, inbuf ); break; case AWAITING_PAD_A: ret = readPadA ( handshake, inbuf ); break; case AWAITING_CRYPTO_PROVIDE: ret = readCryptoProvide( handshake, inbuf ); break; case AWAITING_PAD_C: ret = readPadC ( handshake, inbuf ); break; case AWAITING_IA: ret = readIA ( handshake, inbuf ); break; case AWAITING_YB: ret = readYb ( handshake, inbuf ); break; case AWAITING_VC: ret = readVC ( handshake, inbuf ); break; case AWAITING_CRYPTO_SELECT: ret = readCryptoSelect ( handshake, inbuf ); break; case AWAITING_PAD_D: ret = readPadD ( handshake, inbuf ); break; default: assert( 0 ); } return ret; } static void fireDoneFunc( tr_handshake * handshake, int isConnected ) { const uint8_t * peer_id = isConnected && handshake->havePeerID ? handshake->peer_id : NULL; (*handshake->doneCB)( handshake, handshake->io, isConnected, peer_id, handshake->peerSupportsEncryption, handshake->doneUserData ); } void tr_handshakeDone( tr_handshake * handshake, int isOK ) { dbgmsg( handshake, "handshakeDone: %s", isOK ? "connected" : "aborting" ); tr_peerIoSetIOFuncs( handshake->io, NULL, NULL, NULL, NULL ); fireDoneFunc( handshake, isOK ); tr_free( handshake ); } void tr_handshakeAbort( tr_handshake * handshake ) { tr_handshakeDone( handshake, FALSE ); } static void gotError( struct bufferevent * evbuf UNUSED, short what UNUSED, void * arg ) { tr_handshake * handshake = (tr_handshake *) arg; /* if the error happened while we were sending a public key, we might * have encountered a peer that doesn't do encryption... reconnect and * try a plaintext handshake */ if( ( ( handshake->state == AWAITING_YB ) || ( handshake->state == AWAITING_VC ) ) && ( !handshake->shunUnencryptedPeers ) && ( !tr_peerIoReconnect( handshake->io ) ) ) { int msgSize; uint8_t * msg = buildHandshakeMessage( handshake, HANDSHAKE_EXTPREF_LTEP_PREFER, &msgSize ); setReadState( handshake, AWAITING_HANDSHAKE ); tr_peerIoWrite( handshake->io, msg, msgSize ); tr_free( msg ); } else { tr_handshakeDone( handshake, FALSE ); } } /** *** **/ tr_handshake* tr_handshakeNew( tr_peerIo * io, tr_encryption_mode encryption_mode, handshakeDoneCB doneCB, void * doneUserData ) { tr_handshake * handshake; // w00t //static int count = 0; //if( count++ ) return NULL; handshake = tr_new0( tr_handshake, 1 ); handshake->io = io; handshake->crypto = tr_peerIoGetCrypto( io ); handshake->shunUnencryptedPeers = encryption_mode==TR_ENCRYPTION_REQUIRED; handshake->doneCB = doneCB; handshake->doneUserData = doneUserData; handshake->handle = tr_peerIoGetHandle( io ); tr_peerIoSetIOMode( io, EV_READ|EV_WRITE, 0 ); tr_peerIoSetIOFuncs( io, canRead, NULL, gotError, handshake ); dbgmsg( handshake, "new handshake for io %p", io ); if( tr_peerIoIsIncoming( io ) ) setReadState( handshake, AWAITING_HANDSHAKE ); else sendYa( handshake ); return handshake; } const struct in_addr * tr_handshakeGetAddr( const struct tr_handshake * handshake, uint16_t * port ) { assert( handshake != NULL ); assert( handshake->io != NULL ); return tr_peerIoGetAddress( handshake->io, port ); }