2016-04-28 10:23:33 +00:00
/* ******************************************************************
2016-05-09 14:19:25 +00:00
util . h - utility functions
Copyright ( C ) 2016 , Przemyslaw Skibinski , Yann Collet .
GPL v2 License
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License along
with this program ; if not , write to the Free Software Foundation , Inc . ,
51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA .
You can contact the author at :
- ZSTD homepage : http : //www.zstd.net/
*/
2016-04-28 10:23:33 +00:00
# ifndef UTIL_H_MODULE
# define UTIL_H_MODULE
# if defined (__cplusplus)
extern " C " {
# endif
2016-05-05 09:53:42 +00:00
/* **************************************
* Compiler Options
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-08-18 08:30:21 +00:00
# if defined(__INTEL_COMPILER)
# pragma warning(disable : 177) /* disable: message #177: function was declared but never referenced */
# endif
2016-05-05 09:53:42 +00:00
# if defined(_MSC_VER)
# define _CRT_SECURE_NO_WARNINGS /* Disable some Visual warning messages for fopen, strncpy */
# define _CRT_SECURE_NO_DEPRECATE /* VS2005 */
# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */
2016-05-09 14:19:25 +00:00
# if _MSC_VER <= 1800 /* (1800 = Visual Studio 2013) */
# define snprintf sprintf_s /* snprintf unsupported by Visual <= 2013 */
2016-05-05 09:53:42 +00:00
# endif
2016-05-09 12:37:41 +00:00
# endif
2016-05-05 09:53:42 +00:00
/* Unix Large Files support (>4GB) */
# if !defined(__LP64__) /* No point defining Large file for 64 bit */
# define _FILE_OFFSET_BITS 64 /* turn off_t into a 64-bit type for ftello, fseeko */
# if defined(__sun__) /* Sun Solaris 32-bits requires specific definitions */
# define _LARGEFILE_SOURCE /* fseeko, ftello */
2016-05-20 09:24:35 +00:00
# else
2016-05-05 09:53:42 +00:00
# define _LARGEFILE64_SOURCE /* off64_t, fseeko64, ftello64 */
# endif
# endif
2016-04-28 10:23:33 +00:00
/*-****************************************
* Dependencies
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-05-11 12:11:00 +00:00
# include <stdlib.h> /* features.h with _POSIX_C_SOURCE, malloc */
2016-05-04 22:25:38 +00:00
# include <stdio.h> /* fprintf */
2016-04-28 12:40:45 +00:00
# include <sys/types.h> /* stat */
# include <sys/stat.h> /* stat */
# include "mem.h" /* U32, U64 */
2016-04-28 10:23:33 +00:00
2016-05-11 12:11:00 +00:00
/* *************************************
* Constants
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define LIST_SIZE_INCREASE (8*1024)
2016-04-28 10:23:33 +00:00
/*-****************************************
* Compiler specifics
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if defined(__GNUC__)
# define UTIL_STATIC static __attribute__((unused))
# elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */ )
# define UTIL_STATIC static inline
# elif defined(_MSC_VER)
# define UTIL_STATIC static __inline
# else
# define UTIL_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */
# endif
2016-05-04 22:25:38 +00:00
2016-05-05 09:53:42 +00:00
/*-****************************************
* Sleep functions : Windows - Posix - others
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-05-04 22:25:38 +00:00
# if defined(_WIN32)
2016-04-28 11:16:01 +00:00
# include <windows.h>
2016-05-04 22:25:38 +00:00
# define SET_HIGH_PRIORITY SetPriorityClass(GetCurrentProcess(), REALTIME_PRIORITY_CLASS)
2016-04-28 11:16:01 +00:00
# define UTIL_sleep(s) Sleep(1000*s)
# define UTIL_sleepMilli(milli) Sleep(milli)
2016-05-21 11:15:48 +00:00
# elif (defined(__unix__) || defined(__unix) || defined(__VMS) || defined(__midipix__) || (defined(__APPLE__) && defined(__MACH__)))
2016-05-04 22:25:38 +00:00
# include <unistd.h>
# include <sys / resource.h> /* setpriority */
# include <time.h> /* clock_t, nanosleep, clock, CLOCKS_PER_SEC */
2016-05-21 11:15:48 +00:00
# if defined(PRIO_PROCESS)
# define SET_HIGH_PRIORITY setpriority(PRIO_PROCESS, 0, -20)
# else
# define SET_HIGH_PRIORITY /* disabled */
# endif
2016-05-04 22:25:38 +00:00
# define UTIL_sleep(s) sleep(s)
2016-05-05 09:53:42 +00:00
# if defined(_POSIX_C_SOURCE) && (_POSIX_C_SOURCE >= 199309L)
# define UTIL_sleepMilli(milli) { struct timespec t; t.tv_sec=0; t.tv_nsec=milli*1000000ULL; nanosleep(&t, NULL); }
# else
# define UTIL_sleepMilli(milli) /* disabled */
# endif
2016-04-28 11:16:01 +00:00
# else
2016-05-04 22:25:38 +00:00
# define SET_HIGH_PRIORITY /* disabled */
# define UTIL_sleep(s) /* disabled */
2016-04-28 11:16:01 +00:00
# define UTIL_sleepMilli(milli) /* disabled */
# endif
2016-05-04 22:25:38 +00:00
/*-****************************************
* Time functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-04-28 11:16:01 +00:00
# if !defined(_WIN32)
typedef clock_t UTIL_time_t ;
2016-05-09 14:19:25 +00:00
UTIL_STATIC void UTIL_initTimer ( UTIL_time_t * ticksPerSecond ) { * ticksPerSecond = 0 ; }
UTIL_STATIC void UTIL_getTime ( UTIL_time_t * x ) { * x = clock ( ) ; }
UTIL_STATIC U64 UTIL_getSpanTimeMicro ( UTIL_time_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { ( void ) ticksPerSecond ; return 1000000ULL * ( clockEnd - clockStart ) / CLOCKS_PER_SEC ; }
2016-05-09 16:04:07 +00:00
UTIL_STATIC U64 UTIL_getSpanTimeNano ( UTIL_time_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { ( void ) ticksPerSecond ; return 1000000000ULL * ( clockEnd - clockStart ) / CLOCKS_PER_SEC ; }
2016-04-28 11:16:01 +00:00
# else
typedef LARGE_INTEGER UTIL_time_t ;
2016-05-09 14:19:25 +00:00
UTIL_STATIC void UTIL_initTimer ( UTIL_time_t * ticksPerSecond ) { if ( ! QueryPerformanceFrequency ( ticksPerSecond ) ) fprintf ( stderr , " ERROR: QueryPerformance not present \n " ) ; }
UTIL_STATIC void UTIL_getTime ( UTIL_time_t * x ) { QueryPerformanceCounter ( x ) ; }
UTIL_STATIC U64 UTIL_getSpanTimeMicro ( UTIL_time_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return 1000000ULL * ( clockEnd . QuadPart - clockStart . QuadPart ) / ticksPerSecond . QuadPart ; }
UTIL_STATIC U64 UTIL_getSpanTimeNano ( UTIL_time_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return 1000000000ULL * ( clockEnd . QuadPart - clockStart . QuadPart ) / ticksPerSecond . QuadPart ; }
2016-04-28 11:16:01 +00:00
# endif
/* returns time span in microseconds */
UTIL_STATIC U64 UTIL_clockSpanMicro ( UTIL_time_t clockStart , UTIL_time_t ticksPerSecond )
{
UTIL_time_t clockEnd ;
2016-05-09 14:19:25 +00:00
UTIL_getTime ( & clockEnd ) ;
2016-04-28 11:16:01 +00:00
return UTIL_getSpanTimeMicro ( ticksPerSecond , clockStart , clockEnd ) ;
}
UTIL_STATIC void UTIL_waitForNextTick ( UTIL_time_t ticksPerSecond )
{
UTIL_time_t clockStart , clockEnd ;
2016-05-09 14:19:25 +00:00
UTIL_getTime ( & clockStart ) ;
2016-05-20 09:24:35 +00:00
do {
UTIL_getTime ( & clockEnd ) ;
2016-04-28 12:40:45 +00:00
} while ( UTIL_getSpanTimeNano ( ticksPerSecond , clockStart , clockEnd ) = = 0 ) ;
2016-04-28 11:16:01 +00:00
}
2016-05-04 22:25:38 +00:00
/*-****************************************
* File functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-04-28 10:23:33 +00:00
UTIL_STATIC U64 UTIL_getFileSize ( const char * infilename )
{
int r ;
# if defined(_MSC_VER)
struct _stat64 statbuf ;
r = _stat64 ( infilename , & statbuf ) ;
if ( r | | ! ( statbuf . st_mode & S_IFREG ) ) return 0 ; /* No good... */
# else
struct stat statbuf ;
r = stat ( infilename , & statbuf ) ;
if ( r | | ! S_ISREG ( statbuf . st_mode ) ) return 0 ; /* No good... */
# endif
return ( U64 ) statbuf . st_size ;
}
2016-04-28 14:50:13 +00:00
UTIL_STATIC U64 UTIL_getTotalFileSize ( const char * * fileNamesTable , unsigned nbFiles )
{
U64 total = 0 ;
unsigned n ;
for ( n = 0 ; n < nbFiles ; n + + )
total + = UTIL_getFileSize ( fileNamesTable [ n ] ) ;
return total ;
}
2016-05-04 22:25:38 +00:00
UTIL_STATIC int UTIL_doesFileExists ( const char * infilename )
{
int r ;
# if defined(_MSC_VER)
struct _stat64 statbuf ;
r = _stat64 ( infilename , & statbuf ) ;
if ( r | | ! ( statbuf . st_mode & S_IFREG ) ) return 0 ; /* No good... */
# else
struct stat statbuf ;
r = stat ( infilename , & statbuf ) ;
if ( r | | ! S_ISREG ( statbuf . st_mode ) ) return 0 ; /* No good... */
# endif
return 1 ;
}
2016-04-28 10:23:33 +00:00
UTIL_STATIC U32 UTIL_isDirectory ( const char * infilename )
{
int r ;
# if defined(_MSC_VER)
struct _stat64 statbuf ;
r = _stat64 ( infilename , & statbuf ) ;
if ( ! r & & ( statbuf . st_mode & _S_IFDIR ) ) return 1 ;
# else
struct stat statbuf ;
r = stat ( infilename , & statbuf ) ;
if ( ! r & & S_ISDIR ( statbuf . st_mode ) ) return 1 ;
# endif
return 0 ;
}
2016-05-04 22:25:38 +00:00
# ifdef _WIN32
2016-05-05 09:53:42 +00:00
# define UTIL_HAS_CREATEFILELIST
2016-05-04 22:25:38 +00:00
2016-05-11 12:11:00 +00:00
UTIL_STATIC int UTIL_prepareFileList ( const char * dirName , char * * bufStart , size_t * pos , char * * bufEnd )
2016-05-04 22:25:38 +00:00
{
char path [ MAX_PATH ] ;
int pathLength , nbFiles = 0 ;
WIN32_FIND_DATA cFile ;
HANDLE hFile ;
pathLength = snprintf ( path , MAX_PATH , " %s \\ * " , dirName ) ;
if ( pathLength < 0 | | pathLength > = MAX_PATH ) {
fprintf ( stderr , " Path length has got too long. \n " ) ;
return 0 ;
}
hFile = FindFirstFile ( path , & cFile ) ;
if ( hFile = = INVALID_HANDLE_VALUE ) {
fprintf ( stderr , " Cannot open directory '%s' \n " , dirName ) ;
return 0 ;
}
2016-05-11 12:11:00 +00:00
do {
pathLength = snprintf ( path , MAX_PATH , " %s \\ %s " , dirName , cFile . cFileName ) ;
if ( pathLength < 0 | | pathLength > = MAX_PATH ) {
fprintf ( stderr , " Path length has got too long. \n " ) ;
continue ;
}
2016-05-04 22:25:38 +00:00
if ( cFile . dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY ) {
if ( strcmp ( cFile . cFileName , " .. " ) = = 0 | |
2016-05-11 12:11:00 +00:00
strcmp ( cFile . cFileName , " . " ) = = 0 ) continue ;
2016-08-24 15:32:09 +00:00
2016-05-11 12:11:00 +00:00
nbFiles + = UTIL_prepareFileList ( path , bufStart , pos , bufEnd ) ; /* Recursively call "UTIL_prepareFileList" with the new path. */
if ( * bufStart = = NULL ) { FindClose ( hFile ) ; return 0 ; }
2016-05-04 22:25:38 +00:00
}
else if ( ( cFile . dwFileAttributes & FILE_ATTRIBUTE_NORMAL ) | | ( cFile . dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE ) | | ( cFile . dwFileAttributes & FILE_ATTRIBUTE_COMPRESSED ) ) {
2016-05-11 12:11:00 +00:00
if ( * bufStart + * pos + pathLength > = * bufEnd ) {
ptrdiff_t newListSize = ( * bufEnd - * bufStart ) + LIST_SIZE_INCREASE ;
* bufStart = ( char * ) realloc ( * bufStart , newListSize ) ;
* bufEnd = * bufStart + newListSize ;
if ( * bufStart = = NULL ) { FindClose ( hFile ) ; return 0 ; }
}
if ( * bufStart + * pos + pathLength < * bufEnd ) {
strncpy ( * bufStart + * pos , path , * bufEnd - ( * bufStart + * pos ) ) ;
* pos + = pathLength + 1 ;
nbFiles + + ;
}
2016-05-04 22:25:38 +00:00
}
2016-05-11 12:11:00 +00:00
} while ( FindNextFile ( hFile , & cFile ) ) ;
2016-05-04 22:25:38 +00:00
FindClose ( hFile ) ;
return nbFiles ;
}
2016-05-10 12:22:55 +00:00
# elif (defined(__unix__) || defined(__unix) || defined(__midipix__) || (defined(__APPLE__) && defined(__MACH__))) && defined(_POSIX_C_SOURCE) && (_POSIX_C_SOURCE >= 200112L) /* snprintf, opendir */
2016-05-05 09:53:42 +00:00
# define UTIL_HAS_CREATEFILELIST
2016-05-04 22:25:38 +00:00
# include <dirent.h> /* opendir, readdir */
# include <limits.h> /* PATH_MAX */
# include <errno.h>
2016-05-11 12:11:00 +00:00
UTIL_STATIC int UTIL_prepareFileList ( const char * dirName , char * * bufStart , size_t * pos , char * * bufEnd )
2016-05-04 22:25:38 +00:00
{
DIR * dir ;
struct dirent * entry ;
char path [ PATH_MAX ] ;
int pathLength , nbFiles = 0 ;
if ( ! ( dir = opendir ( dirName ) ) ) {
fprintf ( stderr , " Cannot open directory '%s': %s \n " , dirName , strerror ( errno ) ) ;
return 0 ;
}
2016-05-20 09:24:35 +00:00
2016-07-26 09:07:37 +00:00
errno = 0 ;
2016-05-12 15:15:41 +00:00
while ( ( entry = readdir ( dir ) ) ! = NULL ) {
2016-05-05 11:10:57 +00:00
if ( strcmp ( entry - > d_name , " .. " ) = = 0 | |
strcmp ( entry - > d_name , " . " ) = = 0 ) continue ;
pathLength = snprintf ( path , PATH_MAX , " %s/%s " , dirName , entry - > d_name ) ;
if ( pathLength < 0 | | pathLength > = PATH_MAX ) {
fprintf ( stderr , " Path length has got too long. \n " ) ;
continue ;
}
if ( UTIL_isDirectory ( path ) ) {
2016-05-11 12:11:00 +00:00
nbFiles + = UTIL_prepareFileList ( path , bufStart , pos , bufEnd ) ; /* Recursively call "UTIL_prepareFileList" with the new path. */
if ( * bufStart = = NULL ) { closedir ( dir ) ; return 0 ; }
2016-05-04 22:25:38 +00:00
} else {
2016-05-11 12:11:00 +00:00
if ( * bufStart + * pos + pathLength > = * bufEnd ) {
ptrdiff_t newListSize = ( * bufEnd - * bufStart ) + LIST_SIZE_INCREASE ;
* bufStart = ( char * ) realloc ( * bufStart , newListSize ) ;
* bufEnd = * bufStart + newListSize ;
if ( * bufStart = = NULL ) { closedir ( dir ) ; return 0 ; }
}
if ( * bufStart + * pos + pathLength < * bufEnd ) {
strncpy ( * bufStart + * pos , path , * bufEnd - ( * bufStart + * pos ) ) ;
* pos + = pathLength + 1 ;
nbFiles + + ;
}
2016-05-04 22:25:38 +00:00
}
2016-08-24 15:32:09 +00:00
errno = 0 ; /* clear errno after UTIL_isDirectory, UTIL_prepareFileList */
2016-05-04 22:25:38 +00:00
}
2016-07-26 09:07:37 +00:00
if ( errno ! = 0 ) {
fprintf ( stderr , " readdir(%s) error: %s \n " , dirName , strerror ( errno ) ) ;
free ( * bufStart ) ;
* bufStart = NULL ;
}
2016-05-04 22:25:38 +00:00
closedir ( dir ) ;
return nbFiles ;
}
# else
2016-05-11 12:11:00 +00:00
UTIL_STATIC int UTIL_prepareFileList ( const char * dirName , char * * bufStart , size_t * pos , char * * bufEnd )
2016-05-04 22:25:38 +00:00
{
2016-05-11 12:11:00 +00:00
( void ) bufStart ; ( void ) bufEnd ; ( void ) pos ;
fprintf ( stderr , " Directory %s ignored (zstd compiled without _WIN32 or _POSIX_C_SOURCE) \n " , dirName ) ;
2016-05-04 22:25:38 +00:00
return 0 ;
}
2016-08-24 15:32:09 +00:00
# endif /* #ifdef _WIN32 */
2016-05-04 22:25:38 +00:00
2016-05-20 09:24:35 +00:00
/*
* UTIL_createFileList - takes a list of files and directories ( params : inputNames , inputNamesNb ) , scans directories ,
2016-05-13 08:52:02 +00:00
* and returns a new list of files ( params : return value , allocatedBuffer , allocatedNamesNb ) .
* After finishing usage of the list the structures should be freed with UTIL_freeFileList ( params : return value , allocatedBuffer )
* In case of error UTIL_createFileList returns NULL and UTIL_freeFileList should not be called .
*/
UTIL_STATIC const char * * UTIL_createFileList ( const char * * inputNames , unsigned inputNamesNb , char * * allocatedBuffer , unsigned * allocatedNamesNb )
2016-05-04 22:25:38 +00:00
{
2016-05-11 12:11:00 +00:00
size_t pos ;
2016-05-13 08:52:02 +00:00
unsigned i , nbFiles ;
2016-05-11 12:11:00 +00:00
char * bufend , * buf ;
2016-05-13 08:52:02 +00:00
const char * * fileTable ;
2016-05-04 22:25:38 +00:00
2016-05-11 12:11:00 +00:00
buf = ( char * ) malloc ( LIST_SIZE_INCREASE ) ;
2016-05-13 08:52:02 +00:00
if ( ! buf ) return NULL ;
2016-05-11 12:11:00 +00:00
bufend = buf + LIST_SIZE_INCREASE ;
2016-05-05 09:53:42 +00:00
2016-05-13 08:52:02 +00:00
for ( i = 0 , pos = 0 , nbFiles = 0 ; i < inputNamesNb ; i + + ) {
2016-05-25 13:30:55 +00:00
if ( ! UTIL_isDirectory ( inputNames [ i ] ) ) {
2016-05-05 09:53:42 +00:00
size_t len = strlen ( inputNames [ i ] ) ;
2016-05-11 12:11:00 +00:00
if ( buf + pos + len > = bufend ) {
ptrdiff_t newListSize = ( bufend - buf ) + LIST_SIZE_INCREASE ;
buf = ( char * ) realloc ( buf , newListSize ) ;
bufend = buf + newListSize ;
2016-05-13 08:52:02 +00:00
if ( ! buf ) return NULL ;
2016-05-11 12:11:00 +00:00
}
if ( buf + pos + len < bufend ) {
strncpy ( buf + pos , inputNames [ i ] , bufend - ( buf + pos ) ) ;
pos + = len + 1 ;
nbFiles + + ;
}
2016-08-01 12:26:49 +00:00
} else {
2016-05-11 12:11:00 +00:00
nbFiles + = UTIL_prepareFileList ( inputNames [ i ] , & buf , & pos , & bufend ) ;
2016-05-13 08:52:02 +00:00
if ( buf = = NULL ) return NULL ;
2016-08-01 12:26:49 +00:00
} }
2016-05-04 22:25:38 +00:00
2016-05-13 08:52:02 +00:00
if ( nbFiles = = 0 ) { free ( buf ) ; return NULL ; }
2016-05-04 22:25:38 +00:00
2016-05-13 08:52:02 +00:00
fileTable = ( const char * * ) malloc ( ( nbFiles + 1 ) * sizeof ( const char * ) ) ;
if ( ! fileTable ) { free ( buf ) ; return NULL ; }
2016-05-04 22:25:38 +00:00
2016-05-13 08:52:02 +00:00
for ( i = 0 , pos = 0 ; i < nbFiles ; i + + )
{
fileTable [ i ] = buf + pos ;
pos + = strlen ( fileTable [ i ] ) + 1 ;
2016-05-04 22:25:38 +00:00
}
2016-05-13 08:52:02 +00:00
if ( buf + pos > bufend ) { free ( buf ) ; free ( ( void * ) fileTable ) ; return NULL ; }
* allocatedBuffer = buf ;
* allocatedNamesNb = nbFiles ;
return fileTable ;
2016-05-04 22:25:38 +00:00
}
2016-05-13 08:52:02 +00:00
UTIL_STATIC void UTIL_freeFileList ( const char * * filenameTable , char * allocatedBuffer )
2016-05-04 22:25:38 +00:00
{
2016-05-13 08:52:02 +00:00
if ( allocatedBuffer ) free ( allocatedBuffer ) ;
if ( filenameTable ) free ( ( void * ) filenameTable ) ;
2016-05-04 22:25:38 +00:00
}
2016-04-28 10:23:33 +00:00
# if defined (__cplusplus)
}
# endif
# endif /* UTIL_H_MODULE */