source: trunk/libtransmission/utils.c @ 13470

Last change on this file since 13470 was 13470, checked in by livings124, 10 years ago

Whoops! Revert new changes committed in r13469

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