/* * This file Copyright (C) Mnemosyne LLC * * This file is licensed by the GPL version 2. Works owned by the * Transmission project are granted a special exemption to clause 2(b) * so that the bulk of its code can remain under the MIT license. * This exemption does not extend to derived works not owned by * the Transmission project. * * $Id: utils.c 13191 2012-02-03 16:44:07Z jordan $ */ #ifdef HAVE_MEMMEM #define _GNU_SOURCE /* glibc's string.h needs this to pick up memmem */ #endif #if defined(SYS_DARWIN) #define HAVE_GETPAGESIZE #define HAVE_ICONV_OPEN #define HAVE_MKDTEMP #define HAVE_VALLOC #endif #include #include /* isdigit(), isalpha(), tolower() */ #include #include /* DBL_EPSILON */ #include /* localeconv() */ #include /* pow(), fabs(), floor() */ #include #include #include #include /* strerror(), memset(), memmem() */ #include /* nanosleep() */ #ifdef HAVE_ICONV_OPEN #include #endif #include /* basename() */ #include #include #include #include /* stat(), getcwd(), getpagesize(), unlink() */ #include #include #ifdef WIN32 #include #define WINVER WindowsXP /* freeaddrinfo(), getaddrinfo(), getnameinfo() */ #include /* _getcwd() */ #include /* Sleep() */ #endif #include "transmission.h" #include "bencode.h" #include "fdlimit.h" #include "ConvertUTF.h" #include "list.h" #include "utils.h" #include "platform.h" /* tr_lockLock(), TR_PATH_MAX */ #include "version.h" time_t __tr_current_time = 0; tr_msg_level __tr_message_level = TR_MSG_ERR; static bool messageQueuing = false; static tr_msg_list * messageQueue = NULL; static tr_msg_list ** messageQueueTail = &messageQueue; static int messageQueueCount = 0; #ifndef WIN32 /* make null versions of these win32 functions */ static inline int IsDebuggerPresent( void ) { return false; } static inline void OutputDebugString( const void * unused UNUSED ) { } #endif /*** **** ***/ static tr_lock* getMessageLock( void ) { static tr_lock * l = NULL; if( !l ) l = tr_lockNew( ); return l; } void* tr_getLog( void ) { static bool initialized = false; static FILE * file = NULL; if( !initialized ) { const char * str = getenv( "TR_DEBUG_FD" ); int fd = 0; if( str && *str ) fd = atoi( str ); switch( fd ) { case 1: file = stdout; break; case 2: file = stderr; break; default: file = NULL; break; } initialized = true; } return file; } void tr_setMessageLevel( tr_msg_level level ) { __tr_message_level = level; } void tr_setMessageQueuing( bool enabled ) { messageQueuing = enabled; } bool tr_getMessageQueuing( void ) { return messageQueuing != 0; } tr_msg_list * tr_getQueuedMessages( void ) { tr_msg_list * ret; tr_lockLock( getMessageLock( ) ); ret = messageQueue; messageQueue = NULL; messageQueueTail = &messageQueue; messageQueueCount = 0; tr_lockUnlock( getMessageLock( ) ); return ret; } void tr_freeMessageList( tr_msg_list * list ) { tr_msg_list * next; while( NULL != list ) { next = list->next; free( list->message ); free( list->name ); free( list ); list = next; } } /** *** **/ struct tm * tr_localtime_r( const time_t *_clock, struct tm *_result ) { #ifdef HAVE_LOCALTIME_R return localtime_r( _clock, _result ); #else struct tm *p = localtime( _clock ); if( p ) *(_result) = *p; return p; #endif } char* tr_getLogTimeStr( char * buf, int buflen ) { char tmp[64]; struct tm now_tm; struct timeval tv; time_t seconds; int milliseconds; gettimeofday( &tv, NULL ); seconds = tv.tv_sec; tr_localtime_r( &seconds, &now_tm ); strftime( tmp, sizeof( tmp ), "%H:%M:%S", &now_tm ); milliseconds = tv.tv_usec / 1000; tr_snprintf( buf, buflen, "%s.%03d", tmp, milliseconds ); return buf; } bool tr_deepLoggingIsActive( void ) { static int8_t deepLoggingIsActive = -1; if( deepLoggingIsActive < 0 ) deepLoggingIsActive = IsDebuggerPresent() || (tr_getLog()!=NULL); return deepLoggingIsActive != 0; } void tr_deepLog( const char * file, int line, const char * name, const char * fmt, ... ) { FILE * fp = tr_getLog( ); if( fp || IsDebuggerPresent( ) ) { va_list args; char timestr[64]; struct evbuffer * buf = evbuffer_new( ); char * base = tr_basename( file ); evbuffer_add_printf( buf, "[%s] ", tr_getLogTimeStr( timestr, sizeof( timestr ) ) ); if( name ) evbuffer_add_printf( buf, "%s ", name ); va_start( args, fmt ); evbuffer_add_vprintf( buf, fmt, args ); va_end( args ); evbuffer_add_printf( buf, " (%s:%d)\n", base, line ); /* FIXME(libevent2) ifdef this out for nonwindows platforms */ OutputDebugString( evbuffer_pullup( buf, -1 ) ); if( fp ) fputs( (const char*)evbuffer_pullup( buf, -1 ), fp ); tr_free( base ); evbuffer_free( buf ); } } /*** **** ***/ void tr_msg( const char * file, int line, tr_msg_level level, const char * name, const char * fmt, ... ) { const int err = errno; /* message logging shouldn't affect errno */ char buf[1024]; va_list ap; tr_lockLock( getMessageLock( ) ); /* build the text message */ *buf = '\0'; va_start( ap, fmt ); evutil_vsnprintf( buf, sizeof( buf ), fmt, ap ); va_end( ap ); OutputDebugString( buf ); if( *buf ) { if( messageQueuing ) { tr_msg_list * newmsg; newmsg = tr_new0( tr_msg_list, 1 ); newmsg->level = level; newmsg->when = tr_time( ); newmsg->message = tr_strdup( buf ); newmsg->file = file; newmsg->line = line; newmsg->name = tr_strdup( name ); *messageQueueTail = newmsg; messageQueueTail = &newmsg->next; ++messageQueueCount; if( messageQueueCount > TR_MAX_MSG_LOG ) { tr_msg_list * old = messageQueue; messageQueue = old->next; old->next = NULL; tr_freeMessageList(old); --messageQueueCount; assert( messageQueueCount == TR_MAX_MSG_LOG ); } } else { char timestr[64]; FILE * fp; fp = tr_getLog( ); if( fp == NULL ) fp = stderr; tr_getLogTimeStr( timestr, sizeof( timestr ) ); if( name ) fprintf( fp, "[%s] %s: %s\n", timestr, name, buf ); else fprintf( fp, "[%s] %s\n", timestr, buf ); fflush( fp ); } } tr_lockUnlock( getMessageLock( ) ); errno = err; } /*** **** ***/ void* tr_malloc( size_t size ) { return size ? malloc( size ) : NULL; } void* tr_malloc0( size_t size ) { return size ? calloc( 1, size ) : NULL; } void tr_free( void * p ) { if( p != NULL ) free( p ); } void* tr_memdup( const void * src, size_t byteCount ) { return memcpy( tr_malloc( byteCount ), src, byteCount ); } /*** **** ***/ const char* tr_strip_positional_args( const char* str ) { const char * in = str; static size_t bufsize = 0; static char * buf = NULL; const size_t len = str ? strlen( str ) : 0; char * out; if( !buf || ( bufsize < len ) ) { bufsize = len * 2 + 1; buf = tr_renew( char, buf, bufsize ); } for( out = buf; str && *str; ++str ) { *out++ = *str; if( ( *str == '%' ) && isdigit( str[1] ) ) { const char * tmp = str + 1; while( isdigit( *tmp ) ) ++tmp; if( *tmp == '$' ) str = tmp[1]=='\'' ? tmp+1 : tmp; } if( ( *str == '%' ) && ( str[1] == '\'' ) ) str = str + 1; } *out = '\0'; return !in || strcmp( buf, in ) ? buf : in; } /** *** **/ void tr_timerAdd( struct event * timer, int seconds, int microseconds ) { struct timeval tv; tv.tv_sec = seconds; tv.tv_usec = microseconds; assert( tv.tv_sec >= 0 ); assert( tv.tv_usec >= 0 ); assert( tv.tv_usec < 1000000 ); evtimer_add( timer, &tv ); } void tr_timerAddMsec( struct event * timer, int msec ) { const int seconds = msec / 1000; const int usec = (msec%1000) * 1000; tr_timerAdd( timer, seconds, usec ); } /** *** **/ uint8_t * tr_loadFile( const char * path, size_t * size ) { uint8_t * buf; struct stat sb; int fd; ssize_t n; const char * const err_fmt = _( "Couldn't read \"%1$s\": %2$s" ); /* try to stat the file */ errno = 0; if( stat( path, &sb ) ) { const int err = errno; tr_dbg( err_fmt, path, tr_strerror( errno ) ); errno = err; return NULL; } if( ( sb.st_mode & S_IFMT ) != S_IFREG ) { tr_err( err_fmt, path, _( "Not a regular file" ) ); errno = EISDIR; return NULL; } /* Load the torrent file into our buffer */ fd = tr_open_file_for_scanning( path ); if( fd < 0 ) { const int err = errno; tr_err( err_fmt, path, tr_strerror( errno ) ); errno = err; return NULL; } buf = tr_malloc( sb.st_size + 1 ); if( !buf ) { const int err = errno; tr_err( err_fmt, path, _( "Memory allocation failed" ) ); tr_close_file( fd ); errno = err; return NULL; } n = read( fd, buf, (size_t)sb.st_size ); if( n == -1 ) { const int err = errno; tr_err( err_fmt, path, tr_strerror( errno ) ); tr_close_file( fd ); free( buf ); errno = err; return NULL; } tr_close_file( fd ); buf[ sb.st_size ] = '\0'; *size = sb.st_size; return buf; } char* tr_basename( const char * path ) { char * tmp = tr_strdup( path ); char * ret = tr_strdup( basename( tmp ) ); tr_free( tmp ); return ret; } char* tr_dirname( const char * path ) { char * tmp = tr_strdup( path ); char * ret = tr_strdup( dirname( tmp ) ); tr_free( tmp ); return ret; } char* tr_mkdtemp( char * template ) { #ifdef HAVE_MKDTEMP return mkdtemp( template ); #else if( !mktemp( template ) || mkdir( template, 0700 ) ) return NULL; return template; #endif } int tr_mkdir( const char * path, int permissions #ifdef WIN32 UNUSED #endif ) { #ifdef WIN32 if( path && isalpha( path[0] ) && path[1] == ':' && !path[2] ) return 0; return mkdir( path ); #else return mkdir( path, permissions ); #endif } int tr_mkdirp( const char * path_in, int permissions ) { char * path = tr_strdup( path_in ); char * p, * pp; struct stat sb; int done; /* walk past the root */ p = path; while( *p == TR_PATH_DELIMITER ) ++p; pp = p; done = 0; while( ( p = strchr( pp, TR_PATH_DELIMITER ) ) || ( p = strchr( pp, '\0' ) ) ) { if( !*p ) done = 1; else *p = '\0'; if( stat( path, &sb ) ) { /* Folder doesn't exist yet */ if( tr_mkdir( path, permissions ) ) { const int err = errno; tr_err( _( "Couldn't create \"%1$s\": %2$s" ), path, tr_strerror( err ) ); tr_free( path ); errno = err; return -1; } } else if( ( sb.st_mode & S_IFMT ) != S_IFDIR ) { /* Node exists but isn't a folder */ char * buf = tr_strdup_printf( _( "File \"%s\" is in the way" ), path ); tr_err( _( "Couldn't create \"%1$s\": %2$s" ), path_in, buf ); tr_free( buf ); tr_free( path ); errno = ENOTDIR; return -1; } if( done ) break; *p = TR_PATH_DELIMITER; p++; pp = p; } tr_free( path ); return 0; } char* tr_buildPath( const char *first_element, ... ) { size_t bufLen = 0; const char * element; char * buf; char * pch; va_list vl; /* pass 1: allocate enough space for the string */ va_start( vl, first_element ); element = first_element; while( element ) { bufLen += strlen( element ) + 1; element = va_arg( vl, const char* ); } pch = buf = tr_new( char, bufLen ); va_end( vl ); /* pass 2: build the string piece by piece */ va_start( vl, first_element ); element = first_element; while( element ) { const size_t elementLen = strlen( element ); memcpy( pch, element, elementLen ); pch += elementLen; *pch++ = TR_PATH_DELIMITER; element = va_arg( vl, const char* ); } va_end( vl ); /* terminate the string. if nonempty, eat the unwanted trailing slash */ if( pch != buf ) --pch; *pch++ = '\0'; /* sanity checks & return */ assert( pch - buf == (off_t)bufLen ); return buf; } /**** ***** ****/ char* evbuffer_free_to_str( struct evbuffer * buf ) { const size_t n = evbuffer_get_length( buf ); char * ret = tr_new( char, n + 1 ); evbuffer_copyout( buf, ret, n ); evbuffer_free( buf ); ret[n] = '\0'; return ret; } char* tr_strdup( const void * in ) { return tr_strndup( in, in ? (int)strlen((const char *)in) : 0 ); } char* tr_strndup( const void * in, int len ) { char * out = NULL; if( len < 0 ) { out = tr_strdup( in ); } else if( in ) { out = tr_malloc( len + 1 ); memcpy( out, in, len ); out[len] = '\0'; } return out; } const char* tr_memmem( const char * haystack, size_t haystacklen, const char * needle, size_t needlelen ) { #ifdef HAVE_MEMMEM return memmem( haystack, haystacklen, needle, needlelen ); #else size_t i; if( !needlelen ) return haystack; if( needlelen > haystacklen || !haystack || !needle ) return NULL; for( i=0; i<=haystacklen-needlelen; ++i ) if( !memcmp( haystack+i, needle, needlelen ) ) return haystack+i; return NULL; #endif } char* tr_strdup_printf( const char * fmt, ... ) { va_list ap; char * ret; size_t len; char statbuf[2048]; va_start( ap, fmt ); len = evutil_vsnprintf( statbuf, sizeof( statbuf ), fmt, ap ); va_end( ap ); if( len < sizeof( statbuf ) ) ret = tr_strndup( statbuf, len ); else { ret = tr_new( char, len + 1 ); va_start( ap, fmt ); evutil_vsnprintf( ret, len + 1, fmt, ap ); va_end( ap ); } return ret; } const char* tr_strerror( int i ) { const char * ret = strerror( i ); if( ret == NULL ) ret = "Unknown Error"; return ret; } int tr_strcmp0( const char * str1, const char * str2 ) { if( str1 && str2 ) return strcmp( str1, str2 ); if( str1 ) return 1; if( str2 ) return -1; return 0; } /**** ***** ****/ /* https://bugs.launchpad.net/percona-patches/+bug/526863/+attachment/1160199/+files/solaris_10_fix.patch */ char* tr_strsep( char ** str, const char * delims ) { #ifdef HAVE_STRSEP return strsep( str, delims ); #else char *token; if (*str == NULL) { /* No more tokens */ return NULL; } token = *str; while (**str != '\0') { if (strchr(delims, **str) != NULL) { **str = '\0'; (*str)++; return token; } (*str)++; } /* There is not another token */ *str = NULL; return token; #endif } char* tr_strstrip( char * str ) { if( str != NULL ) { size_t pos; size_t len = strlen( str ); while( len && isspace( str[len - 1] ) ) --len; for( pos = 0; pos < len && isspace( str[pos] ); ) ++pos; len -= pos; memmove( str, str + pos, len ); str[len] = '\0'; } return str; } bool tr_str_has_suffix( const char *str, const char *suffix ) { size_t str_len; size_t suffix_len; if( !str ) return false; if( !suffix ) return true; str_len = strlen( str ); suffix_len = strlen( suffix ); if( str_len < suffix_len ) return false; return !evutil_ascii_strncasecmp( str + str_len - suffix_len, suffix, suffix_len ); } /**** ***** ****/ uint64_t tr_time_msec( void ) { struct timeval tv; gettimeofday( &tv, NULL ); return (uint64_t) tv.tv_sec * 1000 + ( tv.tv_usec / 1000 ); } void tr_wait_msec( long int msec ) { #ifdef WIN32 Sleep( (DWORD)msec ); #else struct timespec ts; ts.tv_sec = msec / 1000; ts.tv_nsec = ( msec % 1000 ) * 1000000; nanosleep( &ts, NULL ); #endif } /*** **** ***/ int tr_snprintf( char * buf, size_t buflen, const char * fmt, ... ) { int len; va_list args; va_start( args, fmt ); len = evutil_vsnprintf( buf, buflen, fmt, args ); va_end( args ); return len; } /* * Copy src to string dst of size siz. At most siz-1 characters * will be copied. Always NUL terminates (unless siz == 0). * Returns strlen(src); if retval >= siz, truncation occurred. */ size_t tr_strlcpy( char * dst, const void * src, size_t siz ) { #ifdef HAVE_STRLCPY return strlcpy( dst, src, siz ); #else char * d = dst; const char *s = src; size_t n = siz; assert( s ); assert( d ); /* Copy as many bytes as will fit */ if( n != 0 ) { while( --n != 0 ) { if( ( *d++ = *s++ ) == '\0' ) break; } } /* Not enough room in dst, add NUL and traverse rest of src */ if( n == 0 ) { if( siz != 0 ) *d = '\0'; /* NUL-terminate dst */ while( *s++ ) ; } return s - (char*)src - 1; /* count does not include NUL */ #endif } /*** **** ***/ double tr_getRatio( uint64_t numerator, uint64_t denominator ) { double ratio; if( denominator > 0 ) ratio = numerator / (double)denominator; else if( numerator > 0 ) ratio = TR_RATIO_INF; else ratio = TR_RATIO_NA; return ratio; } void tr_sha1_to_hex( char * out, const uint8_t * sha1 ) { int i; static const char hex[] = "0123456789abcdef"; for( i=0; i<20; ++i ) { const unsigned int val = *sha1++; *out++ = hex[val >> 4]; *out++ = hex[val & 0xf]; } *out = '\0'; } void tr_hex_to_sha1( uint8_t * out, const char * in ) { int i; static const char hex[] = "0123456789abcdef"; for( i=0; i<20; ++i ) { const int hi = strchr( hex, tolower( *in++ ) ) - hex; const int lo = strchr( hex, tolower( *in++ ) ) - hex; *out++ = (uint8_t)( (hi<<4) | lo ); } } /*** **** ***/ static bool isValidURLChars( const char * url, int url_len ) { const char * c; const char * end; static const char * rfc2396_valid_chars = "abcdefghijklmnopqrstuvwxyz" /* lowalpha */ "ABCDEFGHIJKLMNOPQRSTUVWXYZ" /* upalpha */ "0123456789" /* digit */ "-_.!~*'()" /* mark */ ";/?:@&=+$," /* reserved */ "<>#%<\"" /* delims */ "{}|\\^[]`"; /* unwise */ if( url == NULL ) return false; for( c=url, end=c+url_len; c && *c && c!=end; ++c ) if( !strchr( rfc2396_valid_chars, *c ) ) return false; return true; } /** @brief return true if the URL is a http or https or UDP one that Transmission understands */ bool tr_urlIsValidTracker( const char * url ) { bool valid; if( url == NULL ) { valid = false; } else { const int len = strlen( url ); valid = isValidURLChars( url, len ) && !tr_urlParse( url, len, NULL, NULL, NULL, NULL ) && ( !memcmp(url,"http://",7) || !memcmp(url,"https://",8) || !memcmp(url,"udp://",6) ); } return valid; } /** @brief return true if the URL is a http or https or ftp or sftp one that Transmission understands */ bool tr_urlIsValid( const char * url, int url_len ) { bool valid; if( url == NULL ) { valid = false; } else { if( url_len < 0 ) url_len = strlen( url ); valid = isValidURLChars( url, url_len ) && !tr_urlParse( url, url_len, NULL, NULL, NULL, NULL ) && ( !memcmp(url,"http://",7) || !memcmp(url,"https://",8) || !memcmp(url,"ftp://",6) || !memcmp(url,"sftp://",7) ); } return valid; } bool tr_addressIsIP( const char * str ) { tr_address tmp; return tr_address_from_string( &tmp, str ); } int tr_urlParse( const char * url_in, int len, char ** setme_protocol, char ** setme_host, int * setme_port, char ** setme_path ) { int err; int port = 0; int n; char * tmp; char * pch; size_t host_len; size_t protocol_len; const char * host = NULL; const char * protocol = NULL; const char * path = NULL; tmp = tr_strndup( url_in, len ); if( ( pch = strstr( tmp, "://" ) ) ) { *pch = '\0'; protocol = tmp; protocol_len = pch - protocol; pch += 3; /*fprintf( stderr, "protocol is [%s]... what's left is [%s]\n", protocol, pch);*/ if( ( n = strcspn( pch, ":/" ) ) ) { const int havePort = pch[n] == ':'; host = pch; host_len = n; pch += n; if( pch && *pch ) *pch++ = '\0'; /*fprintf( stderr, "host is [%s]... what's left is [%s]\n", host, pch );*/ if( havePort ) { char * end; port = strtol( pch, &end, 10 ); pch = end; /*fprintf( stderr, "port is [%d]... what's left is [%s]\n", port, pch );*/ } path = pch; /*fprintf( stderr, "path is [%s]\n", path );*/ } } err = !host || !path || !protocol; if( !err && !port ) { if( !strcmp( protocol, "udp" ) ) port = 80; else if( !strcmp( protocol, "ftp" ) ) port = 21; else if( !strcmp( protocol, "sftp" ) ) port = 22; else if( !strcmp( protocol, "http" ) ) port = 80; else if( !strcmp( protocol, "https" ) ) port = 443; } if( !err ) { if( setme_protocol ) *setme_protocol = tr_strndup( protocol, protocol_len ); if( setme_host ){ ( (char*)host )[-3] = ':'; *setme_host = tr_strndup( host, host_len ); } if( setme_path ){ if( !*path ) *setme_path = tr_strdup( "/" ); else if( path[0] == '/' ) *setme_path = tr_strdup( path ); else { ( (char*)path )[-1] = '/'; *setme_path = tr_strdup( path - 1 ); } } if( setme_port ) *setme_port = port; } tr_free( tmp ); return err; } #include #include #include #include #include #include char * tr_base64_encode( const void * input, int length, int * setme_len ) { int retlen = 0; char * ret = NULL; if( input != NULL ) { BIO * b64; BIO * bmem; BUF_MEM * bptr; if( length < 1 ) length = (int)strlen( input ); bmem = BIO_new( BIO_s_mem( ) ); b64 = BIO_new( BIO_f_base64( ) ); BIO_set_flags( b64, BIO_FLAGS_BASE64_NO_NL ); b64 = BIO_push( b64, bmem ); BIO_write( b64, input, length ); (void) BIO_flush( b64 ); BIO_get_mem_ptr( b64, &bptr ); ret = tr_strndup( bptr->data, bptr->length ); retlen = bptr->length; BIO_free_all( b64 ); } if( setme_len ) *setme_len = retlen; return ret; } char * tr_base64_decode( const void * input, int length, int * setme_len ) { char * ret; BIO * b64; BIO * bmem; int retlen; if( length < 1 ) length = strlen( input ); ret = tr_new0( char, length ); b64 = BIO_new( BIO_f_base64( ) ); bmem = BIO_new_mem_buf( (unsigned char*)input, length ); bmem = BIO_push( b64, bmem ); retlen = BIO_read( bmem, ret, length ); if( !retlen ) { /* try again, but with the BIO_FLAGS_BASE64_NO_NL flag */ BIO_free_all( bmem ); b64 = BIO_new( BIO_f_base64( ) ); BIO_set_flags( b64, BIO_FLAGS_BASE64_NO_NL ); bmem = BIO_new_mem_buf( (unsigned char*)input, length ); bmem = BIO_push( b64, bmem ); retlen = BIO_read( bmem, ret, length ); } if( setme_len ) *setme_len = retlen; BIO_free_all( bmem ); return ret; } /*** **** ***/ void tr_removeElementFromArray( void * array, unsigned int index_to_remove, size_t sizeof_element, size_t nmemb ) { char * a = array; memmove( a + sizeof_element * index_to_remove, a + sizeof_element * ( index_to_remove + 1 ), sizeof_element * ( --nmemb - index_to_remove ) ); } int tr_lowerBound( const void * key, const void * base, size_t nmemb, size_t size, int (* compar)(const void* key, const void* arrayMember), bool * exact_match ) { size_t first = 0; const char * cbase = base; bool exact = false; while( nmemb != 0 ) { const size_t half = nmemb / 2; const size_t middle = first + half; const int c = compar( key, cbase + size*middle ); if( c <= 0 ) { if( c == 0 ) exact = true; nmemb = half; } else { first = middle + 1; nmemb = nmemb - half - 1; } } *exact_match = exact; return first; } /*** **** ***/ static char* strip_non_utf8( const char * in, size_t inlen ) { const char * end; const char zero = '\0'; struct evbuffer * buf = evbuffer_new( ); while( !tr_utf8_validate( in, inlen, &end ) ) { const int good_len = end - in; evbuffer_add( buf, in, good_len ); inlen -= ( good_len + 1 ); in += ( good_len + 1 ); evbuffer_add( buf, "?", 1 ); } evbuffer_add( buf, in, inlen ); evbuffer_add( buf, &zero, 1 ); return evbuffer_free_to_str( buf ); } static char* to_utf8( const char * in, size_t inlen ) { char * ret = NULL; #ifdef HAVE_ICONV_OPEN int i; const char * encodings[] = { "CURRENT", "ISO-8859-15" }; const int encoding_count = sizeof(encodings) / sizeof(encodings[1]); const size_t buflen = inlen*4 + 10; char * out = tr_new( char, buflen ); for( i=0; !ret && ilow = MIN( a, b ); setme->high = MAX( a, b ); errno = error; return success; } int compareInt( const void * va, const void * vb ) { const int a = *(const int *)va; const int b = *(const int *)vb; return a - b; } /** * Given a string like "1-4" or "1-4,6,9,14-51", this allocates and returns an * array of setmeCount ints of all the values in the array. * For example, "5-8" will return [ 5, 6, 7, 8 ] and setmeCount will be 4. * It's the caller's responsibility to call tr_free() on the returned array. * If a fragment of the string can't be parsed, NULL is returned. */ int* tr_parseNumberRange( const char * str_in, int len, int * setmeCount ) { int n = 0; int * uniq = NULL; char * str = tr_strndup( str_in, len ); const char * walk; tr_list * ranges = NULL; bool success = true; walk = str; while( walk && *walk && success ) { struct number_range range; const char * pch = strchr( walk, ',' ); if( pch ) { success = parseNumberSection( walk, pch-walk, &range ); walk = pch + 1; } else { success = parseNumberSection( walk, strlen( walk ), &range ); walk += strlen( walk ); } if( success ) tr_list_append( &ranges, tr_memdup( &range, sizeof( struct number_range ) ) ); } if( !success ) { *setmeCount = 0; uniq = NULL; } else { int i; int n2; tr_list * l; int * sorted = NULL; /* build a sorted number array */ n = n2 = 0; for( l=ranges; l!=NULL; l=l->next ) { const struct number_range * r = l->data; n += r->high + 1 - r->low; } sorted = tr_new( int, n ); for( l=ranges; l!=NULL; l=l->next ) { const struct number_range * r = l->data; int i; for( i=r->low; i<=r->high; ++i ) sorted[n2++] = i; } qsort( sorted, n, sizeof( int ), compareInt ); assert( n == n2 ); /* remove duplicates */ uniq = tr_new( int, n ); for( i=n=0; idecimal_point ))) pt[precision ? precision+1 : 0] = '\0'; return atof(buf); } /* return a truncated double as a string */ static char* tr_strtruncd( char * buf, double x, int precision, size_t buflen ) { tr_snprintf( buf, buflen, "%.*f", precision, tr_truncd( x, precision ) ); return buf; } char* tr_strpercent( char * buf, double x, size_t buflen ) { if( x < 10.0 ) tr_strtruncd( buf, x, 2, buflen ); else if( x < 100.0 ) tr_strtruncd( buf, x, 1, buflen ); else tr_strtruncd( buf, x, 0, buflen ); return buf; } char* tr_strratio( char * buf, size_t buflen, double ratio, const char * infinity ) { if( (int)ratio == TR_RATIO_NA ) tr_strlcpy( buf, _( "None" ), buflen ); else if( (int)ratio == TR_RATIO_INF ) tr_strlcpy( buf, infinity, buflen ); else tr_strpercent( buf, ratio, buflen ); return buf; } /*** **** ***/ int tr_moveFile( const char * oldpath, const char * newpath, bool * renamed ) { int in; int out; char * buf; struct stat st; off_t bytesLeft; const size_t buflen = 1024 * 128; /* 128 KiB buffer */ /* make sure the old file exists */ if( stat( oldpath, &st ) ) { const int err = errno; errno = err; return -1; } if( !S_ISREG( st.st_mode ) ) { errno = ENOENT; return -1; } bytesLeft = st.st_size; /* make sure the target directory exists */ { char * newdir = tr_dirname( newpath ); int i = tr_mkdirp( newdir, 0777 ); tr_free( newdir ); if( i ) return i; } /* they might be on the same filesystem... */ { const int i = rename( oldpath, newpath ); if( renamed != NULL ) *renamed = i == 0; if( !i ) return 0; } /* copy the file */ in = tr_open_file_for_scanning( oldpath ); out = tr_open_file_for_writing( newpath ); buf = tr_valloc( buflen ); while( bytesLeft > 0 ) { ssize_t bytesWritten; const off_t bytesThisPass = MIN( bytesLeft, (off_t)buflen ); const int numRead = read( in, buf, bytesThisPass ); if( numRead < 0 ) break; bytesWritten = write( out, buf, numRead ); if( bytesWritten < 0 ) break; bytesLeft -= bytesWritten; } /* cleanup */ tr_free( buf ); tr_close_file( out ); tr_close_file( in ); if( bytesLeft != 0 ) return -1; unlink( oldpath ); return 0; } bool tr_is_same_file( const char * filename1, const char * filename2 ) { struct stat sb1, sb2; return !stat( filename1, &sb1 ) && !stat( filename2, &sb2 ) && ( sb1.st_dev == sb2.st_dev ) && ( sb1.st_ino == sb2.st_ino ); } /*** **** ***/ void* tr_valloc( size_t bufLen ) { size_t allocLen; void * buf = NULL; static size_t pageSize = 0; if( !pageSize ) { #ifdef HAVE_GETPAGESIZE pageSize = (size_t) getpagesize(); #else /* guess */ pageSize = 4096; #endif } allocLen = pageSize; while( allocLen < bufLen ) allocLen += pageSize; #ifdef HAVE_POSIX_MEMALIGN if( !buf ) if( posix_memalign( &buf, pageSize, allocLen ) ) buf = NULL; /* just retry with valloc/malloc */ #endif #ifdef HAVE_VALLOC if( !buf ) buf = valloc( allocLen ); #endif if( !buf ) buf = tr_malloc( allocLen ); return buf; } char * tr_realpath( const char * path, char * resolved_path ) { #ifdef WIN32 /* From a message to the Mingw-msys list, Jun 2, 2005 by Mark Junker. */ if( GetFullPathNameA( path, TR_PATH_MAX, resolved_path, NULL ) == 0 ) return NULL; return resolved_path; #else return realpath( path, resolved_path ); #endif } /*** **** ***/ uint64_t tr_htonll( uint64_t x ) { #ifdef HAVE_HTONLL return htonll( x ); #else /* fallback code by bdonlan at * http://stackoverflow.com/questions/809902/64-bit-ntohl-in-c/875505#875505 */ union { uint32_t lx[2]; uint64_t llx; } u; u.lx[0] = htonl(x >> 32); u.lx[1] = htonl(x & 0xFFFFFFFFULL); return u.llx; #endif } uint64_t tr_ntohll( uint64_t x ) { #ifdef HAVE_NTOHLL return ntohll( x ); #else /* fallback code by bdonlan at * http://stackoverflow.com/questions/809902/64-bit-ntohl-in-c/875505#875505 */ union { uint32_t lx[2]; uint64_t llx; } u; u.llx = x; return ((uint64_t)ntohl(u.lx[0]) << 32) | (uint64_t)ntohl(u.lx[1]); #endif } /*** **** **** **** ***/ struct formatter_unit { char * name; int64_t value; }; struct formatter_units { struct formatter_unit units[4]; }; enum { TR_FMT_KB, TR_FMT_MB, TR_FMT_GB, TR_FMT_TB }; static void formatter_init( struct formatter_units * units, unsigned int kilo, const char * kb, const char * mb, const char * gb, const char * tb ) { uint64_t value = kilo; units->units[TR_FMT_KB].name = tr_strdup( kb ); units->units[TR_FMT_KB].value = value; value *= kilo; units->units[TR_FMT_MB].name = tr_strdup( mb ); units->units[TR_FMT_MB].value = value; value *= kilo; units->units[TR_FMT_GB].name = tr_strdup( gb ); units->units[TR_FMT_GB].value = value; value *= kilo; units->units[TR_FMT_TB].name = tr_strdup( tb ); units->units[TR_FMT_TB].value = value; } static char* formatter_get_size_str( const struct formatter_units * u, char * buf, int64_t bytes, size_t buflen ) { int precision; double value; const char * units; const struct formatter_unit * unit; if( bytes < u->units[1].value ) unit = &u->units[0]; else if( bytes < u->units[2].value ) unit = &u->units[1]; else if( bytes < u->units[3].value ) unit = &u->units[2]; else unit = &u->units[3]; value = (double)bytes / unit->value; units = unit->name; if( unit->value == 1 ) precision = 0; else if( value < 100 ) precision = 2; else precision = 1; tr_snprintf( buf, buflen, "%.*f %s", precision, value, units ); return buf; } static struct formatter_units size_units; void tr_formatter_size_init( unsigned int kilo, const char * kb, const char * mb, const char * gb, const char * tb ) { formatter_init( &size_units, kilo, kb, mb, gb, tb ); } char* tr_formatter_size_B( char * buf, int64_t bytes, size_t buflen ) { return formatter_get_size_str( &size_units, buf, bytes, buflen ); } static struct formatter_units speed_units; unsigned int tr_speed_K = 0u; void tr_formatter_speed_init( unsigned int kilo, const char * kb, const char * mb, const char * gb, const char * tb ) { tr_speed_K = kilo; formatter_init( &speed_units, kilo, kb, mb, gb, tb ); } char* tr_formatter_speed_KBps( char * buf, double KBps, size_t buflen ) { const double K = speed_units.units[TR_FMT_KB].value; double speed = KBps; if( speed <= 999.95 ) /* 0.0 KB to 999.9 KB */ tr_snprintf( buf, buflen, "%d %s", (int)speed, speed_units.units[TR_FMT_KB].name ); else { speed /= K; if( speed <= 99.995 ) /* 0.98 MB to 99.99 MB */ tr_snprintf( buf, buflen, "%.2f %s", speed, speed_units.units[TR_FMT_MB].name ); else if (speed <= 999.95) /* 100.0 MB to 999.9 MB */ tr_snprintf( buf, buflen, "%.1f %s", speed, speed_units.units[TR_FMT_MB].name ); else { speed /= K; tr_snprintf( buf, buflen, "%.1f %s", speed, speed_units.units[TR_FMT_GB].name ); } } return buf; } static struct formatter_units mem_units; unsigned int tr_mem_K = 0u; void tr_formatter_mem_init( unsigned int kilo, const char * kb, const char * mb, const char * gb, const char * tb ) { tr_mem_K = kilo; formatter_init( &mem_units, kilo, kb, mb, gb, tb ); } char* tr_formatter_mem_B( char * buf, int64_t bytes_per_second, size_t buflen ) { return formatter_get_size_str( &mem_units, buf, bytes_per_second, buflen ); } void tr_formatter_get_units( tr_benc * d ) { int i; tr_benc * l; tr_bencDictReserve( d, 6 ); tr_bencDictAddInt( d, "memory-bytes", mem_units.units[TR_FMT_KB].value ); l = tr_bencDictAddList( d, "memory-units", 4 ); for( i=0; i<4; i++ ) tr_bencListAddStr( l, mem_units.units[i].name ); tr_bencDictAddInt( d, "size-bytes", size_units.units[TR_FMT_KB].value ); l = tr_bencDictAddList( d, "size-units", 4 ); for( i=0; i<4; i++ ) tr_bencListAddStr( l, size_units.units[i].name ); tr_bencDictAddInt( d, "speed-bytes", speed_units.units[TR_FMT_KB].value ); l = tr_bencDictAddList( d, "speed-units", 4 ); for( i=0; i<4; i++ ) tr_bencListAddStr( l, speed_units.units[i].name ); }