source: trunk/libtransmission/utils.c @ 11490

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

(trunk libT) #3813 "libtransmission doesn't build on Solaris 10 because of strsep call()" -- fixed.

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