source: trunk/libtransmission/session.c @ 8118

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

(trunk) add a did-the-user-do-this flag to the "alt speed toggled" callback

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