/* * This file Copyright (C) 2008-2010 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$ */ #include /* isprint() */ #include #include /* free() */ #include #ifdef WIN32 #include #endif #ifndef WIN32 #include #endif #include #include #include #include #include #include "ggets.h" #include "transmission.h" #include "platform.h" #include "blocklist.h" #include "net.h" #include "utils.h" /*** **** PRIVATE ***/ struct tr_ip_range { uint32_t begin; uint32_t end; }; struct tr_blocklist { tr_bool isEnabled; int fd; size_t ruleCount; size_t byteCount; char * filename; struct tr_ip_range * rules; }; static void blocklistClose( tr_blocklist * b ) { if( b->rules ) { munmap( b->rules, b->byteCount ); close( b->fd ); b->rules = NULL; b->ruleCount = 0; b->byteCount = 0; b->fd = -1; } } static void blocklistLoad( tr_blocklist * b ) { int fd; struct stat st; const char * err_fmt = _( "Couldn't read \"%1$s\": %2$s" ); blocklistClose( b ); if( stat( b->filename, &st ) == -1 ) return; fd = open( b->filename, O_RDONLY ); if( fd == -1 ) { tr_err( err_fmt, b->filename, tr_strerror( errno ) ); return; } #ifndef WIN32 b->rules = mmap( NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0 ); #else b->rules = mmap( NULL, st.st_size, 0, 0, fd, 0 ); #endif if( !b->rules ) { tr_err( err_fmt, b->filename, tr_strerror( errno ) ); close( fd ); return; } b->byteCount = st.st_size; b->ruleCount = st.st_size / sizeof( struct tr_ip_range ); b->fd = fd; { char * base = tr_basename( b->filename ); tr_inf( _( "Blocklist \"%s\" contains %'zu entries" ), base, b->ruleCount ); tr_free( base ); } } static void blocklistEnsureLoaded( tr_blocklist * b ) { if( !b->rules ) blocklistLoad( b ); } static int compareAddressToRange( const void * va, const void * vb ) { const uint32_t * a = va; const struct tr_ip_range * b = vb; if( *a < b->begin ) return -1; if( *a > b->end ) return 1; return 0; } static void blocklistDelete( tr_blocklist * b ) { blocklistClose( b ); unlink( b->filename ); } /*** **** PACKAGE-VISIBLE ***/ tr_blocklist * _tr_blocklistNew( const char * filename, int isEnabled ) { tr_blocklist * b; b = tr_new0( tr_blocklist, 1 ); b->fd = -1; b->filename = tr_strdup( filename ); b->isEnabled = isEnabled; return b; } const char* _tr_blocklistGetFilename( const tr_blocklist * b ) { return b->filename; } void _tr_blocklistFree( tr_blocklist * b ) { blocklistClose( b ); tr_free( b->filename ); tr_free( b ); } int _tr_blocklistExists( const tr_blocklist * b ) { struct stat st; return !stat( b->filename, &st ); } int _tr_blocklistGetRuleCount( const tr_blocklist * b ) { blocklistEnsureLoaded( (tr_blocklist*)b ); return b->ruleCount; } int _tr_blocklistIsEnabled( tr_blocklist * b ) { return b->isEnabled; } void _tr_blocklistSetEnabled( tr_blocklist * b, int isEnabled ) { b->isEnabled = isEnabled ? 1 : 0; } int _tr_blocklistHasAddress( tr_blocklist * b, const tr_address * addr ) { uint32_t needle; const struct tr_ip_range * range; assert( tr_isAddress( addr ) ); if( !b->isEnabled || addr->type == TR_AF_INET6 ) return 0; blocklistEnsureLoaded( b ); if( !b->rules || !b->ruleCount ) return 0; needle = ntohl( addr->addr.addr4.s_addr ); range = bsearch( &needle, b->rules, b->ruleCount, sizeof( struct tr_ip_range ), compareAddressToRange ); return range != NULL; } int _tr_blocklistSetContent( tr_blocklist * b, const char * filename ) { FILE * in; FILE * out; char * line; int inCount = 0; int outCount = 0; const char * err_fmt = _( "Couldn't read \"%1$s\": %2$s" ); if( !filename ) { blocklistDelete( b ); return 0; } in = fopen( filename, "r" ); if( !in ) { tr_err( err_fmt, filename, tr_strerror( errno ) ); return 0; } blocklistClose( b ); out = fopen( b->filename, "wb+" ); if( !out ) { tr_err( err_fmt, b->filename, tr_strerror( errno ) ); fclose( in ); return 0; } while( !fggets( &line, in ) ) { char * rangeBegin; char * rangeEnd; char * crpos; tr_address addr; struct tr_ip_range range; ++inCount; rangeBegin = strrchr( line, ':' ); if( !rangeBegin ){ free( line ); continue; } ++rangeBegin; rangeEnd = strchr( rangeBegin, '-' ); if( !rangeEnd ){ free( line ); continue; } *rangeEnd++ = '\0'; if(( crpos = strchr( rangeEnd, '\r' ))) *crpos = '\0'; if( !tr_pton( rangeBegin, &addr ) ) { printf( "error rangeBegin: %s", line ); tr_err( _( "blocklist skipped invalid address at line %d" ), inCount ); free( line ); continue; } range.begin = ntohl( addr.addr.addr4.s_addr ); if( !tr_pton( rangeEnd, &addr ) ) { printf( "error rangeEnd: %s", line ); tr_err( _( "blocklist skipped invalid address at line %d" ), inCount ); free( line ); continue; } range.end = ntohl( addr.addr.addr4.s_addr ); free( line ); if( fwrite( &range, sizeof( struct tr_ip_range ), 1, out ) != 1 ) { tr_err( _( "Couldn't save file \"%1$s\": %2$s" ), b->filename, tr_strerror( errno ) ); break; } ++outCount; } { char * base = tr_basename( b->filename ); tr_inf( _( "Blocklist \"%1$s\" updated with %2$'d entries" ), base, outCount ); tr_free( base ); } fclose( out ); fclose( in ); blocklistLoad( b ); return outCount; }