2016-12-21 14:08:44 +00:00
/**
* util . h - utility functions
*
* Copyright ( c ) 2016 - present , Przemyslaw Skibinski , Yann Collet , Facebook , Inc .
* All rights reserved .
*
* This source code is licensed under the BSD - style license found in the
* LICENSE file in the root directory of this source tree . An additional grant
* of patent rights can be found in the PATENTS file in the same directory .
*/
2016-05-09 14:19:25 +00:00
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
2016-12-21 12:23:34 +00:00
2016-04-28 10:23:33 +00:00
/*-****************************************
* Dependencies
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-12-21 12:23:34 +00:00
# include "platform.h" /* PLATFORM_POSIX_VERSION */
# include <stdlib.h> /* malloc */
# include <stddef.h> /* size_t, ptrdiff_t */
# include <stdio.h> /* fprintf */
# include <sys/types.h> /* stat, utime */
# include <sys/stat.h> /* stat */
2016-11-03 08:54:53 +00:00
# if defined(_MSC_VER)
2016-12-21 12:23:34 +00:00
# include <sys / utime.h> /* utime */
# include <io.h> /* _chmod */
2016-11-03 08:54:53 +00:00
# else
2016-12-12 00:03:23 +00:00
# include <unistd.h> /* chown, stat */
# include <utime.h> /* utime */
2016-11-03 08:54:53 +00:00
# endif
2016-12-21 12:23:34 +00:00
# include <time.h> /* time */
2016-11-03 08:54:53 +00:00
# include <errno.h>
2016-12-21 12:23:34 +00:00
# include "mem.h" /* U32, U64 */
2016-04-28 10:23:33 +00:00
2016-05-04 22:25:38 +00:00
2017-02-15 16:03:16 +00:00
/* ************************************************************
* Avoid fseek ( ) ' s 2 GiB barrier with MSVC , MacOS , * BSD , MinGW
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if defined(_MSC_VER) && (_MSC_VER >= 1400)
# define UTIL_fseek _fseeki64
# elif !defined(__64BIT__) && (PLATFORM_POSIX_VERSION >= 200112L) /* No point defining Large file for 64 bit */
# define UTIL_fseek fseeko
# elif defined(__MINGW32__) && defined(__MSVCRT__) && !defined(__STRICT_ANSI__) && !defined(__NO_MINGW_LFS)
# define UTIL_fseek fseeko64
# else
# define UTIL_fseek fseek
# endif
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>
2017-02-07 15:36:19 +00:00
# define SET_REALTIME_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-12-16 14:41:18 +00:00
# elif PLATFORM_POSIX_VERSION >= 0 /* Unix-like operating system */
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)
2017-02-07 15:36:19 +00:00
# define SET_REALTIME_PRIORITY setpriority(PRIO_PROCESS, 0, -20)
2016-05-21 11:15:48 +00:00
# else
2017-02-07 15:36:19 +00:00
# define SET_REALTIME_PRIORITY /* disabled */
2016-05-21 11:15:48 +00:00
# endif
2016-05-04 22:25:38 +00:00
# define UTIL_sleep(s) sleep(s)
2016-12-16 14:41:18 +00:00
# if (defined(__linux__) && (PLATFORM_POSIX_VERSION >= 199309L)) || (PLATFORM_POSIX_VERSION >= 200112L) /* nanosleep requires POSIX.1-2001 */
2016-05-05 09:53:42 +00:00
# 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
2017-02-07 15:36:19 +00:00
# define SET_REALTIME_PRIORITY /* disabled */
2016-05-04 22:25:38 +00:00
# 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
2016-12-21 08:04:59 +00:00
/* *************************************
* Constants
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define LIST_SIZE_INCREASE (8*1024)
/*-****************************************
* Compiler specifics
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-12-21 12:23:34 +00:00
# if defined(__INTEL_COMPILER)
# pragma warning(disable : 177) /* disable: message #177: function was declared but never referenced, useful with UTIL_STATIC */
# endif
2016-12-21 08:04:59 +00:00
# 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
/*-****************************************
* Time functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2017-02-20 10:11:50 +00:00
# if defined(_WIN32) /* Windows */
2017-02-20 10:27:11 +00:00
typedef LARGE_INTEGER UTIL_freq_t ;
2016-04-28 11:16:01 +00:00
typedef LARGE_INTEGER UTIL_time_t ;
2017-02-20 10:27:11 +00:00
UTIL_STATIC void UTIL_initTimer ( UTIL_freq_t * ticksPerSecond ) { if ( ! QueryPerformanceFrequency ( ticksPerSecond ) ) fprintf ( stderr , " ERROR: QueryPerformance not present \n " ) ; }
2016-05-09 14:19:25 +00:00
UTIL_STATIC void UTIL_getTime ( UTIL_time_t * x ) { QueryPerformanceCounter ( x ) ; }
2017-02-20 10:27:11 +00:00
UTIL_STATIC U64 UTIL_getSpanTimeMicro ( UTIL_freq_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return 1000000ULL * ( clockEnd . QuadPart - clockStart . QuadPart ) / ticksPerSecond . QuadPart ; }
UTIL_STATIC U64 UTIL_getSpanTimeNano ( UTIL_freq_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return 1000000000ULL * ( clockEnd . QuadPart - clockStart . QuadPart ) / ticksPerSecond . QuadPart ; }
2017-02-20 11:18:15 +00:00
# elif defined(__APPLE__) && defined(__MACH__)
# include <mach/mach_time.h>
typedef mach_timebase_info_data_t UTIL_freq_t ;
typedef U64 UTIL_time_t ;
2017-02-21 11:22:05 +00:00
UTIL_STATIC void UTIL_initTimer ( UTIL_freq_t * rate ) { mach_timebase_info ( rate ) ; }
2017-02-20 11:18:15 +00:00
UTIL_STATIC void UTIL_getTime ( UTIL_time_t * x ) { * x = mach_absolute_time ( ) ; }
UTIL_STATIC U64 UTIL_getSpanTimeMicro ( UTIL_freq_t rate , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return ( ( ( clockEnd - clockStart ) * ( U64 ) rate . numer ) / ( ( U64 ) rate . denom ) ) / 1000ULL ; }
UTIL_STATIC U64 UTIL_getSpanTimeNano ( UTIL_freq_t rate , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return ( ( clockEnd - clockStart ) * ( U64 ) rate . numer ) / ( ( U64 ) rate . denom ) ; }
2017-02-21 06:33:45 +00:00
# elif (PLATFORM_POSIX_VERSION >= 200112L)
# include <sys/times.h> /* times */
typedef U64 UTIL_freq_t ;
typedef U64 UTIL_time_t ;
UTIL_STATIC void UTIL_initTimer ( UTIL_freq_t * ticksPerSecond ) { * ticksPerSecond = sysconf ( _SC_CLK_TCK ) ; }
UTIL_STATIC void UTIL_getTime ( UTIL_time_t * x ) { struct tms junk ; clock_t newTicks = ( clock_t ) times ( & junk ) ; ( void ) junk ; * x = ( UTIL_time_t ) newTicks ; }
UTIL_STATIC U64 UTIL_getSpanTimeMicro ( UTIL_freq_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return 1000000ULL * ( clockEnd - clockStart ) / ticksPerSecond ; }
UTIL_STATIC U64 UTIL_getSpanTimeNano ( UTIL_freq_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { return 1000000000ULL * ( clockEnd - clockStart ) / ticksPerSecond ; }
2017-01-20 20:23:30 +00:00
# else /* relies on standard C (note : clock_t measurements can be wrong when using multi-threading) */
2017-02-20 10:27:11 +00:00
typedef clock_t UTIL_freq_t ;
2017-01-20 20:23:30 +00:00
typedef clock_t UTIL_time_t ;
2017-02-20 10:27:11 +00:00
UTIL_STATIC void UTIL_initTimer ( UTIL_freq_t * ticksPerSecond ) { * ticksPerSecond = 0 ; }
2017-01-20 20:23:30 +00:00
UTIL_STATIC void UTIL_getTime ( UTIL_time_t * x ) { * x = clock ( ) ; }
2017-02-20 10:27:11 +00:00
UTIL_STATIC U64 UTIL_getSpanTimeMicro ( UTIL_freq_t ticksPerSecond , UTIL_time_t clockStart , UTIL_time_t clockEnd ) { ( void ) ticksPerSecond ; return 1000000ULL * ( clockEnd - clockStart ) / CLOCKS_PER_SEC ; }
UTIL_STATIC U64 UTIL_getSpanTimeNano ( UTIL_freq_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
# endif
/* returns time span in microseconds */
2017-02-20 10:27:11 +00:00
UTIL_STATIC U64 UTIL_clockSpanMicro ( UTIL_time_t clockStart , UTIL_freq_t ticksPerSecond )
2016-04-28 11:16:01 +00:00
{
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 ) ;
}
2017-02-20 10:27:11 +00:00
UTIL_STATIC void UTIL_waitForNextTick ( UTIL_freq_t ticksPerSecond )
2016-04-28 11:16:01 +00:00
{
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-11-02 13:08:07 +00:00
# if defined(_MSC_VER)
2016-11-03 08:54:53 +00:00
# define chmod _chmod
2017-02-10 12:27:43 +00:00
typedef struct __stat64 stat_t ;
2016-11-02 13:08:07 +00:00
# else
typedef struct stat stat_t ;
# endif
UTIL_STATIC int UTIL_setFileStat ( const char * filename , stat_t * statbuf )
2016-11-02 11:52:20 +00:00
{
2016-11-02 13:08:07 +00:00
int res = 0 ;
struct utimbuf timebuf ;
2016-11-03 08:54:53 +00:00
timebuf . actime = time ( NULL ) ;
timebuf . modtime = statbuf - > st_mtime ;
res + = utime ( filename , & timebuf ) ; /* set access and modification times */
2016-11-02 13:08:07 +00:00
# if !defined(_WIN32)
res + = chown ( filename , statbuf - > st_uid , statbuf - > st_gid ) ; /* Copy ownership */
# endif
2016-11-02 11:52:20 +00:00
2016-11-02 13:08:07 +00:00
res + = chmod ( filename , statbuf - > st_mode & 07777 ) ; /* Copy file permissions */
2016-11-02 11:52:20 +00:00
2016-11-02 13:08:07 +00:00
errno = 0 ;
return - res ; /* number of errors is returned */
2016-11-02 11:52:20 +00:00
}
2016-11-02 13:08:07 +00:00
UTIL_STATIC int UTIL_getFileStat ( const char * infilename , stat_t * statbuf )
2016-11-02 11:52:20 +00:00
{
int r ;
# if defined(_MSC_VER)
2016-11-02 13:08:07 +00:00
r = _stat64 ( infilename , statbuf ) ;
if ( r | | ! ( statbuf - > st_mode & S_IFREG ) ) return 0 ; /* No good... */
2016-11-02 11:52:20 +00:00
# else
2016-11-02 13:08:07 +00:00
r = stat ( infilename , statbuf ) ;
if ( r | | ! S_ISREG ( statbuf - > st_mode ) ) return 0 ; /* No good... */
2016-11-02 11:52:20 +00:00
# endif
2016-11-02 13:08:07 +00:00
return 1 ;
2016-11-02 11:52:20 +00:00
}
2017-02-14 08:38:51 +00:00
2017-02-08 00:33:48 +00:00
UTIL_STATIC int UTIL_isRegFile ( const char * infilename )
{
stat_t statbuf ;
return UTIL_getFileStat ( infilename , & statbuf ) ; /* Only need to know whether it is a regular file */
}
2016-11-02 11:52:20 +00:00
2017-02-14 08:38:51 +00:00
UTIL_STATIC U32 UTIL_isDirectory ( const char * infilename )
{
int r ;
stat_t statbuf ;
# if defined(_MSC_VER)
r = _stat64 ( infilename , & statbuf ) ;
if ( ! r & & ( statbuf . st_mode & _S_IFDIR ) ) return 1 ;
# else
r = stat ( infilename , & statbuf ) ;
if ( ! r & & S_ISDIR ( statbuf . st_mode ) ) return 1 ;
# endif
return 0 ;
}
2016-04-28 10:23:33 +00:00
UTIL_STATIC U64 UTIL_getFileSize ( const char * infilename )
{
int r ;
# if defined(_MSC_VER)
2017-02-13 12:57:29 +00:00
struct __stat64 statbuf ;
2016-04-28 10:23:33 +00:00
r = _stat64 ( infilename , & statbuf ) ;
if ( r | | ! ( statbuf . st_mode & S_IFREG ) ) return 0 ; /* No good... */
2017-02-10 17:37:57 +00:00
# elif defined(__MINGW32__) && defined (__MSVCRT__)
struct _stati64 statbuf ;
r = _stati64 ( infilename , & statbuf ) ;
if ( r | | ! ( statbuf . st_mode & S_IFREG ) ) return 0 ; /* No good... */
2016-04-28 10:23:33 +00:00
# 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-09-15 16:58:18 +00:00
/*
* A modified version of realloc ( ) .
* If UTIL_realloc ( ) fails the original block is freed .
*/
UTIL_STATIC void * UTIL_realloc ( void * ptr , size_t size )
{
void * newptr = realloc ( ptr , size ) ;
if ( newptr ) return newptr ;
free ( ptr ) ;
return NULL ;
}
2016-04-28 10:23:33 +00:00
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
{
2016-09-13 11:13:10 +00:00
char * path ;
int dirLength , fnameLength , pathLength , nbFiles = 0 ;
2017-02-15 16:13:35 +00:00
WIN32_FIND_DATAA cFile ;
2016-05-04 22:25:38 +00:00
HANDLE hFile ;
2016-09-13 14:38:54 +00:00
dirLength = ( int ) strlen ( dirName ) ;
2016-09-13 11:13:10 +00:00
path = ( char * ) malloc ( dirLength + 3 ) ;
if ( ! path ) return 0 ;
memcpy ( path , dirName , dirLength ) ;
path [ dirLength ] = ' \\ ' ;
path [ dirLength + 1 ] = ' * ' ;
path [ dirLength + 2 ] = 0 ;
2016-05-04 22:25:38 +00:00
2017-02-15 16:13:35 +00:00
hFile = FindFirstFileA ( path , & cFile ) ;
2016-05-04 22:25:38 +00:00
if ( hFile = = INVALID_HANDLE_VALUE ) {
fprintf ( stderr , " Cannot open directory '%s' \n " , dirName ) ;
return 0 ;
}
2016-09-13 11:13:10 +00:00
free ( path ) ;
2016-05-04 22:25:38 +00:00
2016-05-11 12:11:00 +00:00
do {
2016-09-13 14:38:54 +00:00
fnameLength = ( int ) strlen ( cFile . cFileName ) ;
2016-09-13 11:13:10 +00:00
path = ( char * ) malloc ( dirLength + fnameLength + 2 ) ;
if ( ! path ) { FindClose ( hFile ) ; return 0 ; }
memcpy ( path , dirName , dirLength ) ;
path [ dirLength ] = ' \\ ' ;
memcpy ( path + dirLength + 1 , cFile . cFileName , fnameLength ) ;
pathLength = dirLength + 1 + fnameLength ;
path [ pathLength ] = 0 ;
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. */
2016-09-13 11:13:10 +00:00
if ( * bufStart = = NULL ) { free ( path ) ; 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 ;
2016-09-15 16:58:18 +00:00
* bufStart = ( char * ) UTIL_realloc ( * bufStart , newListSize ) ;
2016-05-11 12:11:00 +00:00
* bufEnd = * bufStart + newListSize ;
2016-09-13 11:13:10 +00:00
if ( * bufStart = = NULL ) { free ( path ) ; FindClose ( hFile ) ; return 0 ; }
2016-05-11 12:11:00 +00:00
}
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-09-13 11:13:10 +00:00
free ( path ) ;
2017-02-15 16:13:35 +00:00
} while ( FindNextFileA ( hFile , & cFile ) ) ;
2016-05-04 22:25:38 +00:00
FindClose ( hFile ) ;
return nbFiles ;
}
2016-12-16 16:12:23 +00:00
# elif defined(__linux__) || (PLATFORM_POSIX_VERSION >= 200112L) /* opendir, readdir require POSIX.1-2001 */
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 */
2016-12-21 08:04:59 +00:00
# include <string.h> /* strerror, memcpy */
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
{
DIR * dir ;
struct dirent * entry ;
2016-09-13 11:13:10 +00:00
char * path ;
int dirLength , fnameLength , pathLength , nbFiles = 0 ;
2016-05-04 22:25:38 +00:00
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-09-13 14:38:54 +00:00
dirLength = ( int ) strlen ( dirName ) ;
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 ;
2016-09-13 14:38:54 +00:00
fnameLength = ( int ) strlen ( entry - > d_name ) ;
2016-09-13 11:13:10 +00:00
path = ( char * ) malloc ( dirLength + fnameLength + 2 ) ;
if ( ! path ) { closedir ( dir ) ; return 0 ; }
memcpy ( path , dirName , dirLength ) ;
path [ dirLength ] = ' / ' ;
memcpy ( path + dirLength + 1 , entry - > d_name , fnameLength ) ;
pathLength = dirLength + 1 + fnameLength ;
path [ pathLength ] = 0 ;
2016-05-05 11:10:57 +00:00
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. */
2016-09-13 11:13:10 +00:00
if ( * bufStart = = NULL ) { free ( path ) ; 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 ;
2016-09-15 16:58:18 +00:00
* bufStart = ( char * ) UTIL_realloc ( * bufStart , newListSize ) ;
2016-05-11 12:11:00 +00:00
* bufEnd = * bufStart + newListSize ;
2016-09-13 11:13:10 +00:00
if ( * bufStart = = NULL ) { free ( path ) ; closedir ( dir ) ; return 0 ; }
2016-05-11 12:11:00 +00:00
}
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-09-13 11:13:10 +00:00
free ( path ) ;
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 ;
2016-12-21 08:04:59 +00:00
fprintf ( stderr , " Directory %s ignored (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-12-12 00:03:23 +00:00
char * buf = ( char * ) malloc ( LIST_SIZE_INCREASE ) ;
char * bufend = buf + LIST_SIZE_INCREASE ;
2016-05-13 08:52:02 +00:00
const char * * fileTable ;
2016-05-04 22:25:38 +00:00
2016-05-13 08:52:02 +00:00
if ( ! buf ) return NULL ;
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-12-12 00:03:23 +00:00
size_t const 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 ;
2016-09-15 16:58:18 +00:00
buf = ( char * ) UTIL_realloc ( buf , newListSize ) ;
2016-05-11 12:11:00 +00:00
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-12-12 00:03:23 +00:00
for ( i = 0 , pos = 0 ; i < nbFiles ; i + + ) {
2016-05-13 08:52:02 +00:00
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 */