source: trunk/libtransmission/utils.c @ 12157

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

(trunk libT) better implementation of tr_htonll() and tr_ntohll()

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