source: trunk/libtransmission/utils.c @ 11599

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

(trunk) Join the 21st century and use only 1 space at the end sentences. This commit is nearly as important as the semi-annual ones that remove trailing spaces from the ends of lines of code... :)

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