source: trunk/libtransmission/utils.c @ 11714

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

(trunk) #3914 "strlsize() passes variable of incompatible type to the tr_formatter_size_B()" -- fixed.

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