source: trunk/libtransmission/utils.c @ 11331

Last change on this file since 11331 was 11331, checked in by charles, 11 years ago

(trunk libT) #3521 "rounding issue in tr_truncd()" -- this time for sure...

  • Property svn:keywords set to Date Rev Author Id
File size: 39.3 KB
Line 
1/*
2 * This file Copyright (C) 2009-2010 Mnemosyne LLC
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: utils.c 11331 2010-10-17 18:27:42Z charles $
11 */
12
13#ifdef HAVE_MEMMEM
14 #define _GNU_SOURCE /* glibc's string.h needs this to pick up memmem */
15#endif
16
17#if defined(SYS_DARWIN)
18 #define HAVE_GETPAGESIZE
19 #define HAVE_ICONV_OPEN
20 #define HAVE_VALLOC
21 #undef HAVE_POSIX_MEMALIGN /* not supported on OS X 10.5 and lower */
22#endif
23
24#include <assert.h>
25#include <ctype.h> /* isalpha(), tolower() */
26#include <errno.h>
27#include <math.h> /* pow(), fabs(), floor() */
28#include <stdarg.h>
29#include <stdio.h>
30#include <stdlib.h>
31#include <string.h> /* strerror(), memset(), memmem() */
32#include <time.h> /* nanosleep() */
33
34#ifdef HAVE_ICONV_OPEN
35 #include <iconv.h>
36#endif
37#include <libgen.h> /* basename() */
38#include <sys/time.h>
39#include <sys/types.h>
40#include <sys/stat.h>
41#include <unistd.h> /* stat(), getcwd(), getpagesize() */
42
43#include "event.h"
44
45#ifdef WIN32
46 #include <w32api.h>
47 #define WINVER WindowsXP /* freeaddrinfo(), getaddrinfo(), getnameinfo() */
48 #include <direct.h> /* _getcwd() */
49 #include <windows.h> /* Sleep() */
50#endif
51
52#include "transmission.h"
53#include "bencode.h"
54#include "fdlimit.h"
55#include "ConvertUTF.h"
56#include "list.h"
57#include "net.h"
58#include "utils.h"
59#include "platform.h"
60#include "version.h"
61
62
63time_t transmission_now = 0;
64
65tr_msg_level messageLevel = TR_MSG_ERR;
66static tr_bool        messageQueuing = FALSE;
67static tr_msg_list *  messageQueue = NULL;
68static tr_msg_list ** messageQueueTail = &messageQueue;
69static int            messageQueueCount = 0;
70
71#ifndef WIN32
72    /* make null versions of these win32 functions */
73    static inline int IsDebuggerPresent( void ) { return FALSE; }
74    static inline void OutputDebugString( const void * unused UNUSED ) { }
75#endif
76
77/***
78****
79***/
80
81static tr_lock*
82getMessageLock( void )
83{
84    static tr_lock * l = NULL;
85
86    if( !l )
87        l = tr_lockNew( );
88
89    return l;
90}
91
92FILE*
93tr_getLog( void )
94{
95    static tr_bool initialized = FALSE;
96    static FILE * file = NULL;
97
98    if( !initialized )
99    {
100        const char * str = getenv( "TR_DEBUG_FD" );
101        int          fd = 0;
102        if( str && *str )
103            fd = atoi( str );
104        switch( fd )
105        {
106            case 1:
107                file = stdout; break;
108
109            case 2:
110                file = stderr; break;
111
112            default:
113                file = NULL; break;
114        }
115        initialized = TRUE;
116    }
117
118    return file;
119}
120
121void
122tr_setMessageLevel( tr_msg_level level )
123{
124    messageLevel = level;
125}
126
127tr_msg_level
128tr_getMessageLevel( void )
129{
130    return messageLevel;
131}
132
133void
134tr_setMessageQueuing( tr_bool enabled )
135{
136    messageQueuing = enabled;
137}
138
139tr_bool
140tr_getMessageQueuing( void )
141{
142    return messageQueuing != 0;
143}
144
145tr_msg_list *
146tr_getQueuedMessages( void )
147{
148    tr_msg_list * ret;
149    tr_lockLock( getMessageLock( ) );
150
151    ret = messageQueue;
152    messageQueue = NULL;
153    messageQueueTail = &messageQueue;
154
155    messageQueueCount = 0;
156
157    tr_lockUnlock( getMessageLock( ) );
158    return ret;
159}
160
161void
162tr_freeMessageList( tr_msg_list * list )
163{
164    tr_msg_list * next;
165
166    while( NULL != list )
167    {
168        next = list->next;
169        free( list->message );
170        free( list->name );
171        free( list );
172        list = next;
173    }
174}
175
176/**
177***
178**/
179
180struct tm *
181tr_localtime_r( const time_t *_clock, struct tm *_result )
182{
183#ifdef HAVE_LOCALTIME_R
184    return localtime_r( _clock, _result );
185#else
186    struct tm *p = localtime( _clock );
187    if( p )
188        *(_result) = *p;
189    return p;
190#endif
191}
192
193char*
194tr_getLogTimeStr( char * buf, int buflen )
195{
196    char           tmp[64];
197    struct tm      now_tm;
198    struct timeval tv;
199    time_t         seconds;
200    int            milliseconds;
201
202    gettimeofday( &tv, NULL );
203
204    seconds = tv.tv_sec;
205    tr_localtime_r( &seconds, &now_tm );
206    strftime( tmp, sizeof( tmp ), "%H:%M:%S", &now_tm );
207    milliseconds = (int)( tv.tv_usec / 1000 );
208    tr_snprintf( buf, buflen, "%s.%03d", tmp, milliseconds );
209
210    return buf;
211}
212
213tr_bool
214tr_deepLoggingIsActive( void )
215{
216    static int8_t deepLoggingIsActive = -1;
217
218    if( deepLoggingIsActive < 0 )
219        deepLoggingIsActive = IsDebuggerPresent() || (tr_getLog()!=NULL);
220
221    return deepLoggingIsActive != 0;
222}
223
224void
225tr_deepLog( const char  * file,
226            int           line,
227            const char  * name,
228            const char  * fmt,
229            ... )
230{
231    FILE * fp = tr_getLog( );
232    if( fp || IsDebuggerPresent( ) )
233    {
234        va_list           args;
235        char              timestr[64];
236        struct evbuffer * buf = evbuffer_new( );
237        char *            base = tr_basename( file );
238
239        evbuffer_add_printf( buf, "[%s] ",
240                            tr_getLogTimeStr( timestr, sizeof( timestr ) ) );
241        if( name )
242            evbuffer_add_printf( buf, "%s ", name );
243        va_start( args, fmt );
244        evbuffer_add_vprintf( buf, fmt, args );
245        va_end( args );
246        evbuffer_add_printf( buf, " (%s:%d)\n", base, line );
247        /* FIXME(libevent2) ifdef this out for nonwindows platforms */
248        OutputDebugString( EVBUFFER_DATA( buf ) );
249        if(fp) /* FIXME(libevent2) tr_getLog() should return an fd, then use evbuffer_write() here ) */
250            (void) fwrite( EVBUFFER_DATA( buf ), 1, EVBUFFER_LENGTH( buf ), fp );
251
252        tr_free( base );
253        evbuffer_free( buf );
254    }
255}
256
257/***
258****
259***/
260
261void
262tr_msg( const char * file, int line,
263        tr_msg_level level,
264        const char * name,
265        const char * fmt, ... )
266{
267    const int err = errno; /* message logging shouldn't affect errno */
268    char buf[1024];
269    va_list ap;
270    tr_lockLock( getMessageLock( ) );
271
272    /* build the text message */
273    *buf = '\0';
274    va_start( ap, fmt );
275    evutil_vsnprintf( buf, sizeof( buf ), fmt, ap );
276    va_end( ap );
277
278    OutputDebugString( buf );
279
280    if( *buf )
281    {
282        if( messageQueuing )
283        {
284            tr_msg_list * newmsg;
285            newmsg = tr_new0( tr_msg_list, 1 );
286            newmsg->level = level;
287            newmsg->when = tr_time( );
288            newmsg->message = tr_strdup( buf );
289            newmsg->file = file;
290            newmsg->line = line;
291            newmsg->name = tr_strdup( name );
292
293            *messageQueueTail = newmsg;
294            messageQueueTail = &newmsg->next;
295            ++messageQueueCount;
296
297            if( messageQueueCount > TR_MAX_MSG_LOG )
298            {
299                tr_msg_list * old = messageQueue;
300                messageQueue = old->next;
301                old->next = NULL;
302                tr_freeMessageList(old);
303
304                --messageQueueCount;
305
306                assert( messageQueueCount == TR_MAX_MSG_LOG );
307            }
308        }
309        else
310        {
311            char timestr[64];
312            FILE * fp;
313
314            fp = tr_getLog( );
315            if( fp == NULL )
316                fp = stderr;
317
318            tr_getLogTimeStr( timestr, sizeof( timestr ) );
319
320            if( name )
321                fprintf( fp, "[%s] %s: %s\n", timestr, name, buf );
322            else
323                fprintf( fp, "[%s] %s\n", timestr, buf );
324            fflush( fp );
325        }
326    }
327
328    tr_lockUnlock( getMessageLock( ) );
329    errno = err;
330}
331
332/***
333****
334***/
335
336void*
337tr_malloc( size_t size )
338{
339    return size ? malloc( size ) : NULL;
340}
341
342void*
343tr_malloc0( size_t size )
344{
345    return size ? calloc( 1, size ) : NULL;
346}
347
348void
349tr_free( void * p )
350{
351    if( p != NULL )
352        free( p );
353}
354
355void*
356tr_memdup( const void * src, size_t byteCount )
357{
358    return memcpy( tr_malloc( byteCount ), src, byteCount );
359}
360
361/***
362****
363***/
364
365void
366tr_set_compare( const void * va,
367                size_t aCount,
368                const void * vb,
369                size_t bCount,
370                int compare( const void * a, const void * b ),
371                size_t elementSize,
372                tr_set_func in_a_cb,
373                tr_set_func in_b_cb,
374                tr_set_func in_both_cb,
375                void * userData )
376{
377    const uint8_t * a = (const uint8_t *) va;
378    const uint8_t * b = (const uint8_t *) vb;
379    const uint8_t * aend = a + elementSize * aCount;
380    const uint8_t * bend = b + elementSize * bCount;
381
382    while( a != aend || b != bend )
383    {
384        if( a == aend )
385        {
386            ( *in_b_cb )( (void*)b, userData );
387            b += elementSize;
388        }
389        else if( b == bend )
390        {
391            ( *in_a_cb )( (void*)a, userData );
392            a += elementSize;
393        }
394        else
395        {
396            const int val = ( *compare )( a, b );
397
398            if( !val )
399            {
400                ( *in_both_cb )( (void*)a, userData );
401                a += elementSize;
402                b += elementSize;
403            }
404            else if( val < 0 )
405            {
406                ( *in_a_cb )( (void*)a, userData );
407                a += elementSize;
408            }
409            else if( val > 0 )
410            {
411                ( *in_b_cb )( (void*)b, userData );
412                b += elementSize;
413            }
414        }
415    }
416}
417
418/***
419****
420***/
421
422const char*
423tr_strip_positional_args( const char* str )
424{
425    const char * in = str;
426    static size_t bufsize = 0;
427    static char * buf = NULL;
428    const size_t  len = strlen( str );
429    char *        out;
430
431    if( bufsize < len )
432    {
433        bufsize = len * 2;
434        buf = tr_renew( char, buf, bufsize );
435    }
436
437    for( out = buf; *str; ++str )
438    {
439        *out++ = *str;
440
441        if( ( *str == '%' ) && isdigit( str[1] ) )
442        {
443            const char * tmp = str + 1;
444            while( isdigit( *tmp ) )
445                ++tmp;
446            if( *tmp == '$' )
447                str = tmp[1]=='\'' ? tmp+1 : tmp;
448        }
449
450        if( ( *str == '%' ) && ( str[1] == '\'' ) )
451            str = str + 1;
452 
453    }
454    *out = '\0';
455
456    return strcmp( buf, in ) ? buf : in;
457}
458
459/**
460***
461**/
462
463void
464tr_timerAdd( struct event * timer, int seconds, int microseconds )
465{
466    struct timeval tv;
467    tv.tv_sec = seconds;
468    tv.tv_usec = microseconds;
469
470    assert( tv.tv_sec >= 0 );
471    assert( tv.tv_usec >= 0 );
472    assert( tv.tv_usec < 1000000 );
473
474    evtimer_add( timer, &tv );
475}
476
477void
478tr_timerAddMsec( struct event * timer, int msec )
479{
480    const int seconds =  msec / 1000;
481    const int usec = (msec%1000) * 1000;
482    tr_timerAdd( timer, seconds, usec );
483}
484
485/**
486***
487**/
488
489uint8_t *
490tr_loadFile( const char * path,
491             size_t *     size )
492{
493    uint8_t * buf;
494    struct stat  sb;
495    int fd;
496    ssize_t n;
497    const char * const err_fmt = _( "Couldn't read \"%1$s\": %2$s" );
498
499    /* try to stat the file */
500    errno = 0;
501    if( stat( path, &sb ) )
502    {
503        const int err = errno;
504        tr_dbg( err_fmt, path, tr_strerror( errno ) );
505        errno = err;
506        return NULL;
507    }
508
509    if( ( sb.st_mode & S_IFMT ) != S_IFREG )
510    {
511        tr_err( err_fmt, path, _( "Not a regular file" ) );
512        errno = EISDIR;
513        return NULL;
514    }
515
516    /* Load the torrent file into our buffer */
517    fd = tr_open_file_for_scanning( path );
518    if( fd < 0 )
519    {
520        const int err = errno;
521        tr_err( err_fmt, path, tr_strerror( errno ) );
522        errno = err;
523        return NULL;
524    }
525    buf = malloc( sb.st_size + 1 );
526    if( !buf )
527    {
528        const int err = errno;
529        tr_err( err_fmt, path, _( "Memory allocation failed" ) );
530        tr_close_file( fd );
531        errno = err;
532        return NULL;
533    }
534    n = read( fd, buf, (size_t)sb.st_size );
535    if( n == -1 )
536    {
537        const int err = errno;
538        tr_err( err_fmt, path, tr_strerror( errno ) );
539        tr_close_file( fd );
540        free( buf );
541        errno = err;
542        return NULL;
543    }
544
545    tr_close_file( fd );
546    buf[ sb.st_size ] = '\0';
547    *size = sb.st_size;
548    return buf;
549}
550
551char*
552tr_basename( const char * path )
553{
554    char * tmp = tr_strdup( path );
555    char * ret = tr_strdup( basename( tmp ) );
556    tr_free( tmp );
557    return ret;
558}
559
560char*
561tr_dirname( const char * path )
562{
563    char * tmp = tr_strdup( path );
564    char * ret = tr_strdup( dirname( tmp ) );
565    tr_free( tmp );
566    return ret;
567}
568
569int
570tr_mkdir( const char * path,
571          int permissions
572#ifdef WIN32
573                       UNUSED
574#endif
575        )
576{
577#ifdef WIN32
578    if( path && isalpha( path[0] ) && path[1] == ':' && !path[2] )
579        return 0;
580    return mkdir( path );
581#else
582    return mkdir( path, permissions );
583#endif
584}
585
586int
587tr_mkdirp( const char * path_in,
588           int          permissions )
589{
590    char *      path = tr_strdup( path_in );
591    char *      p, * pp;
592    struct stat sb;
593    int         done;
594
595    /* walk past the root */
596    p = path;
597    while( *p == TR_PATH_DELIMITER )
598        ++p;
599
600    pp = p;
601    done = 0;
602    while( ( p =
603                strchr( pp, TR_PATH_DELIMITER ) ) || ( p = strchr( pp, '\0' ) ) )
604    {
605        if( !*p )
606            done = 1;
607        else
608            *p = '\0';
609
610        if( stat( path, &sb ) )
611        {
612            /* Folder doesn't exist yet */
613            if( tr_mkdir( path, permissions ) )
614            {
615                const int err = errno;
616                tr_err( _(
617                           "Couldn't create \"%1$s\": %2$s" ), path,
618                       tr_strerror( err ) );
619                tr_free( path );
620                errno = err;
621                return -1;
622            }
623        }
624        else if( ( sb.st_mode & S_IFMT ) != S_IFDIR )
625        {
626            /* Node exists but isn't a folder */
627            char * buf = tr_strdup_printf( _( "File \"%s\" is in the way" ), path );
628            tr_err( _( "Couldn't create \"%1$s\": %2$s" ), path_in, buf );
629            tr_free( buf );
630            tr_free( path );
631            errno = ENOTDIR;
632            return -1;
633        }
634
635        if( done )
636            break;
637
638        *p = TR_PATH_DELIMITER;
639        p++;
640        pp = p;
641    }
642
643    tr_free( path );
644    return 0;
645}
646
647char*
648tr_buildPath( const char *first_element, ... )
649{
650    size_t bufLen = 0;
651    const char * element;
652    char * buf;
653    char * pch;
654    va_list vl;
655
656    /* pass 1: allocate enough space for the string */
657    va_start( vl, first_element );
658    element = first_element;
659    while( element ) {
660        bufLen += strlen( element ) + 1;
661        element = (const char*) va_arg( vl, const char* );
662    }
663    pch = buf = tr_new( char, bufLen );
664    va_end( vl );
665
666    /* pass 2: build the string piece by piece */
667    va_start( vl, first_element );
668    element = first_element;
669    while( element ) {
670        const size_t elementLen = strlen( element );
671        memcpy( pch, element, elementLen );
672        pch += elementLen;
673        *pch++ = TR_PATH_DELIMITER;
674        element = (const char*) va_arg( vl, const char* );
675    }
676    va_end( vl );
677
678    /* terminate the string.  if nonempty, eat the unwanted trailing slash */
679    if( pch != buf )
680        --pch;
681    *pch++ = '\0';
682
683    /* sanity checks & return */
684    assert( pch - buf == (off_t)bufLen );
685    return buf;
686}
687
688/****
689*****
690****/
691
692char*
693tr_strdup( const void * in )
694{
695    return tr_strndup( in, in ? (int)strlen((const char *)in) : 0 );
696}
697
698char*
699tr_strndup( const void * in, int len )
700{
701    char * out = NULL;
702
703    if( len < 0 )
704    {
705        out = tr_strdup( in );
706    }
707    else if( in )
708    {
709        out = tr_malloc( len + 1 );
710        memcpy( out, in, len );
711        out[len] = '\0';
712    }
713
714    return out;
715}
716
717const char*
718tr_memmem( const char * haystack, size_t haystacklen,
719           const char * needle, size_t needlelen )
720{
721#ifdef HAVE_MEMMEM
722    return memmem( haystack, haystacklen, needle, needlelen );
723#else
724    size_t i;
725    if( !needlelen )
726        return haystack;
727    if( needlelen > haystacklen || !haystack || !needle )
728        return NULL;
729    for( i=0; i<=haystacklen-needlelen; ++i )
730        if( !memcmp( haystack+i, needle, needlelen ) )
731            return haystack+i;
732    return NULL;
733#endif
734}
735
736char*
737tr_strdup_printf( const char * fmt, ... )
738{
739    va_list ap;
740    char * ret;
741    size_t len;
742    char statbuf[2048];
743
744    va_start( ap, fmt );
745    len = evutil_vsnprintf( statbuf, sizeof( statbuf ), fmt, ap );
746    va_end( ap );
747    if( len < sizeof( statbuf ) )
748        ret = tr_strndup( statbuf, len );
749    else {
750        ret = tr_new( char, len + 1 );
751        va_start( ap, fmt );
752        evutil_vsnprintf( ret, len + 1, fmt, ap );
753        va_end( ap );
754    }
755
756    return ret;
757}
758
759const char*
760tr_strerror( int i )
761{
762    const char * ret = strerror( i );
763
764    if( ret == NULL )
765        ret = "Unknown Error";
766    return ret;
767}
768
769/****
770*****
771****/
772
773char*
774tr_strstrip( char * str )
775{
776    if( str != NULL )
777    {
778        size_t pos;
779        size_t len = strlen( str );
780
781        while( len && isspace( str[len - 1] ) )
782            --len;
783
784        for( pos = 0; pos < len && isspace( str[pos] ); )
785            ++pos;
786
787        len -= pos;
788        memmove( str, str + pos, len );
789        str[len] = '\0';
790    }
791
792    return str;
793}
794
795tr_bool
796tr_str_has_suffix( const char *str, const char *suffix )
797{
798    size_t str_len;
799    size_t suffix_len;
800
801    if( !str )
802        return FALSE;
803    if( !suffix )
804        return TRUE;
805
806    str_len = strlen( str );
807    suffix_len = strlen( suffix );
808    if( str_len < suffix_len )
809        return FALSE;
810
811    return !strncasecmp( str + str_len - suffix_len, suffix, suffix_len );
812}
813
814/****
815*****
816****/
817
818uint64_t
819tr_time_msec( void )
820{
821    struct timeval tv;
822
823    gettimeofday( &tv, NULL );
824    return (uint64_t) tv.tv_sec * 1000 + ( tv.tv_usec / 1000 );
825}
826
827void
828tr_wait_msec( long int msec )
829{
830#ifdef WIN32
831    Sleep( (DWORD)msec );
832#else
833    struct timespec ts;
834    ts.tv_sec = msec / 1000;
835    ts.tv_nsec = ( msec % 1000 ) * 1000000;
836    nanosleep( &ts, NULL );
837#endif
838}
839
840/***
841****
842***/
843
844int
845tr_snprintf( char * buf, size_t buflen, const char * fmt, ... )
846{
847    int     len;
848    va_list args;
849
850    va_start( args, fmt );
851    len = evutil_vsnprintf( buf, buflen, fmt, args );
852    va_end( args );
853    return len;
854}
855
856/*
857 * Copy src to string dst of size siz.  At most siz-1 characters
858 * will be copied.  Always NUL terminates (unless siz == 0).
859 * Returns strlen(src); if retval >= siz, truncation occurred.
860 */
861size_t
862tr_strlcpy( char *       dst,
863            const void * src,
864            size_t       siz )
865{
866#ifdef HAVE_STRLCPY
867    return strlcpy( dst, src, siz );
868#else
869    char *      d = dst;
870    const char *s = src;
871    size_t      n = siz;
872
873    assert( s );
874    assert( d );
875
876    /* Copy as many bytes as will fit */
877    if( n != 0 )
878    {
879        while( --n != 0 )
880        {
881            if( ( *d++ = *s++ ) == '\0' )
882                break;
883        }
884    }
885
886    /* Not enough room in dst, add NUL and traverse rest of src */
887    if( n == 0 )
888    {
889        if( siz != 0 )
890            *d = '\0'; /* NUL-terminate dst */
891        while( *s++ )
892            ;
893    }
894
895    return s - (char*)src - 1;  /* count does not include NUL */
896#endif
897}
898
899/***
900****
901***/
902
903double
904tr_getRatio( uint64_t numerator, uint64_t denominator )
905{
906    double ratio;
907
908    if( denominator > 0 )
909        ratio = numerator / (double)denominator;
910    else if( numerator > 0 )
911        ratio = TR_RATIO_INF;
912    else
913        ratio = TR_RATIO_NA;
914
915    return ratio;
916}
917
918void
919tr_sha1_to_hex( char * out, const uint8_t * sha1 )
920{
921    int i;
922    static const char hex[] = "0123456789abcdef";
923
924    for( i=0; i<20; ++i )
925    {
926        const unsigned int val = *sha1++;
927        *out++ = hex[val >> 4];
928        *out++ = hex[val & 0xf];
929    }
930
931    *out = '\0';
932}
933
934void
935tr_hex_to_sha1( uint8_t * out, const char * in )
936{
937    int i;
938    static const char hex[] = "0123456789abcdef";
939
940    for( i=0; i<20; ++i )
941    {
942        const int hi = strchr( hex, tolower( *in++ ) ) - hex;
943        const int lo = strchr( hex, tolower( *in++ ) ) - hex;
944        *out++ = (uint8_t)( (hi<<4) | lo );
945    }
946}
947
948/***
949****
950***/
951
952static tr_bool
953isValidURLChars( const char * url, int url_len )
954{
955    const char * c;
956    const char * end;
957    static const char * rfc2396_valid_chars =
958        "abcdefghijklmnopqrstuvwxyz" /* lowalpha */
959        "ABCDEFGHIJKLMNOPQRSTUVWXYZ" /* upalpha */
960        "0123456789"                 /* digit */
961        "-_.!~*'()"                  /* mark */
962        ";/?:@&=+$,"                 /* reserved */
963        "<>#%<\""                    /* delims */
964        "{}|\\^[]`";                 /* unwise */
965
966    if( url == NULL )
967        return FALSE;
968
969    for( c=url, end=c+url_len; c && *c && c!=end; ++c )
970        if( !strchr( rfc2396_valid_chars, *c ) )
971            return FALSE;
972
973    return TRUE;
974}
975
976/** @brief return TRUE if the url is a http or https url that Transmission understands */
977tr_bool
978tr_urlIsValidTracker( const char * url )
979{
980    tr_bool valid;
981    char * scheme = NULL;
982    const int len = url ? strlen(url) : 0;
983
984    valid = isValidURLChars( url, len )
985         && !tr_urlParse( url, len, &scheme, NULL, NULL, NULL )
986         && ( scheme != NULL )
987         && ( !strcmp(scheme,"http") || !strcmp(scheme,"https") );
988
989    tr_free( scheme );
990    return valid;
991}
992
993/** @brief return TRUE if the url is a http or https or ftp or sftp url that Transmission understands */
994tr_bool
995tr_urlIsValid( const char * url, int url_len )
996{
997    tr_bool valid;
998    char * scheme = NULL;
999    if( ( url_len < 0 ) && ( url != NULL ) )
1000        url_len = strlen( url );
1001
1002    valid = isValidURLChars( url, url_len )
1003         && !tr_urlParse( url, url_len, &scheme, NULL, NULL, NULL )
1004         && ( scheme != NULL )
1005         && ( !strcmp(scheme,"http") || !strcmp(scheme,"https") || !strcmp(scheme,"ftp") || !strcmp(scheme,"sftp") );
1006
1007    tr_free( scheme );
1008    return valid;
1009}
1010
1011tr_bool
1012tr_addressIsIP( const char * address )
1013{
1014    tr_address tempAddr;
1015    return tr_pton(address, &tempAddr) != NULL;
1016}
1017
1018int
1019tr_urlParse( const char * url_in,
1020             int          len,
1021             char **      setme_protocol,
1022             char **      setme_host,
1023             int *        setme_port,
1024             char **      setme_path )
1025{
1026    int          err;
1027    int          port = 0;
1028    int          n;
1029    char *       tmp;
1030    char *       pch;
1031    const char * protocol = NULL;
1032    const char * host = NULL;
1033    const char * path = NULL;
1034
1035    tmp = tr_strndup( url_in, len );
1036    if( ( pch = strstr( tmp, "://" ) ) )
1037    {
1038        *pch = '\0';
1039        protocol = tmp;
1040        pch += 3;
1041/*fprintf( stderr, "protocol is [%s]... what's left is [%s]\n", protocol, pch);*/
1042        if( ( n = strcspn( pch, ":/" ) ) )
1043        {
1044            const int havePort = pch[n] == ':';
1045            host = pch;
1046            pch += n;
1047            *pch++ = '\0';
1048/*fprintf( stderr, "host is [%s]... what's left is [%s]\n", host, pch );*/
1049            if( havePort )
1050            {
1051                char * end;
1052                port = strtol( pch, &end, 10 );
1053                pch = end;
1054/*fprintf( stderr, "port is [%d]... what's left is [%s]\n", port, pch );*/
1055            }
1056            path = pch;
1057/*fprintf( stderr, "path is [%s]\n", path );*/
1058        }
1059    }
1060
1061    err = !host || !path || !protocol;
1062
1063    if( !err && !port )
1064    {
1065        if( !strcmp( protocol, "ftp" ) ) port = 21;
1066        if( !strcmp( protocol, "sftp" ) ) port = 22;
1067        if( !strcmp( protocol, "http" ) ) port = 80;
1068        if( !strcmp( protocol, "https" ) ) port = 443;
1069    }
1070
1071    if( !err )
1072    {
1073        if( setme_protocol ) *setme_protocol = tr_strdup( protocol );
1074
1075        if( setme_host ){ ( (char*)host )[-3] = ':'; *setme_host =
1076                              tr_strdup( host ); }
1077        if( setme_path ){ if( path[0] == '/' ) *setme_path = tr_strdup( path );
1078                          else { ( (char*)path )[-1] = '/'; *setme_path = tr_strdup( path - 1 ); } }
1079        if( setme_port ) *setme_port = port;
1080    }
1081
1082
1083    tr_free( tmp );
1084    return err;
1085}
1086
1087#include <string.h>
1088#include <openssl/sha.h>
1089#include <openssl/hmac.h>
1090#include <openssl/evp.h>
1091#include <openssl/bio.h>
1092#include <openssl/buffer.h>
1093
1094char *
1095tr_base64_encode( const void * input, int length, int * setme_len )
1096{
1097    int retlen = 0;
1098    char * ret = NULL;
1099
1100    if( input != NULL )
1101    {
1102        BIO * b64;
1103        BIO * bmem;
1104        BUF_MEM * bptr;
1105
1106        if( length < 1 )
1107            length = (int)strlen( input );
1108
1109        bmem = BIO_new( BIO_s_mem( ) );
1110        b64 = BIO_new( BIO_f_base64( ) );
1111        b64 = BIO_push( b64, bmem );
1112        BIO_write( b64, input, length );
1113        (void) BIO_flush( b64 );
1114        BIO_get_mem_ptr( b64, &bptr );
1115        ret = tr_strndup( bptr->data, bptr->length );
1116        retlen = bptr->length;
1117        BIO_free_all( b64 );
1118    }
1119
1120    if( setme_len )
1121        *setme_len = retlen;
1122
1123    return ret;
1124}
1125
1126char *
1127tr_base64_decode( const void * input,
1128                  int          length,
1129                  int *        setme_len )
1130{
1131    char * ret;
1132    BIO *  b64;
1133    BIO *  bmem;
1134    int    retlen;
1135
1136    if( length < 1 )
1137        length = strlen( input );
1138
1139    ret = tr_new0( char, length );
1140    b64 = BIO_new( BIO_f_base64( ) );
1141    bmem = BIO_new_mem_buf( (unsigned char*)input, length );
1142    bmem = BIO_push( b64, bmem );
1143    retlen = BIO_read( bmem, ret, length );
1144    if( !retlen )
1145    {
1146        /* try again, but with the BIO_FLAGS_BASE64_NO_NL flag */
1147        BIO_free_all( bmem );
1148        b64 = BIO_new( BIO_f_base64( ) );
1149        BIO_set_flags( b64, BIO_FLAGS_BASE64_NO_NL );
1150        bmem = BIO_new_mem_buf( (unsigned char*)input, length );
1151        bmem = BIO_push( b64, bmem );
1152        retlen = BIO_read( bmem, ret, length );
1153    }
1154
1155    if( setme_len )
1156        *setme_len = retlen;
1157
1158    BIO_free_all( bmem );
1159    return ret;
1160}
1161
1162/***
1163****
1164***/
1165
1166void
1167tr_removeElementFromArray( void         * array,
1168                           unsigned int   index_to_remove,
1169                           size_t         sizeof_element,
1170                           size_t         nmemb )
1171{
1172    char * a = (char*) array;
1173
1174    memmove( a + sizeof_element * index_to_remove,
1175             a + sizeof_element * ( index_to_remove  + 1 ),
1176             sizeof_element * ( --nmemb - index_to_remove ) );
1177}
1178
1179int
1180tr_lowerBound( const void * key,
1181               const void * base,
1182               size_t       nmemb,
1183               size_t       size,
1184               int       (* compar)(const void* key, const void* arrayMember),
1185               tr_bool    * exact_match )
1186{
1187    size_t first = 0;
1188    const char * cbase = base;
1189    tr_bool exact = FALSE;
1190
1191    while( nmemb != 0 )
1192    {
1193        const size_t half = nmemb / 2;
1194        const size_t middle = first + half;
1195        const int c = compar( key, cbase + size*middle );
1196
1197        if( c <= 0 ) {
1198            if( c == 0 )
1199                exact = TRUE;
1200            nmemb = half;
1201        } else {
1202            first = middle + 1;
1203            nmemb = nmemb - half - 1;
1204        }
1205    }
1206
1207    *exact_match = exact;
1208
1209    return first;
1210}
1211
1212/***
1213****
1214***/
1215
1216static char*
1217strip_non_utf8( const char * in, size_t inlen )
1218{
1219    char * ret;
1220    const char * end;
1221    const char zero = '\0';
1222    struct evbuffer * buf = evbuffer_new( );
1223 
1224    while( !tr_utf8_validate( in, inlen, &end ) )
1225    {
1226        const int good_len = end - in;
1227
1228        evbuffer_add( buf, in, good_len );
1229        inlen -= ( good_len + 1 );
1230        in += ( good_len + 1 );
1231        evbuffer_add( buf, "?", 1 );
1232    }
1233 
1234    evbuffer_add( buf, in, inlen );
1235    evbuffer_add( buf, &zero, 1 );
1236    ret = tr_memdup( EVBUFFER_DATA( buf ), EVBUFFER_LENGTH( buf ) );
1237    evbuffer_free( buf );
1238    return ret;
1239}
1240
1241static char*
1242to_utf8( const char * in, size_t inlen )
1243{
1244    char * ret = NULL;
1245
1246#ifdef HAVE_ICONV_OPEN
1247    int i;
1248    const char * encodings[] = { "CURRENT", "ISO-8859-15" };
1249    const int encoding_count = sizeof(encodings) / sizeof(encodings[1]);
1250    const size_t buflen = inlen*4 + 10;
1251    char * out = tr_new( char, buflen );
1252
1253    for( i=0; !ret && i<encoding_count; ++i )
1254    {
1255        char * inbuf = (char*) in;
1256        char * outbuf = out;
1257        size_t inbytesleft = inlen;
1258        size_t outbytesleft = buflen;
1259        const char * test_encoding = encodings[i];
1260
1261        iconv_t cd = iconv_open( "UTF-8", test_encoding );
1262        if( cd != (iconv_t)-1 ) {
1263            if( iconv( cd, &inbuf, &inbytesleft, &outbuf, &outbytesleft ) != (size_t)-1 )
1264                ret = tr_strndup( out, buflen-outbytesleft );
1265            iconv_close( cd );
1266        }
1267    }
1268#endif
1269
1270    if( ret == NULL )
1271        ret = strip_non_utf8( in, inlen );
1272
1273    return ret;
1274}
1275
1276char*
1277tr_utf8clean( const char * str, int max_len )
1278{
1279    char * ret;
1280    const char * end;
1281
1282    if( max_len < 0 )
1283        max_len = (int) strlen( str );
1284
1285    if( tr_utf8_validate( str, max_len, &end  ) )
1286        ret = tr_strndup( str, max_len );
1287    else
1288        ret = to_utf8( str, max_len );
1289
1290    assert( tr_utf8_validate( ret, -1, NULL ) );
1291    return ret;
1292}
1293
1294/***
1295****
1296***/
1297
1298struct number_range
1299{
1300    int low;
1301    int high;
1302};
1303
1304/**
1305 * This should be a single number (ex. "6") or a range (ex. "6-9").
1306 * Anything else is an error and will return failure.
1307 */
1308static tr_bool
1309parseNumberSection( const char * str, int len, struct number_range * setme )
1310{
1311    long a, b;
1312    tr_bool success;
1313    char * end;
1314    const int error = errno;
1315    char * tmp = tr_strndup( str, len );
1316
1317    errno = 0;
1318    a = b = strtol( tmp, &end, 10 );
1319    if( errno || ( end == tmp ) ) {
1320        success = FALSE;
1321    } else if( *end != '-' ) {
1322        b = a;
1323        success = TRUE;
1324    } else {
1325        const char * pch = end + 1;
1326        b = strtol( pch, &end, 10 );
1327        if( errno || ( pch == end ) )
1328            success = FALSE;
1329        else if( *end ) /* trailing data */
1330            success = FALSE;
1331        else
1332            success = TRUE;
1333    }
1334    tr_free( tmp );
1335
1336    setme->low = MIN( a, b );
1337    setme->high = MAX( a, b );
1338
1339    errno = error;
1340    return success;
1341}
1342
1343int
1344compareInt( const void * va, const void * vb )
1345{
1346    const int a = *(const int *)va;
1347    const int b = *(const int *)vb;
1348    return a - b;
1349}
1350
1351/**
1352 * Given a string like "1-4" or "1-4,6,9,14-51", this allocates and returns an
1353 * array of setmeCount ints of all the values in the array.
1354 * For example, "5-8" will return [ 5, 6, 7, 8 ] and setmeCount will be 4.
1355 * It's the caller's responsibility to call tr_free() on the returned array.
1356 * If a fragment of the string can't be parsed, NULL is returned.
1357 */
1358int*
1359tr_parseNumberRange( const char * str_in, int len, int * setmeCount )
1360{
1361    int n = 0;
1362    int * uniq = NULL;
1363    char * str = tr_strndup( str_in, len );
1364    const char * walk;
1365    tr_list * ranges = NULL;
1366    tr_bool success = TRUE;
1367
1368    walk = str;
1369    while( walk && *walk && success ) {
1370        struct number_range range;
1371        const char * pch = strchr( walk, ',' );
1372        if( pch ) {
1373            success = parseNumberSection( walk, pch-walk, &range );
1374            walk = pch + 1;
1375        } else {
1376            success = parseNumberSection( walk, strlen( walk ), &range );
1377            walk += strlen( walk );
1378        }
1379        if( success )
1380            tr_list_append( &ranges, tr_memdup( &range, sizeof( struct number_range ) ) );
1381    }
1382
1383    if( !success )
1384    {
1385        *setmeCount = 0;
1386        uniq = NULL;
1387    }
1388    else
1389    {
1390        int i;
1391        int n2;
1392        tr_list * l;
1393        int * sorted = NULL;
1394
1395        /* build a sorted number array */
1396        n = n2 = 0;
1397        for( l=ranges; l!=NULL; l=l->next ) {
1398            const struct number_range * r = l->data;
1399            n += r->high + 1 - r->low;
1400        }
1401        sorted = tr_new( int, n );
1402        for( l=ranges; l!=NULL; l=l->next ) {
1403            const struct number_range * r = l->data;
1404            int i;
1405            for( i=r->low; i<=r->high; ++i )
1406                sorted[n2++] = i;
1407        }
1408        qsort( sorted, n, sizeof( int ), compareInt );
1409        assert( n == n2 );
1410
1411        /* remove duplicates */
1412        uniq = tr_new( int, n );
1413        for( i=n=0; i<n2; ++i )
1414            if( !n || uniq[n-1] != sorted[i] )
1415                uniq[n++] = sorted[i];
1416
1417        tr_free( sorted );
1418    }
1419
1420    /* cleanup */
1421    tr_list_free( &ranges, tr_free );
1422    tr_free( str );
1423
1424    /* return the result */
1425    *setmeCount = n;
1426    return uniq;
1427}
1428
1429/***
1430****
1431***/
1432
1433double
1434tr_truncd( double x, int decimal_places )
1435{
1436    /* sigh... surely there's a better way to do this */
1437    char buf[1024];
1438    const int i = (int) pow( 10, decimal_places );
1439    snprintf( buf, sizeof( buf ), "%f", x*i );
1440    *strchr(buf,'.') = '\0';
1441    return atof(buf) / i;
1442}
1443
1444char*
1445tr_strtruncd( char * buf, double x, int precision, size_t buflen )
1446{
1447    tr_snprintf( buf, buflen, "%.*f", precision, tr_truncd( x, precision ) );
1448    return buf;
1449}
1450
1451char*
1452tr_strpercent( char * buf, double x, size_t buflen )
1453{
1454    if( x < 10.0 )
1455        tr_strtruncd( buf, x, 2, buflen );
1456    else if( x < 100.0 )
1457        tr_strtruncd( buf, x, 1, buflen );
1458    else
1459        tr_strtruncd( buf, x, 0, buflen );
1460    return buf;
1461}
1462
1463char*
1464tr_strratio( char * buf, size_t buflen, double ratio, const char * infinity )
1465{
1466    if( (int)ratio == TR_RATIO_NA )
1467        tr_strlcpy( buf, _( "None" ), buflen );
1468    else if( (int)ratio == TR_RATIO_INF )
1469        tr_strlcpy( buf, infinity, buflen );
1470    else
1471        tr_strpercent( buf, ratio, buflen );
1472    return buf;
1473}
1474
1475/***
1476****
1477***/
1478
1479int
1480tr_moveFile( const char * oldpath, const char * newpath, tr_bool * renamed )
1481{
1482    int in;
1483    int out;
1484    char * buf;
1485    struct stat st;
1486    off_t bytesLeft;
1487    const size_t buflen = 1024 * 128; /* 128 KiB buffer */
1488
1489    /* make sure the old file exists */
1490    if( stat( oldpath, &st ) ) {
1491        const int err = errno;
1492        errno = err;
1493        return -1;
1494    }
1495    if( !S_ISREG( st.st_mode ) ) {
1496        errno = ENOENT;
1497        return -1;
1498    }
1499    bytesLeft = st.st_size;
1500
1501    /* make sure the target directory exists */
1502    {
1503        char * newdir = tr_dirname( newpath );
1504        int i = tr_mkdirp( newdir, 0777 );
1505        tr_free( newdir );
1506        if( i )
1507            return i;
1508    }
1509
1510    /* they might be on the same filesystem... */
1511    {
1512        const int i = rename( oldpath, newpath );
1513        if( renamed != NULL )
1514            *renamed = i == 0;
1515        if( !i )
1516            return 0;
1517    }
1518
1519    /* copy the file */
1520    in = tr_open_file_for_scanning( oldpath );
1521    out = tr_open_file_for_writing( newpath );
1522    buf = tr_valloc( buflen );
1523    while( bytesLeft > 0 )
1524    {
1525        ssize_t bytesWritten;
1526        const off_t bytesThisPass = MIN( bytesLeft, (off_t)buflen );
1527        const int numRead = read( in, buf, bytesThisPass );
1528        if( numRead < 0 )
1529            break;
1530        bytesWritten = write( out, buf, numRead );
1531        if( bytesWritten < 0 )
1532            break;
1533        bytesLeft -= bytesWritten;
1534    }
1535
1536    /* cleanup */
1537    tr_free( buf );
1538    tr_close_file( out );
1539    tr_close_file( in );
1540    if( bytesLeft != 0 )
1541        return -1;
1542
1543    unlink( oldpath );
1544    return 0;
1545}
1546
1547/***
1548****
1549***/
1550
1551void*
1552tr_valloc( size_t bufLen )
1553{
1554    size_t allocLen;
1555    void * buf = NULL;
1556    static size_t pageSize = 0;
1557
1558    if( !pageSize ) {
1559#ifdef HAVE_GETPAGESIZE
1560        pageSize = (size_t) getpagesize();
1561#else /* guess */
1562        pageSize = 4096;
1563#endif
1564    }
1565
1566    allocLen = pageSize;
1567    while( allocLen < bufLen )
1568        allocLen += pageSize;
1569
1570#ifdef HAVE_POSIX_MEMALIGN
1571    if( !buf )
1572        posix_memalign( &buf, pageSize, allocLen );
1573#endif
1574#ifdef HAVE_VALLOC
1575    if( !buf )
1576        buf = valloc( allocLen );
1577#endif
1578    if( !buf )
1579        buf = malloc( allocLen );
1580
1581    return buf;
1582}
1583
1584char *
1585tr_realpath( const char * path, char * resolved_path )
1586{
1587#ifdef WIN32
1588    /* From a message to the Mingw-msys list, Jun 2, 2005 by Mark Junker. */
1589    if( GetFullPathNameA( path, TR_PATH_MAX, resolved_path, NULL ) == 0 )
1590        return NULL;
1591    return resolved_path;
1592#else
1593    return realpath( path, resolved_path );
1594#endif
1595}
1596
1597/***
1598****
1599****
1600****
1601***/
1602
1603struct formatter_unit
1604{
1605    char * name;
1606    uint64_t value;
1607};
1608 
1609struct formatter_units
1610{
1611    struct formatter_unit units[4];
1612};
1613
1614enum { TR_FMT_KB, TR_FMT_MB, TR_FMT_GB, TR_FMT_TB };
1615
1616static void
1617formatter_init( struct formatter_units * units,
1618                unsigned int kilo,
1619                const char * kb, const char * mb,
1620                const char * gb, const char * tb )
1621{
1622    uint64_t value = kilo;
1623    units->units[TR_FMT_KB].name = tr_strdup( kb );
1624    units->units[TR_FMT_KB].value = value;
1625
1626    value *= kilo;
1627    units->units[TR_FMT_MB].name = tr_strdup( mb );
1628    units->units[TR_FMT_MB].value = value;
1629
1630    value *= kilo;
1631    units->units[TR_FMT_GB].name = tr_strdup( gb );
1632    units->units[TR_FMT_GB].value = value;
1633
1634    value *= kilo;
1635    units->units[TR_FMT_TB].name = tr_strdup( tb );
1636    units->units[TR_FMT_TB].value = value;
1637}
1638
1639static char*
1640formatter_get_size_str( const struct formatter_units * u,
1641                        char * buf, uint64_t bytes, size_t buflen )
1642{
1643    int precision;
1644    double value;
1645    const char * units;
1646    const struct formatter_unit * unit;
1647
1648         if( bytes < u->units[1].value ) unit = &u->units[0];
1649    else if( bytes < u->units[2].value ) unit = &u->units[1];
1650    else if( bytes < u->units[3].value ) unit = &u->units[2];
1651    else                                 unit = &u->units[3];
1652
1653    value = (double)bytes / unit->value;
1654    units = unit->name;
1655    if( unit->value == 1 )
1656        precision = 0;
1657    else if( value < 100 )
1658        precision = 2;
1659    else
1660        precision = 1;
1661    tr_snprintf( buf, buflen, "%.*f %s", precision, value, units );
1662    return buf;
1663}
1664
1665static struct formatter_units size_units;
1666
1667void
1668tr_formatter_size_init( unsigned int kilo,
1669                        const char * kb, const char * mb,
1670                        const char * gb, const char * tb )
1671{
1672    formatter_init( &size_units, kilo, kb, mb, gb, tb );
1673}
1674
1675char*
1676tr_formatter_size_B( char * buf, uint64_t bytes, size_t buflen )
1677{
1678    return formatter_get_size_str( &size_units, buf, bytes, buflen );
1679}
1680
1681static struct formatter_units speed_units;
1682
1683unsigned int tr_speed_K = 0u;
1684
1685void
1686tr_formatter_speed_init( unsigned int kilo,
1687                         const char * kb, const char * mb,
1688                         const char * gb, const char * tb )
1689{
1690    tr_speed_K = kilo;
1691    formatter_init( &speed_units, kilo, kb, mb, gb, tb );
1692}
1693
1694char*
1695tr_formatter_speed_KBps( char * buf, double KBps, size_t buflen )
1696{
1697    const double K = speed_units.units[TR_FMT_KB].value;
1698    double speed = KBps;
1699
1700    if( speed <= 999.95 ) /* 0.0 KB to 999.9 KB */
1701        tr_snprintf( buf, buflen, "%.2f %s", speed, speed_units.units[TR_FMT_KB].name );
1702    else {
1703        speed /= K;
1704        if( speed <= 99.995 ) /* 0.98 MB to 99.99 MB */
1705            tr_snprintf( buf, buflen, "%.2f %s", speed, speed_units.units[TR_FMT_MB].name );
1706        else if (speed <= 999.95) /* 100.0 MB to 999.9 MB */
1707            tr_snprintf( buf, buflen, "%.1f %s", speed, speed_units.units[TR_FMT_MB].name );
1708        else {
1709            speed /= K;
1710            tr_snprintf( buf, buflen, "%.1f %s", speed, speed_units.units[TR_FMT_GB].name );
1711        }
1712    }
1713
1714    return buf;
1715}
1716
1717static struct formatter_units mem_units;
1718
1719unsigned int tr_mem_K = 0u;
1720 
1721void
1722tr_formatter_mem_init( unsigned int kilo,
1723                       const char * kb, const char * mb,
1724                       const char * gb, const char * tb )
1725{
1726    tr_mem_K = kilo;
1727    formatter_init( &mem_units, kilo, kb, mb, gb, tb );
1728}
1729
1730char*
1731tr_formatter_mem_B( char * buf, uint64_t bytes_per_second, size_t buflen )
1732{
1733    return formatter_get_size_str( &mem_units, buf, bytes_per_second, buflen );
1734}
1735
1736void
1737tr_formatter_get_units( tr_benc * d )
1738{
1739    int i;
1740    tr_benc * l;
1741
1742    tr_bencDictReserve( d, 6 );
1743
1744    tr_bencDictAddInt( d, "memory-bytes", mem_units.units[TR_FMT_KB].value );
1745    l = tr_bencDictAddList( d, "memory-units", 4 );
1746    for( i=0; i<4; i++ ) tr_bencListAddStr( l, mem_units.units[i].name );
1747
1748    tr_bencDictAddInt( d, "size-bytes",   size_units.units[TR_FMT_KB].value );
1749    l = tr_bencDictAddList( d, "size-units", 4 );
1750    for( i=0; i<4; i++ ) tr_bencListAddStr( l, size_units.units[i].name );
1751
1752    tr_bencDictAddInt( d, "speed-bytes",  speed_units.units[TR_FMT_KB].value );
1753    l = tr_bencDictAddList( d, "speed-units", 4 );
1754    for( i=0; i<4; i++ ) tr_bencListAddStr( l, speed_units.units[i].name );
1755}
1756
Note: See TracBrowser for help on using the repository browser.