source: trunk/libtransmission/session.c @ 7977

Last change on this file since 7977 was 7977, checked in by charles, 13 years ago

(trunk libT) #1823: Ratio limit is only checked when peers are active

  • Property svn:keywords set to Date Rev Author Id
File size: 45.1 KB
Line 
1/*
2 * This file Copyright (C) 2008-2009 Charles Kerr <charles@transmissionbt.com>
3 *
4 * This file is licensed by the GPL version 2.  Works owned by the
5 * Transmission project are granted a special exemption to clause 2(b)
6 * so that the bulk of its code can remain under the MIT license.
7 * This exemption does not extend to derived works not owned by
8 * the Transmission project.
9 *
10 * $Id: session.c 7977 2009-03-01 13:56:22Z charles $
11 */
12
13#include <assert.h>
14#include <stdlib.h>
15#include <string.h> /* memcpy */
16
17#include <signal.h>
18#include <sys/types.h> /* stat */
19#include <sys/stat.h> /* stat */
20#include <unistd.h> /* stat */
21#include <dirent.h> /* opendir */
22
23#include "transmission.h"
24#include "session.h"
25#include "bandwidth.h"
26#include "bencode.h"
27#include "blocklist.h"
28#include "fdlimit.h"
29#include "list.h"
30#include "metainfo.h" /* tr_metainfoFree */
31#include "net.h"
32#include "peer-mgr.h"
33#include "platform.h" /* tr_lock */
34#include "port-forwarding.h"
35#include "rpc-server.h"
36#include "stats.h"
37#include "torrent.h"
38#include "tracker.h"
39#include "trevent.h"
40#include "utils.h"
41#include "web.h"
42#include "crypto.h"
43
44#define dbgmsg( ... ) \
45    do { \
46        if( tr_deepLoggingIsActive( ) ) \
47            tr_deepLog( __FILE__, __LINE__, NULL, __VA_ARGS__ ); \
48    } while( 0 )
49
50static tr_port
51getRandomPort( tr_session * s )
52{
53    return tr_cryptoWeakRandInt( s->randomPortHigh - s->randomPortLow + 1) + s->randomPortLow;
54}
55
56/* Generate a peer id : "-TRxyzb-" + 12 random alphanumeric
57   characters, where x is the major version number, y is the
58   minor version number, z is the maintenance number, and b
59   designates beta (Azureus-style) */
60uint8_t*
61tr_peerIdNew( void )
62{
63    int          i;
64    int          val;
65    int          total = 0;
66    uint8_t *    buf = tr_new( uint8_t, 21 );
67    const char * pool = "0123456789abcdefghijklmnopqrstuvwxyz";
68    const int    base = 36;
69
70    memcpy( buf, PEERID_PREFIX, 8 );
71
72    for( i = 8; i < 19; ++i )
73    {
74        val = tr_cryptoRandInt( base );
75        total += val;
76        buf[i] = pool[val];
77    }
78
79    val = total % base ? base - ( total % base ) : 0;
80    buf[19] = pool[val];
81    buf[20] = '\0';
82
83    return buf;
84}
85
86const uint8_t*
87tr_getPeerId( void )
88{
89    static uint8_t * id = NULL;
90
91    if( id == NULL )
92        id = tr_peerIdNew( );
93    return id;
94}
95
96/***
97****
98***/
99
100tr_encryption_mode
101tr_sessionGetEncryption( tr_session * session )
102{
103    assert( session );
104
105    return session->encryptionMode;
106}
107
108void
109tr_sessionSetEncryption( tr_session *       session,
110                         tr_encryption_mode mode )
111{
112    assert( session );
113    assert( mode == TR_ENCRYPTION_PREFERRED
114          || mode == TR_ENCRYPTION_REQUIRED
115          || mode == TR_CLEAR_PREFERRED );
116
117    session->encryptionMode = mode;
118}
119
120/***
121****
122***/
123
124static int
125tr_stringEndsWith( const char * str,
126                   const char * end )
127{
128    const size_t slen = strlen( str );
129    const size_t elen = strlen( end );
130
131    return slen >= elen && !memcmp( &str[slen - elen], end, elen );
132}
133
134static void
135loadBlocklists( tr_session * session )
136{
137    int         binCount = 0;
138    int         newCount = 0;
139    struct stat sb;
140    char      * dirname;
141    DIR *       odir = NULL;
142    tr_list *   list = NULL;
143    const tr_bool   isEnabled = session->isBlocklistEnabled;
144
145    /* walk through the directory and find blocklists */
146    dirname = tr_buildPath( session->configDir, "blocklists", NULL );
147    if( !stat( dirname,
148               &sb ) && S_ISDIR( sb.st_mode )
149      && ( ( odir = opendir( dirname ) ) ) )
150    {
151        struct dirent *d;
152        for( d = readdir( odir ); d; d = readdir( odir ) )
153        {
154            char * filename;
155
156            if( !d->d_name || d->d_name[0] == '.' ) /* skip dotfiles, ., and ..
157                                                      */
158                continue;
159
160            filename = tr_buildPath( dirname, d->d_name, NULL );
161
162            if( tr_stringEndsWith( filename, ".bin" ) )
163            {
164                /* if we don't already have this blocklist, add it */
165                if( !tr_list_find( list, filename,
166                                   (TrListCompareFunc)strcmp ) )
167                {
168                    tr_list_append( &list,
169                                   _tr_blocklistNew( filename, isEnabled ) );
170                    ++binCount;
171                }
172            }
173            else
174            {
175                /* strip out the file suffix, if there is one, and add ".bin"
176                  instead */
177                tr_blocklist * b;
178                const char *   dot = strrchr( d->d_name, '.' );
179                const int      len = dot ? dot - d->d_name
180                                         : (int)strlen( d->d_name );
181                char         * tmp = tr_strdup_printf(
182                                        "%s" TR_PATH_DELIMITER_STR "%*.*s.bin",
183                                        dirname, len, len, d->d_name );
184                b = _tr_blocklistNew( tmp, isEnabled );
185                _tr_blocklistSetContent( b, filename );
186                tr_list_append( &list, b );
187                ++newCount;
188                tr_free( tmp );
189            }
190
191            tr_free( filename );
192        }
193
194        closedir( odir );
195    }
196
197    session->blocklists = list;
198
199    if( binCount )
200        tr_dbg( "Found %d blocklists in \"%s\"", binCount, dirname );
201    if( newCount )
202        tr_dbg( "Found %d new blocklists in \"%s\"", newCount, dirname );
203
204    tr_free( dirname );
205}
206
207/***
208****
209***/
210
211#ifdef TR_EMBEDDED
212 #define TR_DEFAULT_ENCRYPTION              TR_CLEAR_PREFERRED
213#else
214 #define TR_DEFAULT_ENCRYPTION              TR_ENCRYPTION_PREFERRED
215#endif
216
217void
218tr_sessionGetDefaultSettings( tr_benc * d )
219{
220    assert( tr_bencIsDict( d ) );
221
222    tr_bencDictReserve( d, 30 );
223    tr_bencDictAddInt( d, TR_PREFS_KEY_BLOCKLIST_ENABLED,        FALSE );
224    tr_bencDictAddStr( d, TR_PREFS_KEY_DOWNLOAD_DIR,             tr_getDefaultDownloadDir( ) );
225    tr_bencDictAddInt( d, TR_PREFS_KEY_DSPEED,                   100 );
226    tr_bencDictAddInt( d, TR_PREFS_KEY_DSPEED_ENABLED,           0 );
227    tr_bencDictAddInt( d, TR_PREFS_KEY_ENCRYPTION,               TR_DEFAULT_ENCRYPTION );
228    tr_bencDictAddInt( d, TR_PREFS_KEY_LAZY_BITFIELD,            TRUE );
229    tr_bencDictAddInt( d, TR_PREFS_KEY_MSGLEVEL,                 TR_MSG_INF );
230    tr_bencDictAddInt( d, TR_PREFS_KEY_OPEN_FILE_LIMIT,          atoi( TR_DEFAULT_OPEN_FILE_LIMIT_STR ) );
231    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_LIMIT_GLOBAL,        atoi( TR_DEFAULT_PEER_LIMIT_GLOBAL_STR ) );
232    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_LIMIT_TORRENT,       atoi( TR_DEFAULT_PEER_LIMIT_TORRENT_STR ) );
233    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT,                atoi( TR_DEFAULT_PEER_PORT_STR ) );
234    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT_RANDOM_ENABLED, FALSE );
235    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT_RANDOM_LOW,     1024 );
236    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT_RANDOM_HIGH,    65535 );
237    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_SOCKET_TOS,          atoi( TR_DEFAULT_PEER_SOCKET_TOS_STR ) );
238    tr_bencDictAddInt( d, TR_PREFS_KEY_PEX_ENABLED,              TRUE );
239    tr_bencDictAddInt( d, TR_PREFS_KEY_PORT_FORWARDING,          TRUE );
240    tr_bencDictAddInt( d, TR_PREFS_KEY_PREALLOCATION,            TR_PREALLOCATE_SPARSE );
241    tr_bencDictAddStr( d, TR_PREFS_KEY_PROXY,                    "" );
242    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_AUTH_ENABLED,       FALSE );
243    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_ENABLED,            FALSE );
244    tr_bencDictAddStr( d, TR_PREFS_KEY_PROXY_PASSWORD,           "" );
245    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_PORT,               80 );
246    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_TYPE,               TR_PROXY_HTTP );
247    tr_bencDictAddStr( d, TR_PREFS_KEY_PROXY_USERNAME,           "" );
248    tr_bencDictAddDouble( d, TR_PREFS_KEY_RATIO,                 2.0 );
249    tr_bencDictAddInt( d, TR_PREFS_KEY_RATIO_ENABLED,            FALSE );
250    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_AUTH_REQUIRED,        FALSE );
251    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_ENABLED,              TRUE );
252    tr_bencDictAddStr( d, TR_PREFS_KEY_RPC_PASSWORD,             "" );
253    tr_bencDictAddStr( d, TR_PREFS_KEY_RPC_USERNAME,             "" );
254    tr_bencDictAddStr( d, TR_PREFS_KEY_RPC_WHITELIST,            TR_DEFAULT_RPC_WHITELIST );
255    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_WHITELIST_ENABLED,    TRUE );
256    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_PORT,                 atoi( TR_DEFAULT_RPC_PORT_STR ) );
257    tr_bencDictAddInt( d, TR_PREFS_KEY_USPEED,                   100 );
258    tr_bencDictAddInt( d, TR_PREFS_KEY_USPEED_ENABLED,           0 );
259    tr_bencDictAddInt( d, TR_PREFS_KEY_UPLOAD_SLOTS_PER_TORRENT, 14 );
260}
261
262void
263tr_sessionGetSettings( tr_session * s, struct tr_benc * d )
264{
265    int i, n=0;
266    char * freeme[16];
267
268    assert( tr_bencIsDict( d ) );
269
270    tr_bencDictReserve( d, 30 );
271    tr_bencDictAddInt( d, TR_PREFS_KEY_BLOCKLIST_ENABLED,        tr_blocklistIsEnabled( s ) );
272    tr_bencDictAddStr( d, TR_PREFS_KEY_DOWNLOAD_DIR,             s->downloadDir );
273    tr_bencDictAddInt( d, TR_PREFS_KEY_DSPEED,                   tr_sessionGetSpeedLimit( s, TR_DOWN ) );
274    tr_bencDictAddInt( d, TR_PREFS_KEY_DSPEED_ENABLED,           tr_sessionIsSpeedLimitEnabled( s, TR_DOWN ) );
275    tr_bencDictAddInt( d, TR_PREFS_KEY_ENCRYPTION,               s->encryptionMode );
276    tr_bencDictAddInt( d, TR_PREFS_KEY_LAZY_BITFIELD,            s->useLazyBitfield );
277    tr_bencDictAddInt( d, TR_PREFS_KEY_MSGLEVEL,                 tr_getMessageLevel( ) );
278    tr_bencDictAddInt( d, TR_PREFS_KEY_OPEN_FILE_LIMIT,          s->openFileLimit );
279    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_LIMIT_GLOBAL,        tr_sessionGetPeerLimit( s ) );
280    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_LIMIT_TORRENT,       s->peerLimitPerTorrent );
281    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT,                tr_sessionGetPeerPort( s ) );
282    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT_RANDOM_ENABLED, s->isPortRandom );
283    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT_RANDOM_LOW,     s->randomPortLow );
284    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_PORT_RANDOM_HIGH,    s->randomPortHigh );
285    tr_bencDictAddInt( d, TR_PREFS_KEY_PEER_SOCKET_TOS,          s->peerSocketTOS );
286    tr_bencDictAddInt( d, TR_PREFS_KEY_PEX_ENABLED,              s->isPexEnabled );
287    tr_bencDictAddInt( d, TR_PREFS_KEY_PORT_FORWARDING,          tr_sessionIsPortForwardingEnabled( s ) );
288    tr_bencDictAddInt( d, TR_PREFS_KEY_PREALLOCATION,            s->preallocationMode );
289    tr_bencDictAddStr( d, TR_PREFS_KEY_PROXY,                    s->proxy );
290    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_AUTH_ENABLED,       s->isProxyAuthEnabled );
291    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_ENABLED,            s->isProxyEnabled );
292    tr_bencDictAddStr( d, TR_PREFS_KEY_PROXY_PASSWORD,           s->proxyPassword );
293    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_PORT,               s->proxyPort );
294    tr_bencDictAddInt( d, TR_PREFS_KEY_PROXY_TYPE,               s->proxyType );
295    tr_bencDictAddStr( d, TR_PREFS_KEY_PROXY_USERNAME,           s->proxyUsername );
296    tr_bencDictAddDouble( d, TR_PREFS_KEY_RATIO,                 s->desiredRatio );
297    tr_bencDictAddInt( d, TR_PREFS_KEY_RATIO_ENABLED,            s->isRatioLimited );
298    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_AUTH_REQUIRED,        tr_sessionIsRPCPasswordEnabled( s ) );
299    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_ENABLED,              tr_sessionIsRPCEnabled( s ) );
300    tr_bencDictAddStr( d, TR_PREFS_KEY_RPC_PASSWORD,             freeme[n++] = tr_sessionGetRPCPassword( s ) );
301    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_PORT,                 tr_sessionGetRPCPort( s ) );
302    tr_bencDictAddStr( d, TR_PREFS_KEY_RPC_USERNAME,             freeme[n++] = tr_sessionGetRPCUsername( s ) );
303    tr_bencDictAddStr( d, TR_PREFS_KEY_RPC_WHITELIST,            freeme[n++] = tr_sessionGetRPCWhitelist( s ) );
304    tr_bencDictAddInt( d, TR_PREFS_KEY_RPC_WHITELIST_ENABLED,    tr_sessionGetRPCWhitelistEnabled( s ) );
305    tr_bencDictAddInt( d, TR_PREFS_KEY_USPEED,                   tr_sessionGetSpeedLimit( s, TR_UP ) );
306    tr_bencDictAddInt( d, TR_PREFS_KEY_USPEED_ENABLED,           tr_sessionIsSpeedLimitEnabled( s, TR_UP ) );
307    tr_bencDictAddInt( d, TR_PREFS_KEY_UPLOAD_SLOTS_PER_TORRENT, s->uploadSlotsPerTorrent );
308
309    for( i=0; i<n; ++i )
310        tr_free( freeme[i] );
311}
312
313void
314tr_sessionLoadSettings( tr_benc * d, const char * configDir, const char * appName )
315{
316    char * filename;
317    tr_benc fileSettings;
318
319    assert( tr_bencIsDict( d ) );
320
321    /* get the defaults */
322    tr_sessionGetDefaultSettings( d );
323
324    /* if caller didn't specify a config dir, use the default */
325    if( !configDir || !*configDir )
326        configDir = tr_getDefaultConfigDir( appName );
327
328    /* file settings override the defaults */
329    filename = tr_buildPath( configDir, "settings.json", NULL );
330    if( !tr_bencLoadJSONFile( filename, &fileSettings ) ) {
331        tr_bencMergeDicts( d, &fileSettings );
332        tr_bencFree( &fileSettings );
333    }
334
335    /* cleanup */
336    tr_free( filename );
337}
338
339void
340tr_sessionSaveSettings( tr_session * session, const char * configDir, tr_benc * settings )
341{
342    tr_benc fileSettings;
343    char * filename;
344
345    assert( tr_bencIsDict( settings ) );
346
347    filename = tr_buildPath( configDir, "settings.json", NULL );
348
349    tr_sessionGetSettings( session, settings );
350
351    if( tr_bencLoadJSONFile( filename, &fileSettings ) ) {
352        tr_bencSaveJSONFile( filename, settings );
353    } else {
354        tr_bencMergeDicts( &fileSettings, settings );
355        tr_bencSaveJSONFile( filename, &fileSettings );
356        tr_bencFree( &fileSettings );
357    }
358
359    tr_inf( "Saved \"%s\"", filename );
360    tr_free( filename );
361}
362
363static void metainfoLookupRescan( tr_session * );
364static void tr_sessionInitImpl( void * );
365
366struct init_data
367{
368    tr_session  * session;
369    const char  * configDir;
370    tr_bool       messageQueuingEnabled;
371    tr_benc     * clientSettings;
372};
373
374tr_session *
375tr_sessionInit( const char  * tag,
376                const char  * configDir,
377                tr_bool       messageQueuingEnabled,
378                tr_benc     * clientSettings )
379{
380    tr_session * session;
381    struct init_data data;
382
383    assert( tr_bencIsDict( clientSettings ) );
384
385    /* initialize the bare skeleton of the session object */
386    session = tr_new0( tr_session, 1 );
387    session->bandwidth = tr_bandwidthNew( session, NULL );
388    session->lock = tr_lockNew( );
389    session->tag = tr_strdup( tag );
390    session->magicNumber = SESSION_MAGIC_NUMBER;
391
392    /* start the libtransmission thread */
393    tr_netInit( ); /* must go before tr_eventInit */
394    tr_eventInit( session );
395    assert( session->events != NULL );
396
397    /* run the rest in the libtransmission thread */
398    session->isWaiting = TRUE;
399    data.session = session;
400    data.configDir = configDir;
401    data.messageQueuingEnabled = messageQueuingEnabled;
402    data.clientSettings = clientSettings;
403    tr_runInEventThread( session, tr_sessionInitImpl, &data );
404    while( session->isWaiting )
405        tr_wait( 100 );
406
407    return session;
408}
409
410static void
411tr_sessionInitImpl( void * vdata )
412{
413    int64_t i;
414    int64_t j;
415    double  d;
416    tr_bool found;
417    const char * str;
418    tr_benc settings;
419    char * filename;
420    struct init_data * data = vdata;
421    tr_benc * clientSettings = data->clientSettings;
422    tr_session * session = data->session;
423
424    assert( tr_amInEventThread( session ) );
425    assert( tr_bencIsDict( clientSettings ) );
426
427    dbgmsg( "tr_sessionInit: the session's top-level bandwidth object is %p", session->bandwidth );
428
429    tr_bencInitDict( &settings, 0 );
430    tr_sessionGetDefaultSettings( &settings );
431    tr_bencMergeDicts( &settings, clientSettings );
432
433#ifndef WIN32
434    /* Don't exit when writing on a broken socket */
435    signal( SIGPIPE, SIG_IGN );
436#endif
437
438    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_LIMIT_TORRENT, &i );
439    assert( found );
440    session->peerLimitPerTorrent = i;
441
442    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_MSGLEVEL, &i );
443    assert( found );
444    tr_setMessageLevel( i );
445    tr_setMessageQueuing( data->messageQueuingEnabled );
446
447
448    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEX_ENABLED, &i );
449    assert( found );
450    session->isPexEnabled = i != 0;
451
452    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_ENCRYPTION, &i );
453    assert( found );
454    assert( tr_isEncryptionMode( i ) );
455    session->encryptionMode = i;
456
457    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PREALLOCATION, &i );
458    assert( found );
459    assert( tr_isPreallocationMode( i ) );
460    session->preallocationMode = i;
461
462    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_SOCKET_TOS, &i );
463    assert( found );
464    session->peerSocketTOS = i;
465
466    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_DOWNLOAD_DIR, &str );
467    assert( found );
468    session->downloadDir = tr_strdup( str );
469
470    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_ENABLED, &i );
471    assert( found );
472    session->isProxyEnabled = i != 0;
473
474    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_PROXY, &str );
475    assert( found );
476    session->proxy = tr_strdup( str );
477
478    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_PORT, &i );
479    assert( found );
480    session->proxyPort = i;
481
482    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_TYPE, &i );
483    assert( found );
484    session->proxyType = i;
485
486    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_AUTH_ENABLED, &i );
487    assert( found );
488    session->isProxyAuthEnabled = i != 0;
489
490    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_PROXY_USERNAME, &str );
491    assert( found );
492    session->proxyUsername = tr_strdup( str );
493
494    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_PROXY_PASSWORD, &str );
495    assert( found );
496    session->proxyPassword = tr_strdup( str );
497
498    session->so_sndbuf = 1500 * 3; /* 3x MTU for most ethernet/wireless */
499    session->so_rcvbuf = 8192;
500
501    tr_setConfigDir( session, data->configDir );
502
503    tr_trackerSessionInit( session );
504    assert( session->tracker != NULL );
505
506    session->peerMgr = tr_peerMgrNew( session );
507
508    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_LAZY_BITFIELD, &i );
509    assert( found );
510    session->useLazyBitfield = i != 0;
511
512    /* Initialize rate and file descripts controls */
513
514    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_OPEN_FILE_LIMIT, &i );
515    assert( found );
516    session->openFileLimit = i;
517    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_LIMIT_GLOBAL, &j );
518    assert( found );
519    tr_fdInit( session->openFileLimit, j );
520
521    /**
522    *** random port
523    **/
524
525    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT_RANDOM_ENABLED, &i );
526    assert( found );
527    session->isPortRandom = i != 0;
528
529    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT_RANDOM_LOW, &i );
530    assert( found );
531    session->randomPortLow = i;
532
533    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT_RANDOM_HIGH, &i );
534    assert( found );
535    session->randomPortHigh = i;
536
537    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PORT_FORWARDING, &i )
538         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT, &j );
539    assert( found );
540    session->peerPort = session->isPortRandom ? getRandomPort( session ) : j;
541    session->shared = tr_sharedInit( session, i, session->peerPort );
542    session->isPortSet = session->isPortRandom || j>0;
543
544    /**
545    **/
546
547    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_UPLOAD_SLOTS_PER_TORRENT, &i );
548    assert( found );
549    session->uploadSlotsPerTorrent = i;
550
551    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_USPEED, &i )
552         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_USPEED_ENABLED, &j );
553    assert( found );
554    tr_sessionSetSpeedLimit( session, TR_UP, i );
555    tr_sessionSetSpeedLimitEnabled( session, TR_UP, j );
556
557    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_DSPEED, &i )
558         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_DSPEED_ENABLED, &j );
559    assert( found );
560    tr_sessionSetSpeedLimit( session, TR_DOWN, i );
561    tr_sessionSetSpeedLimitEnabled( session, TR_DOWN, j );
562
563    found = tr_bencDictFindDouble( &settings, TR_PREFS_KEY_RATIO, &d )
564         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_RATIO_ENABLED, &j );
565    assert( found );
566    tr_sessionSetRatioLimit( session, d );
567    tr_sessionSetRatioLimited( session, j );
568
569    /* initialize the blocklist */
570    filename = tr_buildPath( session->configDir, "blocklists", NULL );
571    tr_mkdirp( filename, 0777 );
572    tr_free( filename );
573    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_BLOCKLIST_ENABLED, &i );
574    assert( found );
575    session->isBlocklistEnabled = i;
576    loadBlocklists( session );
577
578    session->rpcServer = tr_rpcInit( session, &settings );
579
580    tr_bencFree( &settings );
581
582    assert( tr_isSession( session ) );
583
584    /* first %s is the application name
585       second %s is the version number */
586    tr_inf( _( "%s %s started" ), TR_NAME, LONG_VERSION_STRING );
587
588    tr_statsInit( session );
589    session->web = tr_webInit( session );
590    metainfoLookupRescan( session );
591    session->isWaiting = FALSE;
592    dbgmsg( "returning session %p; session->tracker is %p", session, session->tracker );
593}
594
595/***
596****
597***/
598
599void
600tr_sessionSetDownloadDir( tr_session * session, const char * dir )
601{
602    assert( tr_isSession( session ) );
603
604    if( session->downloadDir != dir )
605    {
606        tr_free( session->downloadDir );
607        session->downloadDir = tr_strdup( dir );
608    }
609}
610
611const char *
612tr_sessionGetDownloadDir( const tr_session * session )
613{
614    assert( tr_isSession( session ) );
615
616    return session->downloadDir;
617}
618
619/***
620****
621***/
622
623void
624tr_globalLock( tr_session * session )
625{
626    assert( tr_isSession( session ) );
627
628    tr_lockLock( session->lock );
629}
630
631void
632tr_globalUnlock( tr_session * session )
633{
634    assert( tr_isSession( session ) );
635
636    tr_lockUnlock( session->lock );
637}
638
639tr_bool
640tr_globalIsLocked( const tr_session * session )
641{
642    return tr_isSession( session ) && tr_lockHave( session->lock );
643}
644
645/***********************************************************************
646 * tr_setBindPort
647 ***********************************************************************
648 *
649 **********************************************************************/
650
651struct bind_port_data
652{
653    tr_session * session;
654    tr_port      port;
655};
656
657static void
658tr_setBindPortImpl( void * vdata )
659{
660    struct bind_port_data * data = vdata;
661    tr_session * session = data->session;
662    const tr_port port = data->port;
663
664    session->isPortSet = 1;
665    tr_sharedSetPort( session->shared, port );
666
667    tr_free( data );
668}
669
670static void
671setPortImpl( tr_session * session, tr_port port )
672{
673    struct bind_port_data * data;
674
675    assert( tr_isSession( session ) );
676
677    data = tr_new( struct bind_port_data, 1 );
678    data->session = session;
679    data->port = port;
680    tr_runInEventThread( session, tr_setBindPortImpl, data );
681}
682
683void
684tr_sessionSetPeerPort( tr_session * session,
685                       tr_port      port )
686{
687    assert( tr_isSession( session ) );
688
689    session->isPortRandom = FALSE;
690    session->peerPort = port;
691    setPortImpl( session, session->peerPort );
692}
693
694tr_port
695tr_sessionSetPeerPortRandom( tr_session * session )
696{
697    assert( tr_isSession( session ) );
698
699    session->isPortRandom = TRUE;
700    session->peerPort = getRandomPort( session );
701    setPortImpl( session, session->peerPort );
702    return session->peerPort;
703}
704
705tr_port
706tr_sessionGetPeerPort( const tr_session * session )
707{
708    assert( tr_isSession( session ) );
709
710    return session->peerPort;
711}
712
713tr_port_forwarding
714tr_sessionGetPortForwarding( const tr_session * session )
715{
716    assert( tr_isSession( session ) );
717
718    return tr_sharedTraversalStatus( session->shared );
719}
720
721/***
722****
723***/
724
725static void
726updateSeedRatio( tr_session * session )
727{
728    tr_torrent * tor = NULL;
729
730    while(( tor = tr_torrentNext( session, tor )))
731        tr_torrentCheckSeedRatio( tor );
732}
733
734void
735tr_sessionSetRatioLimited( tr_session * session, tr_bool isLimited )
736{
737    assert( tr_isSession( session ) );
738
739    session->isRatioLimited = isLimited;
740    updateSeedRatio( session );
741}
742
743void
744tr_sessionSetRatioLimit( tr_session * session, double desiredRatio )
745{
746    assert( tr_isSession( session ) );
747
748    session->desiredRatio = desiredRatio;
749    updateSeedRatio( session );
750}
751
752tr_bool
753tr_sessionIsRatioLimited( const tr_session  * session )
754{
755    assert( tr_isSession( session ) );
756
757    return session->isRatioLimited;
758}
759
760double
761tr_sessionGetRatioLimit( const tr_session * session )
762{
763    assert( tr_isSession( session ) );
764
765    return session->desiredRatio;
766}
767
768/***
769****
770***/
771
772static void
773updateBandwidth( tr_session * session, tr_direction dir )
774{
775    tr_bool zeroCase;
776
777    assert( tr_isSession( session ) );
778
779    zeroCase = session->speedLimit[dir] < 1 && session->isSpeedLimited[dir];
780
781    tr_bandwidthSetLimited( session->bandwidth, dir, session->isSpeedLimited[dir] && !zeroCase );
782
783    tr_bandwidthSetDesiredSpeed( session->bandwidth, dir, session->speedLimit[dir] );
784}
785
786void
787tr_sessionSetSpeedLimitEnabled( tr_session      * session,
788                                tr_direction      dir,
789                                tr_bool           isLimited )
790{
791    assert( tr_isSession( session ) );
792    assert( tr_isDirection( dir ) );
793
794    session->isSpeedLimited[dir] = isLimited;
795    updateBandwidth( session, dir );
796}
797
798void
799tr_sessionSetSpeedLimit( tr_session    * session,
800                         tr_direction    dir,
801                         int             desiredSpeed )
802{
803    assert( tr_isSession( session ) );
804    assert( tr_isDirection( dir ) );
805
806    session->speedLimit[dir] = desiredSpeed;
807    updateBandwidth( session, dir );
808}
809
810tr_bool
811tr_sessionIsSpeedLimitEnabled( const tr_session * session, tr_direction dir )
812{
813    assert( tr_isSession( session ) );
814    assert( tr_isDirection( dir ) );
815
816    return session->isSpeedLimited[dir];
817}
818
819int
820tr_sessionGetSpeedLimit( const tr_session * session, tr_direction dir )
821{
822    assert( tr_isSession( session ) );
823    assert( tr_isDirection( dir ) );
824
825    return session->speedLimit[dir];
826}
827
828/***
829****
830***/
831
832void
833tr_sessionSetPeerLimit( tr_session * session, uint16_t maxGlobalPeers )
834{
835    assert( tr_isSession( session ) );
836
837    tr_fdSetPeerLimit( maxGlobalPeers );
838}
839
840uint16_t
841tr_sessionGetPeerLimit( const tr_session * session )
842{
843    assert( tr_isSession( session ) );
844
845    return tr_fdGetPeerLimit( );
846}
847
848void
849tr_sessionSetPeerLimitPerTorrent( tr_session  * session, uint16_t n )
850{
851    assert( tr_isSession( session ) );
852
853    session->peerLimitPerTorrent = n;
854}
855
856uint16_t
857tr_sessionGetPeerLimitPerTorrent( const tr_session * session )
858{
859    assert( tr_isSession( session ) );
860
861    return session->peerLimitPerTorrent;
862}
863
864/***
865****
866***/
867
868double
869tr_sessionGetPieceSpeed( const tr_session * session, tr_direction dir )
870{
871    return tr_isSession( session ) ? tr_bandwidthGetPieceSpeed( session->bandwidth, 0, dir ) : 0.0;
872}
873
874double
875tr_sessionGetRawSpeed( const tr_session * session, tr_direction dir )
876{
877    return tr_isSession( session ) ? tr_bandwidthGetPieceSpeed( session->bandwidth, 0, dir ) : 0.0;
878}
879
880int
881tr_sessionCountTorrents( const tr_session * session )
882{
883    return tr_isSession( session ) ? session->torrentCount : 0;
884}
885
886static int
887compareTorrentByCur( const void * va, const void * vb )
888{
889    const tr_torrent * a = *(const tr_torrent**)va;
890    const tr_torrent * b = *(const tr_torrent**)vb;
891    const uint64_t     aCur = a->downloadedCur + a->uploadedCur;
892    const uint64_t     bCur = b->downloadedCur + b->uploadedCur;
893
894    if( aCur != bCur )
895        return aCur > bCur ? -1 : 1; /* close the biggest torrents first */
896
897    return 0;
898}
899
900static void
901tr_closeAllConnections( void * vsession )
902{
903    tr_session *  session = vsession;
904    tr_torrent *  tor;
905    int           i, n;
906    tr_torrent ** torrents;
907
908    assert( tr_isSession( session ) );
909
910    tr_statsClose( session );
911    tr_sharedShuttingDown( session->shared );
912    tr_rpcClose( &session->rpcServer );
913
914    /* close the torrents.  get the most active ones first so that
915     * if we can't get them all closed in a reasonable amount of time,
916     * at least we get the most important ones first. */
917    tor = NULL;
918    n = session->torrentCount;
919    torrents = tr_new( tr_torrent *, session->torrentCount );
920    for( i = 0; i < n; ++i )
921        torrents[i] = tor = tr_torrentNext( session, tor );
922    qsort( torrents, n, sizeof( tr_torrent* ), compareTorrentByCur );
923    for( i = 0; i < n; ++i )
924        tr_torrentFree( torrents[i] );
925    tr_free( torrents );
926
927    tr_peerMgrFree( session->peerMgr );
928
929    tr_trackerSessionClose( session );
930    tr_list_free( &session->blocklists,
931                  (TrListForeachFunc)_tr_blocklistFree );
932    tr_webClose( &session->web );
933
934    session->isClosed = TRUE;
935}
936
937static int
938deadlineReached( const uint64_t deadline )
939{
940    return tr_date( ) >= deadline;
941}
942
943#define SHUTDOWN_MAX_SECONDS 30
944
945void
946tr_sessionClose( tr_session * session )
947{
948    int            i;
949    const int      maxwait_msec = SHUTDOWN_MAX_SECONDS * 1000;
950    const uint64_t deadline = tr_date( ) + maxwait_msec;
951
952    assert( tr_isSession( session ) );
953
954    dbgmsg( "shutting down transmission session %p", session );
955
956    /* close the session */
957    tr_runInEventThread( session, tr_closeAllConnections, session );
958    while( !session->isClosed && !deadlineReached( deadline ) )
959    {
960        dbgmsg(
961            "waiting for the shutdown commands to run in the main thread" );
962        tr_wait( 100 );
963    }
964
965    /* "shared" and "tracker" have live sockets,
966     * so we need to keep the transmission thread alive
967     * for a bit while they tell the router & tracker
968     * that we're closing now */
969    while( ( session->shared
970           || session->tracker ) && !deadlineReached( deadline ) )
971    {
972        dbgmsg( "waiting on port unmap (%p) or tracker (%p)",
973                session->shared, session->tracker );
974        tr_wait( 100 );
975    }
976
977    tr_fdClose( );
978
979    /* close the libtransmission thread */
980    tr_eventClose( session );
981    while( session->events && !deadlineReached( deadline ) )
982    {
983        dbgmsg( "waiting for the libevent thread to shutdown cleanly" );
984        tr_wait( 100 );
985    }
986
987    /* free the session memory */
988    tr_bandwidthFree( session->bandwidth );
989    tr_lockFree( session->lock );
990    for( i = 0; i < session->metainfoLookupCount; ++i )
991        tr_free( session->metainfoLookup[i].filename );
992    tr_free( session->metainfoLookup );
993    tr_free( session->tag );
994    tr_free( session->configDir );
995    tr_free( session->resumeDir );
996    tr_free( session->torrentDir );
997    tr_free( session->downloadDir );
998    tr_free( session->proxy );
999    tr_free( session->proxyUsername );
1000    tr_free( session->proxyPassword );
1001    tr_free( session );
1002}
1003
1004tr_torrent **
1005tr_sessionLoadTorrents( tr_session * session,
1006                        tr_ctor    * ctor,
1007                        int        * setmeCount )
1008{
1009    int           i, n = 0;
1010    struct stat   sb;
1011    DIR *         odir = NULL;
1012    const char *  dirname = tr_getTorrentDir( session );
1013    tr_torrent ** torrents;
1014    tr_list *     l = NULL, *list = NULL;
1015
1016    assert( tr_isSession( session ) );
1017
1018    tr_ctorSetSave( ctor, FALSE ); /* since we already have them */
1019
1020    if( !stat( dirname, &sb )
1021      && S_ISDIR( sb.st_mode )
1022      && ( ( odir = opendir ( dirname ) ) ) )
1023    {
1024        struct dirent *d;
1025        for( d = readdir( odir ); d != NULL; d = readdir( odir ) )
1026        {
1027            if( d->d_name && d->d_name[0] != '.' ) /* skip dotfiles, ., and ..
1028                                                     */
1029            {
1030                tr_torrent * tor;
1031                char * path = tr_buildPath( dirname, d->d_name, NULL );
1032                tr_ctorSetMetainfoFromFile( ctor, path );
1033                if(( tor = tr_torrentNew( session, ctor, NULL )))
1034                {
1035                    tr_list_append( &list, tor );
1036                    ++n;
1037                }
1038                tr_free( path );
1039            }
1040        }
1041        closedir( odir );
1042    }
1043
1044    torrents = tr_new( tr_torrent *, n );
1045    for( i = 0, l = list; l != NULL; l = l->next )
1046        torrents[i++] = (tr_torrent*) l->data;
1047    assert( i == n );
1048
1049    tr_list_free( &list, NULL );
1050
1051    if( n )
1052        tr_inf( _( "Loaded %d torrents" ), n );
1053
1054    if( setmeCount )
1055        *setmeCount = n;
1056    return torrents;
1057}
1058
1059/***
1060****
1061***/
1062
1063void
1064tr_sessionSetPexEnabled( tr_session * session,
1065                         tr_bool      enabled )
1066{
1067    assert( tr_isSession( session ) );
1068
1069    session->isPexEnabled = enabled != 0;
1070}
1071
1072tr_bool
1073tr_sessionIsPexEnabled( const tr_session * session )
1074{
1075    assert( tr_isSession( session ) );
1076
1077    return session->isPexEnabled;
1078}
1079
1080/***
1081****
1082***/
1083
1084void
1085tr_sessionSetLazyBitfieldEnabled( tr_session * session,
1086                                  tr_bool      enabled )
1087{
1088    assert( tr_isSession( session ) );
1089
1090    session->useLazyBitfield = enabled != 0;
1091}
1092
1093tr_bool
1094tr_sessionIsLazyBitfieldEnabled( const tr_session * session )
1095{
1096    assert( tr_isSession( session ) );
1097
1098    return session->useLazyBitfield;
1099}
1100
1101/***
1102****
1103***/
1104
1105void
1106tr_sessionSetPortForwardingEnabled( tr_session  * session,
1107                                    tr_bool       enabled )
1108{
1109    assert( tr_isSession( session ) );
1110
1111    tr_globalLock( session );
1112    tr_sharedTraversalEnable( session->shared, enabled );
1113    tr_globalUnlock( session );
1114}
1115
1116tr_bool
1117tr_sessionIsPortForwardingEnabled( const tr_session * session )
1118{
1119    assert( tr_isSession( session ) );
1120
1121    return tr_sharedTraversalIsEnabled( session->shared );
1122}
1123
1124/***
1125****
1126***/
1127
1128int
1129tr_blocklistGetRuleCount( const tr_session * session )
1130{
1131    int       n = 0;
1132    tr_list * l;
1133
1134    assert( tr_isSession( session ) );
1135
1136    for( l = session->blocklists; l; l = l->next )
1137        n += _tr_blocklistGetRuleCount( l->data );
1138    return n;
1139}
1140
1141tr_bool
1142tr_blocklistIsEnabled( const tr_session * session )
1143{
1144    assert( tr_isSession( session ) );
1145
1146    return session->isBlocklistEnabled;
1147}
1148
1149void
1150tr_blocklistSetEnabled( tr_session * session,
1151                        tr_bool      isEnabled )
1152{
1153    tr_list * l;
1154
1155    assert( tr_isSession( session ) );
1156
1157    session->isBlocklistEnabled = isEnabled != 0;
1158
1159    for( l=session->blocklists; l!=NULL; l=l->next )
1160        _tr_blocklistSetEnabled( l->data, isEnabled );
1161}
1162
1163tr_bool
1164tr_blocklistExists( const tr_session * session )
1165{
1166    assert( tr_isSession( session ) );
1167
1168    return session->blocklists != NULL;
1169}
1170
1171int
1172tr_blocklistSetContent( tr_session * session,
1173                        const char * contentFilename )
1174{
1175    tr_list *      l;
1176    tr_blocklist * b;
1177    const char *   defaultName = "level1.bin";
1178
1179    assert( tr_isSession( session ) );
1180
1181    for( b = NULL, l = session->blocklists; !b && l; l = l->next )
1182        if( tr_stringEndsWith( _tr_blocklistGetFilename( l->data ),
1183                               defaultName ) )
1184            b = l->data;
1185
1186    if( !b )
1187    {
1188        char * path = tr_buildPath( session->configDir, "blocklists", defaultName, NULL );
1189        b = _tr_blocklistNew( path, session->isBlocklistEnabled );
1190        tr_list_append( &session->blocklists, b );
1191        tr_free( path );
1192    }
1193
1194    return _tr_blocklistSetContent( b, contentFilename );
1195}
1196
1197tr_bool
1198tr_sessionIsAddressBlocked( const tr_session * session,
1199                            const tr_address * addr )
1200{
1201    tr_list * l;
1202
1203    assert( tr_isSession( session ) );
1204
1205    for( l = session->blocklists; l; l = l->next )
1206        if( _tr_blocklistHasAddress( l->data, addr ) )
1207            return TRUE;
1208    return FALSE;
1209}
1210
1211/***
1212****
1213***/
1214
1215static int
1216compareLookupEntries( const void * va, const void * vb )
1217{
1218    const struct tr_metainfo_lookup * a = va;
1219    const struct tr_metainfo_lookup * b = vb;
1220
1221    return strcmp( a->hashString, b->hashString );
1222}
1223
1224static void
1225metainfoLookupResort( tr_session * session )
1226{
1227    assert( tr_isSession( session ) );
1228
1229    qsort( session->metainfoLookup,
1230           session->metainfoLookupCount,
1231           sizeof( struct tr_metainfo_lookup ),
1232           compareLookupEntries );
1233}
1234
1235static int
1236compareHashStringToLookupEntry( const void * va, const void * vb )
1237{
1238    const char *                      a = va;
1239    const struct tr_metainfo_lookup * b = vb;
1240
1241    return strcmp( a, b->hashString );
1242}
1243
1244const char*
1245tr_sessionFindTorrentFile( const tr_session * session,
1246                           const char       * hashStr )
1247{
1248    struct tr_metainfo_lookup * l = bsearch( hashStr,
1249                                             session->metainfoLookup,
1250                                             session->metainfoLookupCount,
1251                                             sizeof( struct tr_metainfo_lookup ),
1252                                             compareHashStringToLookupEntry );
1253
1254    return l ? l->filename : NULL;
1255}
1256
1257static void
1258metainfoLookupRescan( tr_session * session )
1259{
1260    int          i;
1261    int          n;
1262    struct stat  sb;
1263    const char * dirname = tr_getTorrentDir( session );
1264    DIR *        odir = NULL;
1265    tr_ctor *    ctor = NULL;
1266    tr_list *    list = NULL;
1267
1268    assert( tr_isSession( session ) );
1269
1270    /* walk through the directory and find the mappings */
1271    ctor = tr_ctorNew( session );
1272    tr_ctorSetSave( ctor, FALSE ); /* since we already have them */
1273    if( !stat( dirname, &sb ) && S_ISDIR( sb.st_mode ) && ( ( odir = opendir( dirname ) ) ) )
1274    {
1275        struct dirent *d;
1276        for( d = readdir( odir ); d != NULL; d = readdir( odir ) )
1277        {
1278            if( d->d_name && d->d_name[0] != '.' ) /* skip dotfiles, ., and ..
1279                                                     */
1280            {
1281                tr_info inf;
1282                char * path = tr_buildPath( dirname, d->d_name, NULL );
1283                tr_ctorSetMetainfoFromFile( ctor, path );
1284                if( !tr_torrentParse( session, ctor, &inf ) )
1285                {
1286                    tr_list_append( &list, tr_strdup( inf.hashString ) );
1287                    tr_list_append( &list, tr_strdup( path ) );
1288                    tr_metainfoFree( &inf );
1289                }
1290                tr_free( path );
1291            }
1292        }
1293        closedir( odir );
1294    }
1295    tr_ctorFree( ctor );
1296
1297    n = tr_list_size( list ) / 2;
1298    session->metainfoLookup = tr_new0( struct tr_metainfo_lookup, n );
1299    session->metainfoLookupCount = n;
1300    for( i = 0; i < n; ++i )
1301    {
1302        char * hashString = tr_list_pop_front( &list );
1303        char * filename = tr_list_pop_front( &list );
1304
1305        memcpy( session->metainfoLookup[i].hashString, hashString,
1306                2 * SHA_DIGEST_LENGTH + 1 );
1307        tr_free( hashString );
1308        session->metainfoLookup[i].filename = filename;
1309    }
1310
1311    metainfoLookupResort( session );
1312    tr_dbg( "Found %d torrents in \"%s\"", n, dirname );
1313}
1314
1315void
1316tr_sessionSetTorrentFile( tr_session * session,
1317                          const char * hashString,
1318                          const char * filename )
1319{
1320    struct tr_metainfo_lookup * l = bsearch( hashString,
1321                                             session->metainfoLookup,
1322                                             session->metainfoLookupCount,
1323                                             sizeof( struct tr_metainfo_lookup ),
1324                                             compareHashStringToLookupEntry );
1325
1326    if( l )
1327    {
1328        if( l->filename != filename )
1329        {
1330            tr_free( l->filename );
1331            l->filename = tr_strdup( filename );
1332        }
1333    }
1334    else
1335    {
1336        const int n = session->metainfoLookupCount++;
1337        struct tr_metainfo_lookup * node;
1338        session->metainfoLookup = tr_renew( struct tr_metainfo_lookup,
1339                                            session->metainfoLookup,
1340                                            session->metainfoLookupCount );
1341        node = session->metainfoLookup + n;
1342        memcpy( node->hashString, hashString, 2 * SHA_DIGEST_LENGTH + 1 );
1343        node->filename = tr_strdup( filename );
1344        metainfoLookupResort( session );
1345    }
1346}
1347
1348tr_torrent*
1349tr_torrentNext( tr_session * session,
1350                tr_torrent * tor )
1351{
1352    assert( tr_isSession( session ) );
1353
1354    return tor ? tor->next : session->torrentList;
1355}
1356
1357/***
1358****
1359***/
1360
1361void
1362tr_sessionSetRPCEnabled( tr_session * session,
1363                         tr_bool      isEnabled )
1364{
1365    assert( tr_isSession( session ) );
1366
1367    tr_rpcSetEnabled( session->rpcServer, isEnabled );
1368}
1369
1370tr_bool
1371tr_sessionIsRPCEnabled( const tr_session * session )
1372{
1373    assert( tr_isSession( session ) );
1374
1375    return tr_rpcIsEnabled( session->rpcServer );
1376}
1377
1378void
1379tr_sessionSetRPCPort( tr_session * session,
1380                      tr_port      port )
1381{
1382    assert( tr_isSession( session ) );
1383
1384    tr_rpcSetPort( session->rpcServer, port );
1385}
1386
1387tr_port
1388tr_sessionGetRPCPort( const tr_session * session )
1389{
1390    assert( tr_isSession( session ) );
1391
1392    return tr_rpcGetPort( session->rpcServer );
1393}
1394
1395void
1396tr_sessionSetRPCCallback( tr_session * session,
1397                          tr_rpc_func  func,
1398                          void *       user_data )
1399{
1400    assert( tr_isSession( session ) );
1401
1402    session->rpc_func = func;
1403    session->rpc_func_user_data = user_data;
1404}
1405
1406void
1407tr_sessionSetRPCWhitelist( tr_session * session,
1408                           const char * whitelist )
1409{
1410    assert( tr_isSession( session ) );
1411
1412    tr_rpcSetWhitelist( session->rpcServer, whitelist );
1413}
1414
1415char*
1416tr_sessionGetRPCWhitelist( const tr_session * session )
1417{
1418    assert( tr_isSession( session ) );
1419
1420    return tr_rpcGetWhitelist( session->rpcServer );
1421}
1422
1423void
1424tr_sessionSetRPCWhitelistEnabled( tr_session * session,
1425                                  tr_bool      isEnabled )
1426{
1427    assert( tr_isSession( session ) );
1428
1429    tr_rpcSetWhitelistEnabled( session->rpcServer, isEnabled );
1430}
1431
1432tr_bool
1433tr_sessionGetRPCWhitelistEnabled( const tr_session * session )
1434{
1435    assert( tr_isSession( session ) );
1436
1437    return tr_rpcGetWhitelistEnabled( session->rpcServer );
1438}
1439
1440
1441void
1442tr_sessionSetRPCPassword( tr_session * session,
1443                          const char * password )
1444{
1445    assert( tr_isSession( session ) );
1446
1447    tr_rpcSetPassword( session->rpcServer, password );
1448}
1449
1450char*
1451tr_sessionGetRPCPassword( const tr_session * session )
1452{
1453    assert( tr_isSession( session ) );
1454
1455    return tr_rpcGetPassword( session->rpcServer );
1456}
1457
1458void
1459tr_sessionSetRPCUsername( tr_session * session,
1460                          const char * username )
1461{
1462    assert( tr_isSession( session ) );
1463
1464    tr_rpcSetUsername( session->rpcServer, username );
1465}
1466
1467char*
1468tr_sessionGetRPCUsername( const tr_session * session )
1469{
1470    assert( tr_isSession( session ) );
1471
1472    return tr_rpcGetUsername( session->rpcServer );
1473}
1474
1475void
1476tr_sessionSetRPCPasswordEnabled( tr_session * session,
1477                                 tr_bool      isEnabled )
1478{
1479    assert( tr_isSession( session ) );
1480
1481    tr_rpcSetPasswordEnabled( session->rpcServer, isEnabled );
1482}
1483
1484tr_bool
1485tr_sessionIsRPCPasswordEnabled( const tr_session * session )
1486{
1487    assert( tr_isSession( session ) );
1488
1489    return tr_rpcIsPasswordEnabled( session->rpcServer );
1490}
1491
1492/***
1493****
1494***/
1495
1496tr_bool
1497tr_sessionIsProxyEnabled( const tr_session * session )
1498{
1499    assert( tr_isSession( session ) );
1500
1501    return session->isProxyEnabled;
1502}
1503
1504void
1505tr_sessionSetProxyEnabled( tr_session * session,
1506                           tr_bool      isEnabled )
1507{
1508    assert( tr_isSession( session ) );
1509
1510    session->isProxyEnabled = isEnabled != 0;
1511}
1512
1513tr_proxy_type
1514tr_sessionGetProxyType( const tr_session * session )
1515{
1516    assert( tr_isSession( session ) );
1517
1518    return session->proxyType;
1519}
1520
1521void
1522tr_sessionSetProxyType( tr_session *  session,
1523                        tr_proxy_type type )
1524{
1525    assert( tr_isSession( session ) );
1526
1527    session->proxyType = type;
1528}
1529
1530const char*
1531tr_sessionGetProxy( const tr_session * session )
1532{
1533    assert( tr_isSession( session ) );
1534
1535    return session->proxy;
1536}
1537
1538tr_port
1539tr_sessionGetProxyPort( const tr_session * session )
1540{
1541    assert( tr_isSession( session ) );
1542
1543    return session->proxyPort;
1544}
1545
1546void
1547tr_sessionSetProxy( tr_session * session,
1548                    const char * proxy )
1549{
1550    assert( tr_isSession( session ) );
1551
1552    if( proxy != session->proxy )
1553    {
1554        tr_free( session->proxy );
1555        session->proxy = tr_strdup( proxy );
1556    }
1557}
1558
1559void
1560tr_sessionSetProxyPort( tr_session * session,
1561                        tr_port      port )
1562{
1563    assert( tr_isSession( session ) );
1564
1565    session->proxyPort = port;
1566}
1567
1568tr_bool
1569tr_sessionIsProxyAuthEnabled( const tr_session * session )
1570{
1571    assert( tr_isSession( session ) );
1572
1573    return session->isProxyAuthEnabled;
1574}
1575
1576void
1577tr_sessionSetProxyAuthEnabled( tr_session * session,
1578                               tr_bool      isEnabled )
1579{
1580    assert( tr_isSession( session ) );
1581
1582    session->isProxyAuthEnabled = isEnabled != 0;
1583}
1584
1585const char*
1586tr_sessionGetProxyUsername( const tr_session * session )
1587{
1588    assert( tr_isSession( session ) );
1589
1590    return session->proxyUsername;
1591}
1592
1593void
1594tr_sessionSetProxyUsername( tr_session * session,
1595                            const char * username )
1596{
1597    assert( tr_isSession( session ) );
1598
1599    if( username != session->proxyUsername )
1600    {
1601        tr_free( session->proxyUsername );
1602        session->proxyUsername = tr_strdup( username );
1603    }
1604}
1605
1606const char*
1607tr_sessionGetProxyPassword( const tr_session * session )
1608{
1609    assert( tr_isSession( session ) );
1610
1611    return session->proxyPassword;
1612}
1613
1614void
1615tr_sessionSetProxyPassword( tr_session * session,
1616                            const char * password )
1617{
1618    assert( tr_isSession( session ) );
1619
1620    if( password != session->proxyPassword )
1621    {
1622        tr_free( session->proxyPassword );
1623        session->proxyPassword = tr_strdup( password );
1624    }
1625}
1626
1627int
1628tr_sessionGetActiveTorrentCount( tr_session * session )
1629{
1630    int ret = 0;
1631    tr_torrent * tor = NULL;
1632
1633    assert( tr_isSession( session ) );
1634
1635    while(( tor = tr_torrentNext( session, tor )))
1636        if( tr_torrentGetActivity( tor ) != TR_STATUS_STOPPED )
1637            ++ret;
1638
1639    return ret;
1640}
Note: See TracBrowser for help on using the repository browser.