source: trunk/libtransmission/utils.c @ 12292

Last change on this file since 12292 was 12292, checked in by jordan, 11 years ago

(trunk libT) avoid an unnecessary malloc/free in tr_urlIsValidTracker() and tr_urlIsValid()

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