source: trunk/libtransmission/session.c @ 8050

Last change on this file since 8050 was 8047, checked in by charles, 12 years ago

(trunk gtk) launchpad bug #338046: XDG_DOWNLOAD_DIR isn't honored in gtk client

  • Property svn:keywords set to Date Rev Author Id
File size: 45.4 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 8047 2009-03-10 15:24:40Z 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    tr_benc sessionDefaults;
319    tr_benc tmp;
320
321    assert( tr_bencIsDict( d ) );
322
323    /* initializing the defaults: caller may have passed in some app-level defaults.
324     * preserve those and use the session defaults to fill in any missing gaps. */
325    tr_bencInitDict( &sessionDefaults, 0 );
326    tr_sessionGetDefaultSettings( &sessionDefaults );
327    tr_bencMergeDicts( &sessionDefaults, d );
328    tmp = *d; *d = sessionDefaults; sessionDefaults = tmp;
329
330    /* if caller didn't specify a config dir, use the default */
331    if( !configDir || !*configDir )
332        configDir = tr_getDefaultConfigDir( appName );
333
334    /* file settings override the defaults */
335    filename = tr_buildPath( configDir, "settings.json", NULL );
336    if( !tr_bencLoadJSONFile( filename, &fileSettings ) ) {
337        tr_bencMergeDicts( d, &fileSettings );
338        tr_bencFree( &fileSettings );
339    }
340
341    /* cleanup */
342    tr_bencFree( &sessionDefaults );
343    tr_free( filename );
344}
345
346void
347tr_sessionSaveSettings( tr_session * session, const char * configDir, tr_benc * settings )
348{
349    tr_benc fileSettings;
350    char * filename;
351
352    assert( tr_bencIsDict( settings ) );
353
354    filename = tr_buildPath( configDir, "settings.json", NULL );
355
356    tr_sessionGetSettings( session, settings );
357
358    if( tr_bencLoadJSONFile( filename, &fileSettings ) ) {
359        tr_bencSaveJSONFile( filename, settings );
360    } else {
361        tr_bencMergeDicts( &fileSettings, settings );
362        tr_bencSaveJSONFile( filename, &fileSettings );
363        tr_bencFree( &fileSettings );
364    }
365
366    tr_inf( "Saved \"%s\"", filename );
367    tr_free( filename );
368}
369
370static void metainfoLookupRescan( tr_session * );
371static void tr_sessionInitImpl( void * );
372
373struct init_data
374{
375    tr_session  * session;
376    const char  * configDir;
377    tr_bool       messageQueuingEnabled;
378    tr_benc     * clientSettings;
379};
380
381tr_session *
382tr_sessionInit( const char  * tag,
383                const char  * configDir,
384                tr_bool       messageQueuingEnabled,
385                tr_benc     * clientSettings )
386{
387    tr_session * session;
388    struct init_data data;
389
390    assert( tr_bencIsDict( clientSettings ) );
391
392    /* initialize the bare skeleton of the session object */
393    session = tr_new0( tr_session, 1 );
394    session->bandwidth = tr_bandwidthNew( session, NULL );
395    session->lock = tr_lockNew( );
396    session->tag = tr_strdup( tag );
397    session->magicNumber = SESSION_MAGIC_NUMBER;
398
399    /* start the libtransmission thread */
400    tr_netInit( ); /* must go before tr_eventInit */
401    tr_eventInit( session );
402    assert( session->events != NULL );
403
404    /* run the rest in the libtransmission thread */
405    session->isWaiting = TRUE;
406    data.session = session;
407    data.configDir = configDir;
408    data.messageQueuingEnabled = messageQueuingEnabled;
409    data.clientSettings = clientSettings;
410    tr_runInEventThread( session, tr_sessionInitImpl, &data );
411    while( session->isWaiting )
412        tr_wait( 100 );
413
414    return session;
415}
416
417static void
418tr_sessionInitImpl( void * vdata )
419{
420    int64_t i;
421    int64_t j;
422    double  d;
423    tr_bool found;
424    const char * str;
425    tr_benc settings;
426    char * filename;
427    struct init_data * data = vdata;
428    tr_benc * clientSettings = data->clientSettings;
429    tr_session * session = data->session;
430
431    assert( tr_amInEventThread( session ) );
432    assert( tr_bencIsDict( clientSettings ) );
433
434    dbgmsg( "tr_sessionInit: the session's top-level bandwidth object is %p", session->bandwidth );
435
436    tr_bencInitDict( &settings, 0 );
437    tr_sessionGetDefaultSettings( &settings );
438    tr_bencMergeDicts( &settings, clientSettings );
439
440#ifndef WIN32
441    /* Don't exit when writing on a broken socket */
442    signal( SIGPIPE, SIG_IGN );
443#endif
444
445    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_LIMIT_TORRENT, &i );
446    assert( found );
447    session->peerLimitPerTorrent = i;
448
449    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_MSGLEVEL, &i );
450    assert( found );
451    tr_setMessageLevel( i );
452    tr_setMessageQueuing( data->messageQueuingEnabled );
453
454
455    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEX_ENABLED, &i );
456    assert( found );
457    session->isPexEnabled = i != 0;
458
459    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_ENCRYPTION, &i );
460    assert( found );
461    assert( tr_isEncryptionMode( i ) );
462    session->encryptionMode = i;
463
464    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PREALLOCATION, &i );
465    assert( found );
466    assert( tr_isPreallocationMode( i ) );
467    session->preallocationMode = i;
468
469    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_SOCKET_TOS, &i );
470    assert( found );
471    session->peerSocketTOS = i;
472
473    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_DOWNLOAD_DIR, &str );
474    assert( found );
475    session->downloadDir = tr_strdup( str );
476
477    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_ENABLED, &i );
478    assert( found );
479    session->isProxyEnabled = i != 0;
480
481    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_PROXY, &str );
482    assert( found );
483    session->proxy = tr_strdup( str );
484
485    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_PORT, &i );
486    assert( found );
487    session->proxyPort = i;
488
489    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_TYPE, &i );
490    assert( found );
491    session->proxyType = i;
492
493    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PROXY_AUTH_ENABLED, &i );
494    assert( found );
495    session->isProxyAuthEnabled = i != 0;
496
497    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_PROXY_USERNAME, &str );
498    assert( found );
499    session->proxyUsername = tr_strdup( str );
500
501    found = tr_bencDictFindStr( &settings, TR_PREFS_KEY_PROXY_PASSWORD, &str );
502    assert( found );
503    session->proxyPassword = tr_strdup( str );
504
505    session->so_sndbuf = 1500 * 3; /* 3x MTU for most ethernet/wireless */
506    session->so_rcvbuf = 8192;
507
508    tr_setConfigDir( session, data->configDir );
509
510    tr_trackerSessionInit( session );
511    assert( session->tracker != NULL );
512
513    session->peerMgr = tr_peerMgrNew( session );
514
515    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_LAZY_BITFIELD, &i );
516    assert( found );
517    session->useLazyBitfield = i != 0;
518
519    /* Initialize rate and file descripts controls */
520
521    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_OPEN_FILE_LIMIT, &i );
522    assert( found );
523    session->openFileLimit = i;
524    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_LIMIT_GLOBAL, &j );
525    assert( found );
526    tr_fdInit( session->openFileLimit, j );
527
528    /**
529    *** random port
530    **/
531
532    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT_RANDOM_ENABLED, &i );
533    assert( found );
534    session->isPortRandom = i != 0;
535
536    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT_RANDOM_LOW, &i );
537    assert( found );
538    session->randomPortLow = i;
539
540    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT_RANDOM_HIGH, &i );
541    assert( found );
542    session->randomPortHigh = i;
543
544    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_PORT_FORWARDING, &i )
545         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_PEER_PORT, &j );
546    assert( found );
547    session->peerPort = session->isPortRandom ? getRandomPort( session ) : j;
548    session->shared = tr_sharedInit( session, i, session->peerPort );
549    session->isPortSet = session->isPortRandom || j>0;
550
551    /**
552    **/
553
554    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_UPLOAD_SLOTS_PER_TORRENT, &i );
555    assert( found );
556    session->uploadSlotsPerTorrent = i;
557
558    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_USPEED, &i )
559         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_USPEED_ENABLED, &j );
560    assert( found );
561    tr_sessionSetSpeedLimit( session, TR_UP, i );
562    tr_sessionSetSpeedLimitEnabled( session, TR_UP, j );
563
564    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_DSPEED, &i )
565         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_DSPEED_ENABLED, &j );
566    assert( found );
567    tr_sessionSetSpeedLimit( session, TR_DOWN, i );
568    tr_sessionSetSpeedLimitEnabled( session, TR_DOWN, j );
569
570    found = tr_bencDictFindDouble( &settings, TR_PREFS_KEY_RATIO, &d )
571         && tr_bencDictFindInt( &settings, TR_PREFS_KEY_RATIO_ENABLED, &j );
572    assert( found );
573    tr_sessionSetRatioLimit( session, d );
574    tr_sessionSetRatioLimited( session, j );
575
576    /* initialize the blocklist */
577    filename = tr_buildPath( session->configDir, "blocklists", NULL );
578    tr_mkdirp( filename, 0777 );
579    tr_free( filename );
580    found = tr_bencDictFindInt( &settings, TR_PREFS_KEY_BLOCKLIST_ENABLED, &i );
581    assert( found );
582    session->isBlocklistEnabled = i;
583    loadBlocklists( session );
584
585    session->rpcServer = tr_rpcInit( session, &settings );
586
587    tr_bencFree( &settings );
588
589    assert( tr_isSession( session ) );
590
591    /* first %s is the application name
592       second %s is the version number */
593    tr_inf( _( "%s %s started" ), TR_NAME, LONG_VERSION_STRING );
594
595    tr_statsInit( session );
596    session->web = tr_webInit( session );
597    metainfoLookupRescan( session );
598    session->isWaiting = FALSE;
599    dbgmsg( "returning session %p; session->tracker is %p", session, session->tracker );
600}
601
602/***
603****
604***/
605
606void
607tr_sessionSetDownloadDir( tr_session * session, const char * dir )
608{
609    assert( tr_isSession( session ) );
610
611    if( session->downloadDir != dir )
612    {
613        tr_free( session->downloadDir );
614        session->downloadDir = tr_strdup( dir );
615    }
616}
617
618const char *
619tr_sessionGetDownloadDir( const tr_session * session )
620{
621    assert( tr_isSession( session ) );
622
623    return session->downloadDir;
624}
625
626/***
627****
628***/
629
630void
631tr_globalLock( tr_session * session )
632{
633    assert( tr_isSession( session ) );
634
635    tr_lockLock( session->lock );
636}
637
638void
639tr_globalUnlock( tr_session * session )
640{
641    assert( tr_isSession( session ) );
642
643    tr_lockUnlock( session->lock );
644}
645
646tr_bool
647tr_globalIsLocked( const tr_session * session )
648{
649    return tr_isSession( session ) && tr_lockHave( session->lock );
650}
651
652/***********************************************************************
653 * tr_setBindPort
654 ***********************************************************************
655 *
656 **********************************************************************/
657
658struct bind_port_data
659{
660    tr_session * session;
661    tr_port      port;
662};
663
664static void
665tr_setBindPortImpl( void * vdata )
666{
667    struct bind_port_data * data = vdata;
668    tr_session * session = data->session;
669    const tr_port port = data->port;
670
671    session->isPortSet = 1;
672    tr_sharedSetPort( session->shared, port );
673
674    tr_free( data );
675}
676
677static void
678setPortImpl( tr_session * session, tr_port port )
679{
680    struct bind_port_data * data;
681
682    assert( tr_isSession( session ) );
683
684    data = tr_new( struct bind_port_data, 1 );
685    data->session = session;
686    data->port = port;
687    tr_runInEventThread( session, tr_setBindPortImpl, data );
688}
689
690void
691tr_sessionSetPeerPort( tr_session * session,
692                       tr_port      port )
693{
694    assert( tr_isSession( session ) );
695
696    session->isPortRandom = FALSE;
697    session->peerPort = port;
698    setPortImpl( session, session->peerPort );
699}
700
701tr_port
702tr_sessionSetPeerPortRandom( tr_session * session )
703{
704    assert( tr_isSession( session ) );
705
706    session->isPortRandom = TRUE;
707    session->peerPort = getRandomPort( session );
708    setPortImpl( session, session->peerPort );
709    return session->peerPort;
710}
711
712tr_port
713tr_sessionGetPeerPort( const tr_session * session )
714{
715    assert( tr_isSession( session ) );
716
717    return session->peerPort;
718}
719
720tr_port_forwarding
721tr_sessionGetPortForwarding( const tr_session * session )
722{
723    assert( tr_isSession( session ) );
724
725    return tr_sharedTraversalStatus( session->shared );
726}
727
728/***
729****
730***/
731
732static void
733updateSeedRatio( tr_session * session )
734{
735    tr_torrent * tor = NULL;
736
737    while(( tor = tr_torrentNext( session, tor )))
738        tr_torrentCheckSeedRatio( tor );
739}
740
741void
742tr_sessionSetRatioLimited( tr_session * session, tr_bool isLimited )
743{
744    assert( tr_isSession( session ) );
745
746    session->isRatioLimited = isLimited;
747    updateSeedRatio( session );
748}
749
750void
751tr_sessionSetRatioLimit( tr_session * session, double desiredRatio )
752{
753    assert( tr_isSession( session ) );
754
755    session->desiredRatio = desiredRatio;
756    updateSeedRatio( session );
757}
758
759tr_bool
760tr_sessionIsRatioLimited( const tr_session  * session )
761{
762    assert( tr_isSession( session ) );
763
764    return session->isRatioLimited;
765}
766
767double
768tr_sessionGetRatioLimit( const tr_session * session )
769{
770    assert( tr_isSession( session ) );
771
772    return session->desiredRatio;
773}
774
775/***
776****
777***/
778
779static void
780updateBandwidth( tr_session * session, tr_direction dir )
781{
782    tr_bool zeroCase;
783
784    assert( tr_isSession( session ) );
785
786    zeroCase = session->speedLimit[dir] < 1 && session->isSpeedLimited[dir];
787
788    tr_bandwidthSetLimited( session->bandwidth, dir, session->isSpeedLimited[dir] && !zeroCase );
789
790    tr_bandwidthSetDesiredSpeed( session->bandwidth, dir, session->speedLimit[dir] );
791}
792
793void
794tr_sessionSetSpeedLimitEnabled( tr_session      * session,
795                                tr_direction      dir,
796                                tr_bool           isLimited )
797{
798    assert( tr_isSession( session ) );
799    assert( tr_isDirection( dir ) );
800
801    session->isSpeedLimited[dir] = isLimited;
802    updateBandwidth( session, dir );
803}
804
805void
806tr_sessionSetSpeedLimit( tr_session    * session,
807                         tr_direction    dir,
808                         int             desiredSpeed )
809{
810    assert( tr_isSession( session ) );
811    assert( tr_isDirection( dir ) );
812
813    session->speedLimit[dir] = desiredSpeed;
814    updateBandwidth( session, dir );
815}
816
817tr_bool
818tr_sessionIsSpeedLimitEnabled( const tr_session * session, tr_direction dir )
819{
820    assert( tr_isSession( session ) );
821    assert( tr_isDirection( dir ) );
822
823    return session->isSpeedLimited[dir];
824}
825
826int
827tr_sessionGetSpeedLimit( const tr_session * session, tr_direction dir )
828{
829    assert( tr_isSession( session ) );
830    assert( tr_isDirection( dir ) );
831
832    return session->speedLimit[dir];
833}
834
835/***
836****
837***/
838
839void
840tr_sessionSetPeerLimit( tr_session * session, uint16_t maxGlobalPeers )
841{
842    assert( tr_isSession( session ) );
843
844    tr_fdSetPeerLimit( maxGlobalPeers );
845}
846
847uint16_t
848tr_sessionGetPeerLimit( const tr_session * session )
849{
850    assert( tr_isSession( session ) );
851
852    return tr_fdGetPeerLimit( );
853}
854
855void
856tr_sessionSetPeerLimitPerTorrent( tr_session  * session, uint16_t n )
857{
858    assert( tr_isSession( session ) );
859
860    session->peerLimitPerTorrent = n;
861}
862
863uint16_t
864tr_sessionGetPeerLimitPerTorrent( const tr_session * session )
865{
866    assert( tr_isSession( session ) );
867
868    return session->peerLimitPerTorrent;
869}
870
871/***
872****
873***/
874
875double
876tr_sessionGetPieceSpeed( const tr_session * session, tr_direction dir )
877{
878    return tr_isSession( session ) ? tr_bandwidthGetPieceSpeed( session->bandwidth, 0, dir ) : 0.0;
879}
880
881double
882tr_sessionGetRawSpeed( const tr_session * session, tr_direction dir )
883{
884    return tr_isSession( session ) ? tr_bandwidthGetPieceSpeed( session->bandwidth, 0, dir ) : 0.0;
885}
886
887int
888tr_sessionCountTorrents( const tr_session * session )
889{
890    return tr_isSession( session ) ? session->torrentCount : 0;
891}
892
893static int
894compareTorrentByCur( const void * va, const void * vb )
895{
896    const tr_torrent * a = *(const tr_torrent**)va;
897    const tr_torrent * b = *(const tr_torrent**)vb;
898    const uint64_t     aCur = a->downloadedCur + a->uploadedCur;
899    const uint64_t     bCur = b->downloadedCur + b->uploadedCur;
900
901    if( aCur != bCur )
902        return aCur > bCur ? -1 : 1; /* close the biggest torrents first */
903
904    return 0;
905}
906
907static void
908tr_closeAllConnections( void * vsession )
909{
910    tr_session *  session = vsession;
911    tr_torrent *  tor;
912    int           i, n;
913    tr_torrent ** torrents;
914
915    assert( tr_isSession( session ) );
916
917    tr_statsClose( session );
918    tr_sharedShuttingDown( session->shared );
919    tr_rpcClose( &session->rpcServer );
920
921    /* close the torrents.  get the most active ones first so that
922     * if we can't get them all closed in a reasonable amount of time,
923     * at least we get the most important ones first. */
924    tor = NULL;
925    n = session->torrentCount;
926    torrents = tr_new( tr_torrent *, session->torrentCount );
927    for( i = 0; i < n; ++i )
928        torrents[i] = tor = tr_torrentNext( session, tor );
929    qsort( torrents, n, sizeof( tr_torrent* ), compareTorrentByCur );
930    for( i = 0; i < n; ++i )
931        tr_torrentFree( torrents[i] );
932    tr_free( torrents );
933
934    tr_peerMgrFree( session->peerMgr );
935
936    tr_trackerSessionClose( session );
937    tr_list_free( &session->blocklists,
938                  (TrListForeachFunc)_tr_blocklistFree );
939    tr_webClose( &session->web );
940
941    session->isClosed = TRUE;
942}
943
944static int
945deadlineReached( const uint64_t deadline )
946{
947    return tr_date( ) >= deadline;
948}
949
950#define SHUTDOWN_MAX_SECONDS 30
951
952void
953tr_sessionClose( tr_session * session )
954{
955    int            i;
956    const int      maxwait_msec = SHUTDOWN_MAX_SECONDS * 1000;
957    const uint64_t deadline = tr_date( ) + maxwait_msec;
958
959    assert( tr_isSession( session ) );
960
961    dbgmsg( "shutting down transmission session %p", session );
962
963    /* close the session */
964    tr_runInEventThread( session, tr_closeAllConnections, session );
965    while( !session->isClosed && !deadlineReached( deadline ) )
966    {
967        dbgmsg(
968            "waiting for the shutdown commands to run in the main thread" );
969        tr_wait( 100 );
970    }
971
972    /* "shared" and "tracker" have live sockets,
973     * so we need to keep the transmission thread alive
974     * for a bit while they tell the router & tracker
975     * that we're closing now */
976    while( ( session->shared
977           || session->tracker ) && !deadlineReached( deadline ) )
978    {
979        dbgmsg( "waiting on port unmap (%p) or tracker (%p)",
980                session->shared, session->tracker );
981        tr_wait( 100 );
982    }
983
984    tr_fdClose( );
985
986    /* close the libtransmission thread */
987    tr_eventClose( session );
988    while( session->events && !deadlineReached( deadline ) )
989    {
990        dbgmsg( "waiting for the libevent thread to shutdown cleanly" );
991        tr_wait( 100 );
992    }
993
994    /* free the session memory */
995    tr_bandwidthFree( session->bandwidth );
996    tr_lockFree( session->lock );
997    for( i = 0; i < session->metainfoLookupCount; ++i )
998        tr_free( session->metainfoLookup[i].filename );
999    tr_free( session->metainfoLookup );
1000    tr_free( session->tag );
1001    tr_free( session->configDir );
1002    tr_free( session->resumeDir );
1003    tr_free( session->torrentDir );
1004    tr_free( session->downloadDir );
1005    tr_free( session->proxy );
1006    tr_free( session->proxyUsername );
1007    tr_free( session->proxyPassword );
1008    tr_free( session );
1009}
1010
1011tr_torrent **
1012tr_sessionLoadTorrents( tr_session * session,
1013                        tr_ctor    * ctor,
1014                        int        * setmeCount )
1015{
1016    int           i, n = 0;
1017    struct stat   sb;
1018    DIR *         odir = NULL;
1019    const char *  dirname = tr_getTorrentDir( session );
1020    tr_torrent ** torrents;
1021    tr_list *     l = NULL, *list = NULL;
1022
1023    assert( tr_isSession( session ) );
1024
1025    tr_ctorSetSave( ctor, FALSE ); /* since we already have them */
1026
1027    if( !stat( dirname, &sb )
1028      && S_ISDIR( sb.st_mode )
1029      && ( ( odir = opendir ( dirname ) ) ) )
1030    {
1031        struct dirent *d;
1032        for( d = readdir( odir ); d != NULL; d = readdir( odir ) )
1033        {
1034            if( d->d_name && d->d_name[0] != '.' ) /* skip dotfiles, ., and ..
1035                                                     */
1036            {
1037                tr_torrent * tor;
1038                char * path = tr_buildPath( dirname, d->d_name, NULL );
1039                tr_ctorSetMetainfoFromFile( ctor, path );
1040                if(( tor = tr_torrentNew( session, ctor, NULL )))
1041                {
1042                    tr_list_append( &list, tor );
1043                    ++n;
1044                }
1045                tr_free( path );
1046            }
1047        }
1048        closedir( odir );
1049    }
1050
1051    torrents = tr_new( tr_torrent *, n );
1052    for( i = 0, l = list; l != NULL; l = l->next )
1053        torrents[i++] = (tr_torrent*) l->data;
1054    assert( i == n );
1055
1056    tr_list_free( &list, NULL );
1057
1058    if( n )
1059        tr_inf( _( "Loaded %d torrents" ), n );
1060
1061    if( setmeCount )
1062        *setmeCount = n;
1063    return torrents;
1064}
1065
1066/***
1067****
1068***/
1069
1070void
1071tr_sessionSetPexEnabled( tr_session * session,
1072                         tr_bool      enabled )
1073{
1074    assert( tr_isSession( session ) );
1075
1076    session->isPexEnabled = enabled != 0;
1077}
1078
1079tr_bool
1080tr_sessionIsPexEnabled( const tr_session * session )
1081{
1082    assert( tr_isSession( session ) );
1083
1084    return session->isPexEnabled;
1085}
1086
1087/***
1088****
1089***/
1090
1091void
1092tr_sessionSetLazyBitfieldEnabled( tr_session * session,
1093                                  tr_bool      enabled )
1094{
1095    assert( tr_isSession( session ) );
1096
1097    session->useLazyBitfield = enabled != 0;
1098}
1099
1100tr_bool
1101tr_sessionIsLazyBitfieldEnabled( const tr_session * session )
1102{
1103    assert( tr_isSession( session ) );
1104
1105    return session->useLazyBitfield;
1106}
1107
1108/***
1109****
1110***/
1111
1112void
1113tr_sessionSetPortForwardingEnabled( tr_session  * session,
1114                                    tr_bool       enabled )
1115{
1116    assert( tr_isSession( session ) );
1117
1118    tr_globalLock( session );
1119    tr_sharedTraversalEnable( session->shared, enabled );
1120    tr_globalUnlock( session );
1121}
1122
1123tr_bool
1124tr_sessionIsPortForwardingEnabled( const tr_session * session )
1125{
1126    assert( tr_isSession( session ) );
1127
1128    return tr_sharedTraversalIsEnabled( session->shared );
1129}
1130
1131/***
1132****
1133***/
1134
1135int
1136tr_blocklistGetRuleCount( const tr_session * session )
1137{
1138    int       n = 0;
1139    tr_list * l;
1140
1141    assert( tr_isSession( session ) );
1142
1143    for( l = session->blocklists; l; l = l->next )
1144        n += _tr_blocklistGetRuleCount( l->data );
1145    return n;
1146}
1147
1148tr_bool
1149tr_blocklistIsEnabled( const tr_session * session )
1150{
1151    assert( tr_isSession( session ) );
1152
1153    return session->isBlocklistEnabled;
1154}
1155
1156void
1157tr_blocklistSetEnabled( tr_session * session,
1158                        tr_bool      isEnabled )
1159{
1160    tr_list * l;
1161
1162    assert( tr_isSession( session ) );
1163
1164    session->isBlocklistEnabled = isEnabled != 0;
1165
1166    for( l=session->blocklists; l!=NULL; l=l->next )
1167        _tr_blocklistSetEnabled( l->data, isEnabled );
1168}
1169
1170tr_bool
1171tr_blocklistExists( const tr_session * session )
1172{
1173    assert( tr_isSession( session ) );
1174
1175    return session->blocklists != NULL;
1176}
1177
1178int
1179tr_blocklistSetContent( tr_session * session,
1180                        const char * contentFilename )
1181{
1182    tr_list *      l;
1183    tr_blocklist * b;
1184    const char *   defaultName = "level1.bin";
1185
1186    assert( tr_isSession( session ) );
1187
1188    for( b = NULL, l = session->blocklists; !b && l; l = l->next )
1189        if( tr_stringEndsWith( _tr_blocklistGetFilename( l->data ),
1190                               defaultName ) )
1191            b = l->data;
1192
1193    if( !b )
1194    {
1195        char * path = tr_buildPath( session->configDir, "blocklists", defaultName, NULL );
1196        b = _tr_blocklistNew( path, session->isBlocklistEnabled );
1197        tr_list_append( &session->blocklists, b );
1198        tr_free( path );
1199    }
1200
1201    return _tr_blocklistSetContent( b, contentFilename );
1202}
1203
1204tr_bool
1205tr_sessionIsAddressBlocked( const tr_session * session,
1206                            const tr_address * addr )
1207{
1208    tr_list * l;
1209
1210    assert( tr_isSession( session ) );
1211
1212    for( l = session->blocklists; l; l = l->next )
1213        if( _tr_blocklistHasAddress( l->data, addr ) )
1214            return TRUE;
1215    return FALSE;
1216}
1217
1218/***
1219****
1220***/
1221
1222static int
1223compareLookupEntries( const void * va, const void * vb )
1224{
1225    const struct tr_metainfo_lookup * a = va;
1226    const struct tr_metainfo_lookup * b = vb;
1227
1228    return strcmp( a->hashString, b->hashString );
1229}
1230
1231static void
1232metainfoLookupResort( tr_session * session )
1233{
1234    assert( tr_isSession( session ) );
1235
1236    qsort( session->metainfoLookup,
1237           session->metainfoLookupCount,
1238           sizeof( struct tr_metainfo_lookup ),
1239           compareLookupEntries );
1240}
1241
1242static int
1243compareHashStringToLookupEntry( const void * va, const void * vb )
1244{
1245    const char *                      a = va;
1246    const struct tr_metainfo_lookup * b = vb;
1247
1248    return strcmp( a, b->hashString );
1249}
1250
1251const char*
1252tr_sessionFindTorrentFile( const tr_session * session,
1253                           const char       * hashStr )
1254{
1255    struct tr_metainfo_lookup * l = bsearch( hashStr,
1256                                             session->metainfoLookup,
1257                                             session->metainfoLookupCount,
1258                                             sizeof( struct tr_metainfo_lookup ),
1259                                             compareHashStringToLookupEntry );
1260
1261    return l ? l->filename : NULL;
1262}
1263
1264static void
1265metainfoLookupRescan( tr_session * session )
1266{
1267    int          i;
1268    int          n;
1269    struct stat  sb;
1270    const char * dirname = tr_getTorrentDir( session );
1271    DIR *        odir = NULL;
1272    tr_ctor *    ctor = NULL;
1273    tr_list *    list = NULL;
1274
1275    assert( tr_isSession( session ) );
1276
1277    /* walk through the directory and find the mappings */
1278    ctor = tr_ctorNew( session );
1279    tr_ctorSetSave( ctor, FALSE ); /* since we already have them */
1280    if( !stat( dirname, &sb ) && S_ISDIR( sb.st_mode ) && ( ( odir = opendir( dirname ) ) ) )
1281    {
1282        struct dirent *d;
1283        for( d = readdir( odir ); d != NULL; d = readdir( odir ) )
1284        {
1285            if( d->d_name && d->d_name[0] != '.' ) /* skip dotfiles, ., and ..
1286                                                     */
1287            {
1288                tr_info inf;
1289                char * path = tr_buildPath( dirname, d->d_name, NULL );
1290                tr_ctorSetMetainfoFromFile( ctor, path );
1291                if( !tr_torrentParse( session, ctor, &inf ) )
1292                {
1293                    tr_list_append( &list, tr_strdup( inf.hashString ) );
1294                    tr_list_append( &list, tr_strdup( path ) );
1295                    tr_metainfoFree( &inf );
1296                }
1297                tr_free( path );
1298            }
1299        }
1300        closedir( odir );
1301    }
1302    tr_ctorFree( ctor );
1303
1304    n = tr_list_size( list ) / 2;
1305    session->metainfoLookup = tr_new0( struct tr_metainfo_lookup, n );
1306    session->metainfoLookupCount = n;
1307    for( i = 0; i < n; ++i )
1308    {
1309        char * hashString = tr_list_pop_front( &list );
1310        char * filename = tr_list_pop_front( &list );
1311
1312        memcpy( session->metainfoLookup[i].hashString, hashString,
1313                2 * SHA_DIGEST_LENGTH + 1 );
1314        tr_free( hashString );
1315        session->metainfoLookup[i].filename = filename;
1316    }
1317
1318    metainfoLookupResort( session );
1319    tr_dbg( "Found %d torrents in \"%s\"", n, dirname );
1320}
1321
1322void
1323tr_sessionSetTorrentFile( tr_session * session,
1324                          const char * hashString,
1325                          const char * filename )
1326{
1327    struct tr_metainfo_lookup * l = bsearch( hashString,
1328                                             session->metainfoLookup,
1329                                             session->metainfoLookupCount,
1330                                             sizeof( struct tr_metainfo_lookup ),
1331                                             compareHashStringToLookupEntry );
1332
1333    if( l )
1334    {
1335        if( l->filename != filename )
1336        {
1337            tr_free( l->filename );
1338            l->filename = tr_strdup( filename );
1339        }
1340    }
1341    else
1342    {
1343        const int n = session->metainfoLookupCount++;
1344        struct tr_metainfo_lookup * node;
1345        session->metainfoLookup = tr_renew( struct tr_metainfo_lookup,
1346                                            session->metainfoLookup,
1347                                            session->metainfoLookupCount );
1348        node = session->metainfoLookup + n;
1349        memcpy( node->hashString, hashString, 2 * SHA_DIGEST_LENGTH + 1 );
1350        node->filename = tr_strdup( filename );
1351        metainfoLookupResort( session );
1352    }
1353}
1354
1355tr_torrent*
1356tr_torrentNext( tr_session * session,
1357                tr_torrent * tor )
1358{
1359    assert( tr_isSession( session ) );
1360
1361    return tor ? tor->next : session->torrentList;
1362}
1363
1364/***
1365****
1366***/
1367
1368void
1369tr_sessionSetRPCEnabled( tr_session * session,
1370                         tr_bool      isEnabled )
1371{
1372    assert( tr_isSession( session ) );
1373
1374    tr_rpcSetEnabled( session->rpcServer, isEnabled );
1375}
1376
1377tr_bool
1378tr_sessionIsRPCEnabled( const tr_session * session )
1379{
1380    assert( tr_isSession( session ) );
1381
1382    return tr_rpcIsEnabled( session->rpcServer );
1383}
1384
1385void
1386tr_sessionSetRPCPort( tr_session * session,
1387                      tr_port      port )
1388{
1389    assert( tr_isSession( session ) );
1390
1391    tr_rpcSetPort( session->rpcServer, port );
1392}
1393
1394tr_port
1395tr_sessionGetRPCPort( const tr_session * session )
1396{
1397    assert( tr_isSession( session ) );
1398
1399    return tr_rpcGetPort( session->rpcServer );
1400}
1401
1402void
1403tr_sessionSetRPCCallback( tr_session * session,
1404                          tr_rpc_func  func,
1405                          void *       user_data )
1406{
1407    assert( tr_isSession( session ) );
1408
1409    session->rpc_func = func;
1410    session->rpc_func_user_data = user_data;
1411}
1412
1413void
1414tr_sessionSetRPCWhitelist( tr_session * session,
1415                           const char * whitelist )
1416{
1417    assert( tr_isSession( session ) );
1418
1419    tr_rpcSetWhitelist( session->rpcServer, whitelist );
1420}
1421
1422char*
1423tr_sessionGetRPCWhitelist( const tr_session * session )
1424{
1425    assert( tr_isSession( session ) );
1426
1427    return tr_rpcGetWhitelist( session->rpcServer );
1428}
1429
1430void
1431tr_sessionSetRPCWhitelistEnabled( tr_session * session,
1432                                  tr_bool      isEnabled )
1433{
1434    assert( tr_isSession( session ) );
1435
1436    tr_rpcSetWhitelistEnabled( session->rpcServer, isEnabled );
1437}
1438
1439tr_bool
1440tr_sessionGetRPCWhitelistEnabled( const tr_session * session )
1441{
1442    assert( tr_isSession( session ) );
1443
1444    return tr_rpcGetWhitelistEnabled( session->rpcServer );
1445}
1446
1447
1448void
1449tr_sessionSetRPCPassword( tr_session * session,
1450                          const char * password )
1451{
1452    assert( tr_isSession( session ) );
1453
1454    tr_rpcSetPassword( session->rpcServer, password );
1455}
1456
1457char*
1458tr_sessionGetRPCPassword( const tr_session * session )
1459{
1460    assert( tr_isSession( session ) );
1461
1462    return tr_rpcGetPassword( session->rpcServer );
1463}
1464
1465void
1466tr_sessionSetRPCUsername( tr_session * session,
1467                          const char * username )
1468{
1469    assert( tr_isSession( session ) );
1470
1471    tr_rpcSetUsername( session->rpcServer, username );
1472}
1473
1474char*
1475tr_sessionGetRPCUsername( const tr_session * session )
1476{
1477    assert( tr_isSession( session ) );
1478
1479    return tr_rpcGetUsername( session->rpcServer );
1480}
1481
1482void
1483tr_sessionSetRPCPasswordEnabled( tr_session * session,
1484                                 tr_bool      isEnabled )
1485{
1486    assert( tr_isSession( session ) );
1487
1488    tr_rpcSetPasswordEnabled( session->rpcServer, isEnabled );
1489}
1490
1491tr_bool
1492tr_sessionIsRPCPasswordEnabled( const tr_session * session )
1493{
1494    assert( tr_isSession( session ) );
1495
1496    return tr_rpcIsPasswordEnabled( session->rpcServer );
1497}
1498
1499/***
1500****
1501***/
1502
1503tr_bool
1504tr_sessionIsProxyEnabled( const tr_session * session )
1505{
1506    assert( tr_isSession( session ) );
1507
1508    return session->isProxyEnabled;
1509}
1510
1511void
1512tr_sessionSetProxyEnabled( tr_session * session,
1513                           tr_bool      isEnabled )
1514{
1515    assert( tr_isSession( session ) );
1516
1517    session->isProxyEnabled = isEnabled != 0;
1518}
1519
1520tr_proxy_type
1521tr_sessionGetProxyType( const tr_session * session )
1522{
1523    assert( tr_isSession( session ) );
1524
1525    return session->proxyType;
1526}
1527
1528void
1529tr_sessionSetProxyType( tr_session *  session,
1530                        tr_proxy_type type )
1531{
1532    assert( tr_isSession( session ) );
1533
1534    session->proxyType = type;
1535}
1536
1537const char*
1538tr_sessionGetProxy( const tr_session * session )
1539{
1540    assert( tr_isSession( session ) );
1541
1542    return session->proxy;
1543}
1544
1545tr_port
1546tr_sessionGetProxyPort( const tr_session * session )
1547{
1548    assert( tr_isSession( session ) );
1549
1550    return session->proxyPort;
1551}
1552
1553void
1554tr_sessionSetProxy( tr_session * session,
1555                    const char * proxy )
1556{
1557    assert( tr_isSession( session ) );
1558
1559    if( proxy != session->proxy )
1560    {
1561        tr_free( session->proxy );
1562        session->proxy = tr_strdup( proxy );
1563    }
1564}
1565
1566void
1567tr_sessionSetProxyPort( tr_session * session,
1568                        tr_port      port )
1569{
1570    assert( tr_isSession( session ) );
1571
1572    session->proxyPort = port;
1573}
1574
1575tr_bool
1576tr_sessionIsProxyAuthEnabled( const tr_session * session )
1577{
1578    assert( tr_isSession( session ) );
1579
1580    return session->isProxyAuthEnabled;
1581}
1582
1583void
1584tr_sessionSetProxyAuthEnabled( tr_session * session,
1585                               tr_bool      isEnabled )
1586{
1587    assert( tr_isSession( session ) );
1588
1589    session->isProxyAuthEnabled = isEnabled != 0;
1590}
1591
1592const char*
1593tr_sessionGetProxyUsername( const tr_session * session )
1594{
1595    assert( tr_isSession( session ) );
1596
1597    return session->proxyUsername;
1598}
1599
1600void
1601tr_sessionSetProxyUsername( tr_session * session,
1602                            const char * username )
1603{
1604    assert( tr_isSession( session ) );
1605
1606    if( username != session->proxyUsername )
1607    {
1608        tr_free( session->proxyUsername );
1609        session->proxyUsername = tr_strdup( username );
1610    }
1611}
1612
1613const char*
1614tr_sessionGetProxyPassword( const tr_session * session )
1615{
1616    assert( tr_isSession( session ) );
1617
1618    return session->proxyPassword;
1619}
1620
1621void
1622tr_sessionSetProxyPassword( tr_session * session,
1623                            const char * password )
1624{
1625    assert( tr_isSession( session ) );
1626
1627    if( password != session->proxyPassword )
1628    {
1629        tr_free( session->proxyPassword );
1630        session->proxyPassword = tr_strdup( password );
1631    }
1632}
1633
1634int
1635tr_sessionGetActiveTorrentCount( tr_session * session )
1636{
1637    int ret = 0;
1638    tr_torrent * tor = NULL;
1639
1640    assert( tr_isSession( session ) );
1641
1642    while(( tor = tr_torrentNext( session, tor )))
1643        if( tr_torrentGetActivity( tor ) != TR_STATUS_STOPPED )
1644            ++ret;
1645
1646    return ret;
1647}
Note: See TracBrowser for help on using the repository browser.