2007-11-14 22:18:48 -05:00
/*
* " $Id: rsdir.cc,v 1.1 2007-02-19 20:08:30 rmf24 Exp $ "
*
* RetroShare C + + Interface .
*
* Copyright 2004 - 2007 by Robert Fernie .
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Library General Public
* License Version 2 as published by the Free Software Foundation .
*
* This library 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
* Library General Public License for more details .
*
* You should have received a copy of the GNU Library General Public
* License along with this library ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307
* USA .
*
* Please report all bugs and problems to " retroshare@lunamutt.com " .
*
*/
// Includes for directory creation.
# include <sys/types.h>
# include <sys/stat.h>
2015-10-20 12:41:40 -04:00
# include <fcntl.h>
2007-11-14 22:18:48 -05:00
# include <unistd.h>
# include "util/rsdir.h"
2012-04-15 10:37:44 -04:00
# include "util/rsstring.h"
2013-12-27 15:06:47 -05:00
# include "util/rsrandom.h"
2016-01-12 21:10:11 -05:00
# include "util/rsmemory.h"
2016-10-09 15:43:00 -04:00
# include "util/folderiterator.h"
2010-08-06 05:40:23 -04:00
# include "retroshare/rstypes.h"
2016-11-22 17:39:09 -05:00
# include "retroshare/rsnotify.h"
2011-03-03 18:30:08 -05:00
# include "rsthreads.h"
2007-11-14 22:18:48 -05:00
# include <iostream>
2008-05-30 06:36:47 -04:00
# include <algorithm>
2009-05-11 10:30:53 -04:00
# include <stdio.h>
2010-02-09 09:11:03 -05:00
# include <dirent.h>
2010-07-06 01:04:11 -04:00
# include <openssl/sha.h>
# include <iomanip>
2015-03-25 04:19:45 -04:00
# include <sstream>
2010-02-09 09:11:03 -05:00
2010-03-10 09:08:52 -05:00
# include <fstream>
2012-03-15 15:55:43 -04:00
# include <stdexcept>
2007-11-14 22:18:48 -05:00
2009-03-29 09:58:28 -04:00
# if defined(WIN32) || defined(__CYGWIN__)
2011-10-01 09:12:28 -04:00
# include "util/rsstring.h"
2009-03-29 09:58:28 -04:00
# include "wtypes.h"
# include <winioctl.h>
# else
# include <errno.h>
# endif
2008-07-09 05:55:09 -04:00
/****
* # define RSDIR_DEBUG 1
* * * */
2010-10-01 16:05:09 -04:00
std : : string RsDirUtil : : getTopDir ( const std : : string & dir )
2007-11-14 22:18:48 -05:00
{
std : : string top ;
/* find the subdir: [/][dir1.../]<top>[/]
*/
int i , j ;
int len = dir . length ( ) ;
2011-06-29 14:02:44 -04:00
for ( j = len - 1 ; ( j > 0 ) & & ( dir [ j ] = = ' / ' ) ; j - - ) ;
for ( i = j ; ( i > 0 ) & & ( dir [ i ] ! = ' / ' ) ; i - - ) ;
2007-11-14 22:18:48 -05:00
if ( dir [ i ] = = ' / ' )
i + + ;
for ( ; i < = j ; i + + )
{
top + = dir [ i ] ;
}
return top ;
}
2014-01-04 15:05:08 -05:00
const char * RsDirUtil : : scanf_string_for_uint ( int bytes )
{
const char * strgs [ 3 ] = { " %u " , " %lu " , " %llu " } ;
2014-01-24 12:34:27 -05:00
//std::cerr << "RsDirUtil::scanf_string_for_uint(): returning for bytes=" << bytes << std::endl;
2014-01-04 15:05:08 -05:00
if ( sizeof ( unsigned int ) = = bytes )
return strgs [ 0 ] ;
if ( sizeof ( long unsigned int ) = = bytes )
return strgs [ 1 ] ;
if ( sizeof ( long long unsigned int ) = = bytes )
return strgs [ 2 ] ;
std : : cerr < < " RsDirUtil::scanf_string_for_uint(): no corresponding scan string for " < < bytes < < " bytes. This will probably cause inconsistencies. " < < std : : endl ;
return strgs [ 0 ] ;
}
2012-06-01 16:51:14 -04:00
void RsDirUtil : : removeTopDir ( const std : : string & dir , std : : string & path )
2007-11-14 22:18:48 -05:00
{
2012-06-01 16:51:14 -04:00
path . clear ( ) ;
2007-11-14 22:18:48 -05:00
/* remove the subdir: [/][dir1.../]<top>[/]
*/
2012-06-01 16:51:14 -04:00
int j = dir . find_last_not_of ( ' / ' ) ;
int i = dir . rfind ( ' / ' , j ) ;
2007-11-14 22:18:48 -05:00
/* remove any more slashes */
2012-06-01 16:51:14 -04:00
if ( i > 0 )
2007-11-14 22:18:48 -05:00
{
2012-06-01 16:51:14 -04:00
i = dir . find_last_not_of ( ' / ' , i ) ;
2007-11-14 22:18:48 -05:00
}
2012-06-01 16:51:14 -04:00
if ( i > 0 )
{
path . assign ( dir , 0 , i + 1 ) ;
}
2007-11-14 22:18:48 -05:00
}
2010-10-01 16:05:09 -04:00
std : : string RsDirUtil : : getRootDir ( const std : : string & dir )
2007-11-14 22:18:48 -05:00
{
std : : string root ;
/* find the subdir: [/]root[/...]
*/
int i , j ;
int len = dir . length ( ) ;
2011-06-29 14:02:44 -04:00
for ( i = 0 ; ( i < len ) & & ( dir [ i ] = = ' / ' ) ; i + + ) ;
for ( j = i ; ( j < len ) & & ( dir [ j ] ! = ' / ' ) ; j + + ) ;
2007-11-14 22:18:48 -05:00
if ( i = = j )
return root ; /* empty */
for ( ; i < j ; i + + )
{
root + = dir [ i ] ;
}
return root ;
}
2010-10-01 16:05:09 -04:00
std : : string RsDirUtil : : removeRootDir ( const std : : string & path )
2007-11-14 22:18:48 -05:00
{
unsigned int i , j ;
unsigned int len = path . length ( ) ;
std : : string output ;
/* chew leading '/'s */
2011-06-29 14:02:44 -04:00
for ( i = 0 ; ( i < len ) & & ( path [ i ] = = ' / ' ) ; i + + ) ;
2007-11-14 22:18:48 -05:00
if ( i = = len )
return output ; /* empty string */
2011-06-29 14:02:44 -04:00
for ( j = i ; ( j < len ) & & ( path [ j ] ! = ' / ' ) ; j + + ) ; /* run to next '/' */
for ( ; ( j < len ) & & ( path [ j ] = = ' / ' ) ; j + + ) ; /* chew leading '/'s */
2007-11-14 22:18:48 -05:00
for ( ; j < len ; j + + )
{
output + = path [ j ] ;
}
return output ;
}
2010-10-01 16:05:09 -04:00
std : : string RsDirUtil : : removeRootDirs ( const std : : string & path , const std : : string & root )
2007-11-14 22:18:48 -05:00
{
/* too tired */
std : : string notroot ;
unsigned int i = 0 , j = 0 ;
/* catch empty data */
if ( ( root . length ( ) < 1 ) | | ( path . length ( ) < 1 ) )
return notroot ;
if ( ( path [ 0 ] = = ' / ' ) & & ( root [ 0 ] ! = ' / ' ) )
{
i + + ;
}
2011-06-29 14:02:44 -04:00
for ( ; ( i < path . length ( ) ) & & ( j < root . length ( ) ) & & ( path [ i ] = = root [ j ] ) ; i + + , j + + ) ;
2007-11-14 22:18:48 -05:00
/* should have consumed root. */
if ( j = = root . length ( ) )
{
//std::cerr << "matched root!" << std::endl;
}
else
{
//std::cerr << "failed i: " << i << ", j: " << j << std::endl;
//std::cerr << "root: " << root << " path: " << path << std::endl;
return notroot ;
}
if ( path [ i ] = = ' / ' )
{
i + + ;
}
for ( ; i < path . length ( ) ; i + + )
{
notroot + = path [ i ] ;
}
//std::cerr << "Found NotRoot: " << notroot << std::endl;
return notroot ;
}
2010-10-01 16:05:09 -04:00
int RsDirUtil : : breakupDirList ( const std : : string & path ,
2007-11-14 22:18:48 -05:00
std : : list < std : : string > & subdirs )
{
int start = 0 ;
unsigned int i ;
for ( i = 0 ; i < path . length ( ) ; i + + )
{
if ( path [ i ] = = ' / ' )
{
if ( i - start > 0 )
{
subdirs . push_back ( path . substr ( start , i - start ) ) ;
}
start = i + 1 ;
}
}
// get the final one.
if ( i - start > 0 )
{
subdirs . push_back ( path . substr ( start , i - start ) ) ;
}
return 1 ;
}
2013-07-25 16:05:31 -04:00
/**** Copied and Tweaked from ftcontroller ***/
bool RsDirUtil : : fileExists ( const std : : string & filename )
{
return ( access ( filename . c_str ( ) , F_OK ) ! = - 1 ) ;
}
2007-11-14 22:18:48 -05:00
2016-11-22 17:39:09 -05:00
bool RsDirUtil : : moveFile ( const std : : string & source , const std : : string & dest )
{
// First try a rename
//
if ( renameFile ( source , dest ) )
return true ;
// If not, try to copy. The src and dest probably belong to different file systems
if ( ! copyFile ( source , dest ) )
return false ;
// copy was successful, let's delete the original
if ( ! removeFile ( source ) )
return false ;
return true ;
}
bool RsDirUtil : : removeFile ( const std : : string & filename )
{
# ifdef CONTROL_DEBUG
std : : cerr < < " deleting original file " < < source < < std : : endl ;
# endif
# ifdef WINDOWS_SYS
std : : wstring filenameW ;
librs : : util : : ConvertUtf8ToUtf16 ( filename , filenameW ) ;
if ( 0 ! = DeleteFileW ( filenameW . c_str ( ) ) )
# else
if ( 0 = = remove ( filename . c_str ( ) ) )
# endif
return true ;
else
{
std : : cerr < < " (EE) File erase error while removing file " < < filename < < " . Read-only file system ? " < < std : : endl ;
return false ;
}
}
2010-10-28 18:59:46 -04:00
/**** Copied and Tweaked from ftcontroller ***/
bool RsDirUtil : : copyFile ( const std : : string & source , const std : : string & dest )
{
# ifdef WINDOWS_SYS
std : : wstring sourceW ;
std : : wstring destW ;
librs : : util : : ConvertUtf8ToUtf16 ( source , sourceW ) ;
librs : : util : : ConvertUtf8ToUtf16 ( dest , destW ) ;
return ( CopyFileW ( sourceW . c_str ( ) , destW . c_str ( ) , FALSE ) ! = 0 ) ;
# else
2011-02-24 17:44:41 -05:00
FILE * in = fopen64 ( source . c_str ( ) , " rb " ) ;
2010-10-28 18:59:46 -04:00
if ( in = = NULL )
{
return false ;
}
2011-02-24 17:44:41 -05:00
FILE * out = fopen64 ( dest . c_str ( ) , " wb " ) ;
2010-10-28 18:59:46 -04:00
if ( out = = NULL )
{
fclose ( in ) ;
return false ;
}
size_t s = 0 ;
size_t T = 0 ;
static const int BUFF_SIZE = 10485760 ; // 10 MB buffer to speed things up.
2016-01-12 21:10:11 -05:00
RsTemporaryMemory buffer ( BUFF_SIZE ) ;
if ( ! buffer )
{
fclose ( in ) ;
fclose ( out ) ;
return false ;
}
2010-10-28 18:59:46 -04:00
bool bRet = true ;
while ( ( s = fread ( buffer , 1 , BUFF_SIZE , in ) ) > 0 )
{
size_t t = fwrite ( buffer , 1 , s , out ) ;
T + = t ;
if ( t ! = s )
{
bRet = false ;
break ;
}
}
fclose ( in ) ;
fclose ( out ) ;
2016-06-02 08:42:32 -04:00
return bRet ;
2010-10-28 18:59:46 -04:00
# endif
}
2014-03-29 10:18:05 -04:00
bool RsDirUtil : : checkFile ( const std : : string & filename , uint64_t & file_size , bool disallow_empty_file )
2010-10-28 18:59:46 -04:00
{
int val ;
mode_t st_mode ;
# ifdef WINDOWS_SYS
std : : wstring wfilename ;
librs : : util : : ConvertUtf8ToUtf16 ( filename , wfilename ) ;
struct _stat buf ;
val = _wstat ( wfilename . c_str ( ) , & buf ) ;
st_mode = buf . st_mode ;
# else
2012-08-06 13:04:43 -04:00
struct stat64 buf ;
val = stat64 ( filename . c_str ( ) , & buf ) ;
2010-10-28 18:59:46 -04:00
st_mode = buf . st_mode ;
# endif
if ( val = = - 1 )
{
# ifdef RSDIR_DEBUG
std : : cerr < < " RsDirUtil::checkFile() " ;
std : : cerr < < filename < < " doesn't exist " < < std : : endl ;
# endif
return false ;
}
else if ( ! S_ISREG ( st_mode ) )
{
// Some other type - error.
# ifdef RSDIR_DEBUG
std : : cerr < < " RsDirUtil::checkFile() " ;
std : : cerr < < filename < < " is not a Regular File " < < std : : endl ;
# endif
return false ;
}
2012-08-06 13:04:43 -04:00
2014-03-29 10:18:05 -04:00
file_size = buf . st_size ;
2012-08-06 13:04:43 -04:00
if ( disallow_empty_file & & buf . st_size = = 0 )
return false ;
2010-10-28 18:59:46 -04:00
return true ;
}
2007-11-14 22:18:48 -05:00
2010-10-01 16:05:09 -04:00
bool RsDirUtil : : checkDirectory ( const std : : string & dir )
2008-07-23 18:01:59 -04:00
{
2010-09-17 14:27:30 -04:00
int val ;
mode_t st_mode ;
# ifdef WINDOWS_SYS
std : : wstring wdir ;
librs : : util : : ConvertUtf8ToUtf16 ( dir , wdir ) ;
struct _stat buf ;
val = _wstat ( wdir . c_str ( ) , & buf ) ;
st_mode = buf . st_mode ;
# else
2008-07-23 18:01:59 -04:00
struct stat buf ;
2010-09-17 14:27:30 -04:00
val = stat ( dir . c_str ( ) , & buf ) ;
st_mode = buf . st_mode ;
# endif
2008-07-23 18:01:59 -04:00
if ( val = = - 1 )
{
# ifdef RSDIR_DEBUG
std : : cerr < < " RsDirUtil::checkDirectory() " ;
std : : cerr < < dir < < " doesn't exist " < < std : : endl ;
# endif
return false ;
}
2010-09-17 14:27:30 -04:00
else if ( ! S_ISDIR ( st_mode ) )
2008-07-23 18:01:59 -04:00
{
// Some other type - error.
# ifdef RSDIR_DEBUG
std : : cerr < < " RsDirUtil::checkDirectory() " ;
std : : cerr < < dir < < " is not Directory " < < std : : endl ;
# endif
return false ;
}
return true ;
}
2010-10-01 16:05:09 -04:00
bool RsDirUtil : : checkCreateDirectory ( const std : : string & dir )
2007-11-14 22:18:48 -05:00
{
2010-02-09 09:11:03 -05:00
# ifdef RSDIR_DEBUG
2011-03-06 15:19:16 -05:00
std : : cerr < < " RsDirUtil::checkCreateDirectory() dir: " < < dir < < std : : endl ;
2010-02-09 09:11:03 -05:00
# endif
2011-03-06 15:19:16 -05:00
# ifdef WINDOWS_SYS
std : : wstring wdir ;
librs : : util : : ConvertUtf8ToUtf16 ( dir , wdir ) ;
_WDIR * direc = _wopendir ( wdir . c_str ( ) ) ;
if ( ! direc )
# else
DIR * direc = opendir ( dir . c_str ( ) ) ;
if ( ! direc )
# endif
{
// directory don't exist. create.
2007-11-14 22:18:48 -05:00
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
# ifndef WINDOWS_SYS // UNIX
2011-03-06 15:19:16 -05:00
if ( - 1 = = mkdir ( dir . c_str ( ) , 0777 ) )
2007-11-14 22:18:48 -05:00
# else // WIN
2011-03-06 15:19:16 -05:00
if ( - 1 = = _wmkdir ( wdir . c_str ( ) ) )
2007-11-14 22:18:48 -05:00
# endif
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
2011-03-06 15:19:16 -05:00
{
2010-02-09 09:11:03 -05:00
# ifdef RSDIR_DEBUG
2011-03-06 15:19:16 -05:00
std : : cerr < < " check_create_directory() Fatal Error et oui-- " ;
std : : cerr < < std : : endl < < " \t cannot create: " < < dir < < std : : endl ;
2008-07-09 05:55:09 -04:00
# endif
2011-03-06 15:19:16 -05:00
return 0 ;
}
2007-11-14 22:18:48 -05:00
2010-02-09 09:11:03 -05:00
# ifdef RSDIR_DEBUG
2011-03-06 15:19:16 -05:00
std : : cerr < < " check_create_directory() " ;
std : : cerr < < std : : endl < < " \t created: " < < dir < < std : : endl ;
2008-07-09 05:55:09 -04:00
# endif
2011-03-04 22:43:40 -05:00
return 1 ;
2011-03-06 15:19:16 -05:00
}
2010-02-09 09:11:03 -05:00
# ifdef RSDIR_DEBUG
2011-03-06 15:19:16 -05:00
std : : cerr < < " check_create_directory() " ;
std : : cerr < < std : : endl < < " \t Dir Exists: " < < dir < < std : : endl ;
2008-07-09 05:55:09 -04:00
# endif
2007-11-14 22:18:48 -05:00
2011-03-06 15:19:16 -05:00
# ifdef WINDOWS_SYS
_wclosedir ( direc ) ;
# else
closedir ( direc ) ;
# endif
return 1 ;
}
2007-11-14 22:18:48 -05:00
2013-09-24 11:31:15 -04:00
2016-11-24 17:42:56 -05:00
std : : string RsDirUtil : : removeSymLinks ( const std : : string & path )
{
2016-12-08 09:56:23 -05:00
# if defined(WINDOWS_SYS) || defined(__APPLE__) || defined(__ANDROID__)
# warning (Mr.Alice): I don't know how to do this on windows / MacOS / Android. See https: //msdn.microsoft.com/en-us/library/windows/desktop/hh707084(v=vs.85).aspx'
2016-11-24 17:42:56 -05:00
//if(!S_OK == PathCchCanonicalizeEx(tmp,...) ;
return path ;
# else
char * tmp = canonicalize_file_name ( path . c_str ( ) ) ;
std : : string result ( tmp ) ;
free ( tmp ) ;
return result ;
# endif
}
2013-09-24 11:31:15 -04:00
bool RsDirUtil : : cleanupDirectory ( const std : : string & cleandir , const std : : set < std : : string > & keepFiles )
2007-11-14 22:18:48 -05:00
{
2016-11-16 15:41:32 -05:00
for ( librs : : util : : FolderIterator it ( cleandir , false ) ; it . isValid ( ) ; it . next ( ) )
2016-10-09 15:43:00 -04:00
if ( it . file_type ( ) = = librs : : util : : FolderIterator : : TYPE_FILE & & ( keepFiles . end ( ) = = std : : find ( keepFiles . begin ( ) , keepFiles . end ( ) , it . file_name ( ) ) ) )
remove ( ( cleandir + " / " + it . file_name ( ) ) . c_str ( ) ) ;
2007-11-14 22:18:48 -05:00
2016-10-09 15:43:00 -04:00
return true ;
2007-11-14 22:18:48 -05:00
}
2008-03-17 09:51:04 -04:00
2008-07-23 18:01:59 -04:00
/* slightly nicer helper function */
2010-10-01 16:05:09 -04:00
bool RsDirUtil : : hashFile ( const std : : string & filepath ,
2014-03-17 16:56:06 -04:00
std : : string & name , RsFileHash & hash , uint64_t & size )
2008-07-23 18:01:59 -04:00
{
if ( getFileHash ( filepath , hash , size ) )
{
/* extract file name */
name = RsDirUtil : : getTopDir ( filepath ) ;
return true ;
}
return false ;
}
2008-03-17 09:51:04 -04:00
# include <openssl/sha.h>
# include <iomanip>
/* Function to hash, and get details of a file */
2014-03-17 16:56:06 -04:00
bool RsDirUtil : : getFileHash ( const std : : string & filepath , RsFileHash & hash , uint64_t & size , RsThread * thread /*= NULL*/ )
2008-03-17 09:51:04 -04:00
{
FILE * fd ;
2015-06-14 17:09:23 -04:00
if ( NULL = = ( fd = RsDirUtil : : rs_fopen ( filepath . c_str ( ) , " rb " ) ) )
return false ;
2008-03-17 09:51:04 -04:00
int len ;
SHA_CTX * sha_ctx = new SHA_CTX ;
unsigned char sha_buf [ SHA_DIGEST_LENGTH ] ;
unsigned char gblBuf [ 512 ] ;
/* determine size */
2011-02-24 17:19:04 -05:00
fseeko64 ( fd , 0 , SEEK_END ) ;
size = ftello64 ( fd ) ;
fseeko64 ( fd , 0 , SEEK_SET ) ;
2008-03-17 09:51:04 -04:00
2011-03-03 18:30:08 -05:00
/* check if thread is running */
bool isRunning = thread ? thread - > isRunning ( ) : true ;
int runningCheckCount = 0 ;
2008-03-17 09:51:04 -04:00
SHA1_Init ( sha_ctx ) ;
2011-03-03 18:30:08 -05:00
while ( isRunning & & ( len = fread ( gblBuf , 1 , 512 , fd ) ) > 0 )
2008-03-17 09:51:04 -04:00
{
SHA1_Update ( sha_ctx , gblBuf , len ) ;
2011-03-03 18:30:08 -05:00
if ( thread & & + + runningCheckCount > ( 10 * 1024 ) ) {
/* check all 50MB if thread is running */
isRunning = thread - > isRunning ( ) ;
runningCheckCount = 0 ;
}
}
/* Thread has ended */
if ( isRunning = = false )
{
delete sha_ctx ;
fclose ( fd ) ;
return false ;
2008-03-17 09:51:04 -04:00
}
/* reading failed for some reason */
2011-03-03 18:30:08 -05:00
if ( ferror ( fd ) )
2008-03-17 09:51:04 -04:00
{
delete sha_ctx ;
fclose ( fd ) ;
return false ;
}
SHA1_Final ( & sha_buf [ 0 ] , sha_ctx ) ;
2014-04-20 12:34:26 -04:00
hash = Sha1CheckSum ( sha_buf ) ;
2008-03-17 09:51:04 -04:00
delete sha_ctx ;
fclose ( fd ) ;
return true ;
}
2012-03-15 15:55:43 -04:00
/* Function to hash, and get details of a file */
2014-03-17 16:56:06 -04:00
Sha1CheckSum RsDirUtil : : sha1sum ( const unsigned char * data , uint32_t size )
2012-03-15 15:55:43 -04:00
{
SHA_CTX sha_ctx ;
if ( SHA_DIGEST_LENGTH ! = 20 )
throw std : : runtime_error ( " Warning: can't compute Sha1Sum with sum size != 20 " ) ;
SHA1_Init ( & sha_ctx ) ;
while ( size > 512 )
{
SHA1_Update ( & sha_ctx , data , 512 ) ;
data = & data [ 512 ] ;
size - = 512 ;
}
SHA1_Update ( & sha_ctx , data , size ) ;
unsigned char sha_buf [ SHA_DIGEST_LENGTH ] ;
SHA1_Final ( & sha_buf [ 0 ] , & sha_ctx ) ;
return Sha1CheckSum ( sha_buf ) ;
}
2015-03-25 04:19:45 -04:00
bool RsDirUtil : : saveStringToFile ( const std : : string & file , const std : : string & str )
{
std : : ofstream out ( file . c_str ( ) , std : : ios_base : : out | std : : ios_base : : binary ) ;
if ( ! out . is_open ( ) )
{
std : : cerr < < " RsDirUtil::saveStringToFile() ERROR: can't open file " < < file < < std : : endl ;
return false ;
}
out < < str ;
2015-03-26 03:21:07 -04:00
return true ;
2015-03-25 04:19:45 -04:00
}
bool RsDirUtil : : loadStringFromFile ( const std : : string & file , std : : string & str )
{
std : : ifstream in ( file . c_str ( ) , std : : ios_base : : in | std : : ios_base : : binary ) ;
if ( ! in . is_open ( ) )
{
std : : cerr < < " RsDirUtil::loadStringFromFile() ERROR: can't open file " < < file < < std : : endl ;
return false ;
}
std : : stringstream buffer ;
buffer < < in . rdbuf ( ) ;
str = buffer . str ( ) ;
return true ;
}
2009-03-29 09:58:28 -04:00
bool RsDirUtil : : renameFile ( const std : : string & from , const std : : string & to )
{
2011-04-03 19:11:38 -04:00
int loops = 0 ;
2009-03-29 09:58:28 -04:00
2011-04-03 19:11:38 -04:00
# ifdef WINDOWS_SYS
std : : wstring f ;
librs : : util : : ConvertUtf8ToUtf16 ( from , f ) ;
std : : wstring t ;
librs : : util : : ConvertUtf8ToUtf16 ( to , t ) ;
2014-09-20 15:54:04 -04:00
while ( ! MoveFileEx ( f . c_str ( ) , t . c_str ( ) , MOVEFILE_REPLACE_EXISTING | MOVEFILE_WRITE_THROUGH ) )
2009-09-03 16:51:12 -04:00
# else
2009-03-29 09:58:28 -04:00
std : : string f ( from ) , t ( to ) ;
2011-04-03 19:11:38 -04:00
2009-03-29 09:58:28 -04:00
while ( rename ( from . c_str ( ) , to . c_str ( ) ) < 0 )
# endif
{
# ifdef WIN32
if ( GetLastError ( ) ! = ERROR_ACCESS_DENIED )
# else
if ( errno ! = EACCES )
# endif
/* set errno? */
return false ;
2014-10-24 17:31:58 -04:00
usleep ( 100 * 1000 ) ; // 100 msec
2009-03-29 09:58:28 -04:00
if ( loops > = 30 )
return false ;
loops + + ;
}
return true ;
}
2011-08-14 18:31:05 -04:00
# ifdef UNUSED_CODE
// not used
2010-10-01 16:05:09 -04:00
bool RsDirUtil : : createBackup ( const std : : string & sFilename , unsigned int nCount )
2010-06-04 19:39:33 -04:00
{
# ifdef WINDOWS_SYS
2010-06-26 20:18:33 -04:00
if ( GetFileAttributesA ( sFilename . c_str ( ) ) = = - 1 ) {
2010-06-04 19:39:33 -04:00
// file doesn't exist
return true ;
}
// search last backup
int nLast ;
for ( nLast = nCount ; nLast > = 1 ; nLast - - ) {
2012-04-15 10:37:44 -04:00
std : : ostringstream sStream ; // please do not use std::ostringstream
2010-06-04 19:39:33 -04:00
sStream < < sFilename < < nLast < < " .bak " ;
2010-06-26 20:18:33 -04:00
if ( GetFileAttributesA ( sStream . str ( ) . c_str ( ) ) ! = - 1 ) {
2010-06-04 19:39:33 -04:00
break ;
}
}
// delete max backup
if ( nLast = = nCount ) {
2012-04-15 10:37:44 -04:00
std : : ostringstream sStream ; // please do not use std::ostringstream
2010-06-04 19:39:33 -04:00
sStream < < sFilename < < nCount < < " .bak " ;
2010-06-26 20:18:33 -04:00
if ( DeleteFileA ( sStream . str ( ) . c_str ( ) ) = = FALSE ) {
2010-06-04 19:39:33 -04:00
getPqiNotify ( ) - > AddSysMessage ( 0 , RS_SYS_WARNING , " File delete error " , " Error while deleting file " + sStream . str ( ) ) ;
return false ;
}
nLast - - ;
}
// rename backups
for ( int nIndex = nLast ; nIndex > = 1 ; nIndex - - ) {
2012-04-15 10:37:44 -04:00
std : : ostringstream sStream ; // please do not use std::ostringstream
2010-06-04 19:39:33 -04:00
sStream < < sFilename < < nIndex < < " .bak " ;
2012-04-15 10:37:44 -04:00
std : : ostringstream sStream1 ; // please do not use std::ostringstream
2010-06-04 19:39:33 -04:00
sStream1 < < sFilename < < nIndex + 1 < < " .bak " ;
if ( renameFile ( sStream . str ( ) , sStream1 . str ( ) ) = = false ) {
getPqiNotify ( ) - > AddSysMessage ( 0 , RS_SYS_WARNING , " File rename error " , " Error while renaming file " + sStream . str ( ) + " to " + sStream1 . str ( ) ) ;
return false ;
}
}
// copy backup
2012-04-15 10:37:44 -04:00
std : : ostringstream sStream ; // please do not use std::ostringstream
2010-06-04 19:39:33 -04:00
sStream < < sFilename < < 1 < < " .bak " ;
2010-06-26 20:18:33 -04:00
if ( CopyFileA ( sFilename . c_str ( ) , sStream . str ( ) . c_str ( ) , FALSE ) = = FALSE ) {
2010-06-04 19:39:33 -04:00
getPqiNotify ( ) - > AddSysMessage ( 0 , RS_SYS_WARNING , " File rename error " , " Error while renaming file " + sFilename + " to " + sStream . str ( ) ) ;
return false ;
}
2011-08-12 09:42:30 -04:00
# else
/* remove unused parameter warnings */
( void ) sFilename ;
( void ) nCount ;
2010-06-04 19:39:33 -04:00
# endif
return true ;
}
2011-08-14 18:31:05 -04:00
# endif
2010-06-04 19:39:33 -04:00
2011-04-03 19:11:38 -04:00
FILE * RsDirUtil : : rs_fopen ( const char * filename , const char * mode )
{
# ifdef WINDOWS_SYS
std : : wstring wfilename ;
librs : : util : : ConvertUtf8ToUtf16 ( filename , wfilename ) ;
std : : wstring wmode ;
librs : : util : : ConvertUtf8ToUtf16 ( mode , wmode ) ;
return _wfopen ( wfilename . c_str ( ) , wmode . c_str ( ) ) ;
# else
2011-04-03 19:27:36 -04:00
return fopen64 ( filename , mode ) ;
2011-04-03 19:11:38 -04:00
# endif
}
2011-05-15 08:42:55 -04:00
std : : string RsDirUtil : : convertPathToUnix ( std : : string path )
{
for ( unsigned int i = 0 ; i < path . length ( ) ; i + + )
{
if ( path [ i ] = = ' \\ ' )
path [ i ] = ' / ' ;
}
return path ;
}
std : : string RsDirUtil : : makePath ( const std : : string & path1 , const std : : string & path2 )
{
std : : string path = path1 ;
if ( path . empty ( ) = = false & & * path . rbegin ( ) ! = ' / ' ) {
path + = " / " ;
}
path + = path2 ;
return path ;
}
2012-06-14 18:53:02 -04:00
int RsDirUtil : : createLockFile ( const std : : string & lock_file_path , rs_lock_handle_t & lock_handle )
2012-06-12 16:31:13 -04:00
{
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
# ifndef WINDOWS_SYS
// Suspended. The user should make sure he's not already using the file descriptor.
// if(lock_handle != -1)
// close(lock_handle);
// open the file in write mode, create it if necessary, truncate it (it should be empty)
lock_handle = open ( lock_file_path . c_str ( ) , O_WRONLY | O_CREAT | O_TRUNC , S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH ) ;
if ( lock_handle = = - 1 )
{
std : : cerr < < " Could not open lock file " < < lock_file_path . c_str ( ) < < std : : flush ;
perror ( NULL ) ;
return 2 ;
}
// see "man fcntl" for the details, in short: non blocking lock creation on the whole file contents
struct flock lockDetails ;
lockDetails . l_type = F_WRLCK ;
lockDetails . l_whence = SEEK_SET ;
lockDetails . l_start = 0 ;
lockDetails . l_len = 0 ;
if ( fcntl ( lock_handle , F_SETLK , & lockDetails ) = = - 1 )
{
int fcntlErr = errno ;
std : : cerr < < " Could not request lock on file " < < lock_file_path . c_str ( ) < < std : : flush ;
perror ( NULL ) ;
// there's no lock so let's release the file handle immediately
close ( lock_handle ) ;
lock_handle = - 1 ;
if ( fcntlErr = = EACCES | | fcntlErr = = EAGAIN )
return 1 ;
else
return 2 ;
}
return 0 ;
# else
// Suspended. The user should make sure he's not already using the file descriptor.
//
// if (lock_handle) {
// CloseHandle(lock_handle);
// }
std : : wstring wlockFile ;
librs : : util : : ConvertUtf8ToUtf16 ( lock_file_path , wlockFile ) ;
// open the file in write mode, create it if necessary
lock_handle = CreateFile ( wlockFile . c_str ( ) , GENERIC_READ | GENERIC_WRITE , 0 , NULL , CREATE_ALWAYS , 0 , NULL ) ;
if ( lock_handle = = INVALID_HANDLE_VALUE )
{
DWORD lasterror = GetLastError ( ) ;
std : : cerr < < " Could not open lock file " < < lock_file_path . c_str ( ) < < std : : endl ;
std : : cerr < < " Last error: " < < lasterror < < std : : endl < < std : : flush ;
perror ( NULL ) ;
if ( lasterror = = ERROR_SHARING_VIOLATION | | lasterror = = ERROR_ACCESS_DENIED )
return 1 ;
return 2 ;
}
return 0 ;
# endif
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
}
2012-06-14 18:53:02 -04:00
void RsDirUtil : : releaseLockFile ( rs_lock_handle_t lockHandle )
2012-06-12 16:31:13 -04:00
{
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
# ifndef WINDOWS_SYS
if ( lockHandle ! = - 1 )
{
close ( lockHandle ) ;
lockHandle = - 1 ;
}
# else
if ( lockHandle )
{
CloseHandle ( lockHandle ) ;
lockHandle = 0 ;
}
# endif
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
}
RsStackFileLock : : RsStackFileLock ( const std : : string & file_path )
{
while ( RsDirUtil : : createLockFile ( file_path , _file_handle ) )
{
std : : cerr < < " Cannot acquire file lock " < < file_path < < " , waiting 1 sec. " < < std : : endl ;
2014-10-29 05:38:22 -04:00
usleep ( 1 * 1000 * 1000 ) ; // 1 sec
2012-06-12 16:31:13 -04:00
}
2012-09-17 16:09:38 -04:00
# ifdef RSDIR_DEBUG
2012-06-12 16:31:13 -04:00
std : : cerr < < " Acquired file handle " < < _file_handle < < " , lock file: " < < file_path < < std : : endl ;
2012-09-17 16:09:38 -04:00
# endif
2012-06-12 16:31:13 -04:00
}
RsStackFileLock : : ~ RsStackFileLock ( )
{
RsDirUtil : : releaseLockFile ( _file_handle ) ;
2012-09-17 16:09:38 -04:00
# ifdef RSDIR_DEBUG
2012-06-12 16:31:13 -04:00
std : : cerr < < " Released file lock with handle " < < _file_handle < < std : : endl ;
2012-09-17 16:09:38 -04:00
# endif
2012-06-12 16:31:13 -04:00
}
2009-09-08 17:30:59 -04:00
#if 0 // NOT ENABLED YET!
2009-09-08 16:18:43 -04:00
/************************* WIDE STRING ***************************/
/************************* WIDE STRING ***************************/
/************************* WIDE STRING ***************************/
std : : wstring RsDirUtil : : getWideTopDir ( std : : wstring dir )
{
std : : wstring top ;
/* find the subdir: [/][dir1.../]<top>[/]
*/
int i , j ;
int len = dir . length ( ) ;
for ( j = len - 1 ; ( j > 0 ) & & ( dir [ j ] = = ' / ' ) ; j - - ) ;
for ( i = j ; ( i > 0 ) & & ( dir [ i ] ! = ' / ' ) ; i - - ) ;
if ( dir [ i ] = = ' / ' )
i + + ;
for ( ; i < = j ; i + + )
{
top + = dir [ i ] ;
}
return top ;
}
std : : wstring RsDirUtil : : removeWideTopDir ( std : : wstring dir )
{
std : : wstring rest ;
/* remove the subdir: [/][dir1.../]<top>[/]
*/
int i , j ;
int len = dir . length ( ) ;
for ( j = len - 1 ; ( j > 0 ) & & ( dir [ j ] = = ' / ' ) ; j - - ) ;
for ( i = j ; ( i > = 0 ) & & ( dir [ i ] ! = ' / ' ) ; i - - ) ;
/* remove any more slashes */
for ( ; ( i > = 0 ) & & ( dir [ i ] = = ' / ' ) ; i - - ) ;
for ( j = 0 ; j < = i ; j + + )
{
rest + = dir [ j ] ;
}
return rest ;
}
std : : wstring RsDirUtil : : getWideRootDir ( std : : wstring dir )
{
std : : wstring root ;
/* find the subdir: [/]root[/...]
*/
int i , j ;
int len = dir . length ( ) ;
for ( i = 0 ; ( i < len ) & & ( dir [ i ] = = ' / ' ) ; i + + ) ;
for ( j = i ; ( j < len ) & & ( dir [ j ] ! = ' / ' ) ; j + + ) ;
if ( i = = j )
return root ; /* empty */
for ( ; i < j ; i + + )
{
root + = dir [ i ] ;
}
return root ;
}
std : : wstring RsDirUtil : : removeWideRootDir ( std : : wstring path )
{
unsigned int i , j ;
unsigned int len = path . length ( ) ;
std : : wstring output ;
/* chew leading '/'s */
for ( i = 0 ; ( i < len ) & & ( path [ i ] = = ' / ' ) ; i + + ) ;
if ( i = = len )
return output ; /* empty string */
for ( j = i ; ( j < len ) & & ( path [ j ] ! = ' / ' ) ; j + + ) ; /* run to next '/' */
for ( ; ( j < len ) & & ( path [ j ] = = ' / ' ) ; j + + ) ; /* chew leading '/'s */
for ( ; j < len ; j + + )
{
output + = path [ j ] ;
}
return output ;
}
std : : wstring RsDirUtil : : removeWideRootDirs ( std : : wstring path , std : : wstring root )
{
/* too tired */
std : : wstring notroot ;
unsigned int i = 0 , j = 0 ;
/* catch empty data */
if ( ( root . length ( ) < 1 ) | | ( path . length ( ) < 1 ) )
return notroot ;
if ( ( path [ 0 ] = = ' / ' ) & & ( root [ 0 ] ! = ' / ' ) )
{
i + + ;
}
for ( ; ( i < path . length ( ) ) & & ( j < root . length ( ) ) & & ( path [ i ] = = root [ j ] ) ; i + + , j + + ) ;
/* should have consumed root. */
if ( j = = root . length ( ) )
{
//std::cerr << "matched root!" << std::endl;
}
else
{
//std::cerr << "failed i: " << i << ", j: " << j << std::endl;
//std::cerr << "root: " << root << " path: " << path << std::endl;
return notroot ;
}
if ( path [ i ] = = ' / ' )
{
i + + ;
}
for ( ; i < path . length ( ) ; i + + )
{
notroot + = path [ i ] ;
}
//std::cerr << "Found NotRoot: " << notroot << std::endl;
return notroot ;
}
int RsDirUtil : : breakupWideDirList ( std : : wstring path ,
std : : list < std : : wstring > & subdirs )
{
int start = 0 ;
unsigned int i ;
for ( i = 0 ; i < path . length ( ) ; i + + )
{
if ( path [ i ] = = ' / ' )
{
if ( i - start > 0 )
{
subdirs . push_back ( path . substr ( start , i - start ) ) ;
}
start = i + 1 ;
}
}
// get the final one.
if ( i - start > 0 )
{
subdirs . push_back ( path . substr ( start , i - start ) ) ;
}
return 1 ;
}
bool RsDirUtil : : checkWideDirectory ( std : : wstring dir )
{
struct stat buf ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : string d ( dir . begin ( ) , dir . end ( ) ) ;
int val = stat ( d . c_str ( ) , & buf ) ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
if ( val = = - 1 )
{
# ifdef RSDIR_DEBUG
std : : cerr < < " RsDirUtil::checkDirectory() " ;
std : : cerr < < d < < " doesn't exist " < < std : : endl ;
# endif
return false ;
}
else if ( ! S_ISDIR ( buf . st_mode ) )
{
// Some other type - error.
# ifdef RSDIR_DEBUG
std : : cerr < < " RsDirUtil::checkDirectory() " ;
std : : cerr < < d < < " is not Directory " < < std : : endl ;
# endif
return false ;
}
return true ;
}
bool RsDirUtil : : checkWideCreateDirectory ( std : : wstring dir )
{
struct stat buf ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : string d ( dir . begin ( ) , dir . end ( ) ) ;
int val = stat ( d . c_str ( ) , & buf ) ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
if ( val = = - 1 )
{
// directory don't exist. create.
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
# ifndef WINDOWS_SYS // UNIX
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
if ( - 1 = = mkdir ( d . c_str ( ) , 0777 ) )
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
# else // WIN
if ( - 1 = = mkdir ( d . c_str ( ) ) )
# endif
/******************************** WINDOWS/UNIX SPECIFIC PART ******************/
{
# ifdef RSDIR_DEBUG
std : : cerr < < " check_create_directory() Fatal Error -- " ;
std : : cerr < < std : : endl < < " \t cannot create: " < < d < < std : : endl ;
# endif
return 0 ;
}
# ifdef RSDIR_DEBUG
std : : cerr < < " check_create_directory() " ;
std : : cerr < < std : : endl < < " \t created: " < < d < < std : : endl ;
# endif
}
else if ( ! S_ISDIR ( buf . st_mode ) )
{
// Some other type - error.
# ifdef RSDIR_DEBUG
std : : cerr < < " check_create_directory() Fatal Error -- " ;
std : : cerr < < std : : endl < < " \t " < < d < < " is nor Directory " < < std : : endl ;
# endif
return 0 ;
}
# ifdef RSDIR_DEBUG
std : : cerr < < " check_create_directory() " ;
std : : cerr < < std : : endl < < " \t Dir Exists: " < < d < < std : : endl ;
# endif
return 1 ;
}
# include <dirent.h>
bool RsDirUtil : : cleanupWideDirectory ( std : : wstring cleandir , std : : list < std : : wstring > keepFiles )
{
/* check for the dir existance */
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : string cd ( cleandir . begin ( ) , cleandir . end ( ) ) ;
DIR * dir = opendir ( cd . c_str ( ) ) ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : list < std : : wstring > : : const_iterator it ;
if ( ! dir )
{
return false ;
}
struct dirent * dent ;
struct stat buf ;
while ( NULL ! = ( dent = readdir ( dir ) ) )
{
/* check entry type */
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : string fname ( dent - > d_name ) ;
std : : wstring wfname ( fname . begin ( ) , fname . end ( ) ) ;
std : : string fullname = cd + " / " + fname ;
if ( - 1 ! = stat ( fullname . c_str ( ) , & buf ) )
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
{
/* only worry about files */
if ( S_ISREG ( buf . st_mode ) )
{
/* check if we should keep it */
if ( keepFiles . end ( ) = = ( it = std : : find ( keepFiles . begin ( ) , keepFiles . end ( ) , wfname ) ) )
{
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
/* can remove */
remove ( fullname . c_str ( ) ) ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
}
}
}
}
/* close directory */
closedir ( dir ) ;
return true ;
}
/* slightly nicer helper function */
bool RsDirUtil : : hashWideFile ( std : : wstring filepath ,
std : : wstring & name , std : : string & hash , uint64_t & size )
{
if ( getWideFileHash ( filepath , hash , size ) )
{
/* extract file name */
name = RsDirUtil : : getWideTopDir ( filepath ) ;
return true ;
}
return false ;
}
# include <openssl/sha.h>
# include <sstream>
# include <iomanip>
/* Function to hash, and get details of a file */
bool RsDirUtil : : getWideFileHash ( std : : wstring filepath ,
std : : string & hash , uint64_t & size )
{
FILE * fd ;
int len ;
SHA_CTX * sha_ctx = new SHA_CTX ;
unsigned char sha_buf [ SHA_DIGEST_LENGTH ] ;
unsigned char gblBuf [ 512 ] ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : string fp ( filepath . begin ( ) , filepath . end ( ) ) ;
2011-02-24 17:44:41 -05:00
if ( NULL = = ( fd = fopen64 ( fp . c_str ( ) , " rb " ) ) )
2009-09-08 16:18:43 -04:00
return false ;
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
/* determine size */
2011-02-24 17:19:04 -05:00
fseeko64 ( fd , 0 , SEEK_END ) ;
size = ftello64 ( fd ) ;
fseeko64 ( fd , 0 , SEEK_SET ) ;
2009-09-08 16:18:43 -04:00
SHA1_Init ( sha_ctx ) ;
while ( ( len = fread ( gblBuf , 1 , 512 , fd ) ) > 0 )
{
SHA1_Update ( sha_ctx , gblBuf , len ) ;
}
/* reading failed for some reason */
if ( ferror ( fd ) )
{
delete sha_ctx ;
fclose ( fd ) ;
return false ;
}
SHA1_Final ( & sha_buf [ 0 ] , sha_ctx ) ;
2012-04-15 10:37:44 -04:00
std : : ostringstream tmpout ; // please do not use std::ostringstream
2009-09-08 16:18:43 -04:00
for ( int i = 0 ; i < SHA_DIGEST_LENGTH ; i + + )
{
tmpout < < std : : setw ( 2 ) < < std : : setfill ( ' 0 ' ) < < std : : hex < < ( unsigned int ) ( sha_buf [ i ] ) ;
}
hash = tmpout . str ( ) ;
delete sha_ctx ;
fclose ( fd ) ;
return true ;
}
bool RsDirUtil : : renameWideFile ( const std : : wstring & from , const std : : wstring & to )
{
int loops = 0 ;
# if defined(WIN32) || defined(MINGW) || defined(__CYGWIN__)
# ifdef WIN_CROSS_UBUNTU
std : : wstring f , t ;
for ( int i = 0 ; i < from . size ( ) ; + + i ) f . push_back ( from [ i ] ) ;
for ( int i = 0 ; i < to . size ( ) ; + + i ) t . push_back ( to [ i ] ) ;
# else
std : : wstring f ( from ) , t ( to ) ;
# endif
2014-09-20 15:54:04 -04:00
while ( ! MoveFileEx ( f . c_str ( ) , t . c_str ( ) , MOVEFILE_REPLACE_EXISTING | MOVEFILE_WRITE_THROUGH ) )
2009-09-08 16:18:43 -04:00
# else
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
std : : string f ( from . begin ( ) , from . end ( ) ) ;
std : : string t ( to . begin ( ) , to . end ( ) ) ;
while ( rename ( f . c_str ( ) , t . c_str ( ) ) < 0 )
/***** XXX TO MAKE WIDE SYSTEM CALL ******************************************************/
# endif
{
# ifdef WIN32
if ( GetLastError ( ) ! = ERROR_ACCESS_DENIED )
# else
if ( errno ! = EACCES )
# endif
/* set errno? */
return false ;
2014-10-24 17:31:58 -04:00
usleep ( 100 * 1000 ) ; //100 msec
2009-09-08 16:18:43 -04:00
if ( loops > = 30 )
return false ;
loops + + ;
}
return true ;
}
2009-03-29 09:58:28 -04:00
2009-09-08 17:30:59 -04:00
# endif // WIDE STUFF NOT ENABLED YET!