Bruno Luciani wrote:
Which is the function that you say ?
I have attached the code. It is a hack of the Harbour DIRECTORY() implementation. It may not be up to date with the code in SVN. I hope it helps.
Regards Alex
/* * $Id: direct.c 11528 2009-06-26 00:33:38Z druzus $ */ /* * Harbour Project source code: * DIRECTORY() function * * Copyright 1999 Leslee Griffith <les.griff...@vantagesystems.ca> * www - http://www.harbour-project.org * * 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, 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 software; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 59 Temple Place, Suite 330, * Boston, MA 02111-1307 USA (or visit the web site http://www.gnu.org/). * * As a special exception, the Harbour Project gives permission for * additional uses of the text contained in its release of Harbour. * * The exception is that, if you link the Harbour libraries with other * files to produce an executable, this does not by itself cause the * resulting executable to be covered by the GNU General Public License. * Your use of that executable is in no way restricted on account of * linking the Harbour library code into it. * * This exception does not however invalidate any other reasons why * the executable file might be covered by the GNU General Public License. * * This exception applies only to the code released by the Harbour * Project under the name Harbour. If you copy code from other * Harbour Project or Free Software Foundation releases into a copy of * Harbour, as the General Public License permits, the exception does * not apply to the code that you add in this way. To avoid misleading * anyone as to the status of such modified files, you must delete * this exception notice from them. * * If you write modifications of your own for Harbour, it is your choice * whether to permit this exception to apply to your modifications. * If you do not wish that, delete this exception notice. * */ /* * Notes from the fringe... <ptuc...@sympatico.ca> * * Clipper is a bit schizoid with the treatment of file attributes, but we've * emulated that weirdness here for your viewing amusement. * * In Clippers' homeworld of DOS, there are 5 basic attributes: 'A'rchive, * 'H'idden, 'S'ystem, 'R'eadonly and 'D'irectory. In addition, a file can * have no attributes, and only 1 file per physical partition can have the * 'V'olume label. * * For a given file request, it is implied that the attribute mask includes * all attributes except 'H'idden, 'S'ystem, 'D'irectory and 'V'olume. * The returned file list will always include (for instance) 'R'eadOnly files * unless they also happen to be 'H'idden and that attribute was not requested. * * "V" is a special case - you will get back the entry that describes the * volume label for the drive implied by the filemask. * * Differences from the 'standard' (where supported): * - Filenames will be returned in the same case as they are stored in the * directory. Clipper (and VO too) will convert the names to upper case * - Filenames will be the full filename as supported by the OS in use. * - There are a number of additional file attributes returned. * They are: * 'I' - DEVICE File is a device * 'T' - TEMPORARY File is a Temporary file * 'P' - SPARSE File is Sparse * 'L' - REPARSE File/Dir is a reparse point * 'C' - COMPRESSED File/Dir is compressed * 'O' - OFFLINE File/Dir is not online * 'X' - NOTINDEXED Exclude File/Dir from Indexing Service * 'E' - ENCRYPTED File/Dir is Encrypted * 'M' - VOLCOMP Volume Supports Compression * - Clipper can sometimes drop the ReadOnly indication of directories. * Harbour detects this correctly. * * TODO: - check that path support vis stat works on all platforms * - UNC Support? ie: dir \\myserver\root * */ #include "hbapi.h" #include "hbapifs.h" /* File Find API functions */ extern HB_EXPORT PHB_FFIND sfn_hb_fsFindFirst( const char * pszFileName, ULONG ulAttrMask, BOOL * bUse ); extern HB_EXPORT BOOL sfn_hb_fsFindNext( PHB_FFIND ffind, BOOL * bUse ); extern HB_EXPORT void sfn_hb_fsFindClose( PHB_FFIND ffind ); #include "hbapiitm.h" #include "directry.ch" /* NOTE: 8.3 support should be added in a separate way, like as a function which converts full names to 8.3 names, since this issue is very much platform specific, and this is not the only place which may need the conversion [vszakats]. */ HB_FUNC( SFN_DIRECTORY ) { const char * szDirSpec = hb_parc( 1 ); const char * szAttributes = hb_parc( 2 ); char * pszFree = NULL; ULONG ulMask; PHB_ITEM pDir = hb_itemArrayNew( 0 ); PHB_FFIND ffind; BOOL bUse = FALSE; /* Get the passed attributes and convert them to Harbour Flags */ ulMask = HB_FA_ARCHIVE | HB_FA_READONLY | HB_FA_DEVICE | HB_FA_TEMPORARY | HB_FA_SPARSE | HB_FA_REPARSE | HB_FA_COMPRESSED | HB_FA_OFFLINE | HB_FA_NOTINDEXED | HB_FA_ENCRYPTED | HB_FA_VOLCOMP; if( szAttributes && *szAttributes ) { if( ( ulMask |= hb_fsAttrEncode( szAttributes ) ) & HB_FA_LABEL ) { /* NOTE: This is Clipper Doc compatible. (not operationally) */ ulMask = HB_FA_LABEL; } } if( szDirSpec && *szDirSpec ) { szDirSpec = hb_fsNameConv( szDirSpec, &pszFree ); if( ulMask != HB_FA_LABEL ) { if( *szDirSpec ) { /* CA-Cl*pper compatible behavior - add all file mask when * last character is directory or drive separator */ int iLen = strlen( szDirSpec ) - 1; #ifdef HB_OS_HAS_DRIVE_LETTER if( szDirSpec[ iLen ] == HB_OS_PATH_DELIM_CHR || szDirSpec[ iLen ] == HB_OS_DRIVE_DELIM_CHR ) #else if( szDirSpec[ iLen ] == HB_OS_PATH_DELIM_CHR ) #endif { if( pszFree ) { char * pszTemp = hb_xstrcpy( NULL, szDirSpec, HB_OS_ALLFILE_MASK, NULL ); hb_xfree( pszFree ); szDirSpec = pszFree = pszTemp; } else { szDirSpec = pszFree = hb_xstrcpy( NULL, szDirSpec, HB_OS_ALLFILE_MASK, NULL ); } } } else szDirSpec = HB_OS_ALLFILE_MASK; } } else szDirSpec = HB_OS_ALLFILE_MASK; /* Get the file list */ if( ( ffind = sfn_hb_fsFindFirst( szDirSpec, ulMask, &bUse ) ) != NULL ) { PHB_ITEM pSubarray = hb_itemNew( NULL ); do { char buffer[ 32 ]; if ( bUse ) { hb_arrayNew( pSubarray, F_LEN ); hb_arraySetC ( pSubarray, F_NAME, ffind->szName ); hb_arraySetNInt( pSubarray, F_SIZE, ffind->size ); hb_arraySetDL ( pSubarray, F_DATE, ffind->lDate ); hb_arraySetC ( pSubarray, F_TIME, ffind->szTime ); hb_arraySetC ( pSubarray, F_ATTR, hb_fsAttrDecode( ffind->attr, buffer ) ); /* Don't exit when array limit is reached */ hb_arrayAddForward( pDir, pSubarray ); } bUse = FALSE; } while( sfn_hb_fsFindNext( ffind, &bUse ) ); hb_itemRelease( pSubarray ); sfn_hb_fsFindClose( ffind ); } if( pszFree ) hb_xfree( pszFree ); hb_itemReturnRelease( pDir ); }
/* * $Id: hbffind.c 12793 2009-10-30 16:56:47Z druzus $ */ /* * Harbour Project source code: * Harbour File Find API (C level) * * Copyright 2001-2002 Luiz Rafael Culik <cu...@sl.conex.net> * Viktor Szakats (harbour.01 syenar.hu) * Paul Tucker <ptuc...@sympatico.ca> * www - http://www.harbour-project.org * * 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, 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 software; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 59 Temple Place, Suite 330, * Boston, MA 02111-1307 USA (or visit the web site http://www.gnu.org/). * * As a special exception, the Harbour Project gives permission for * additional uses of the text contained in its release of Harbour. * * The exception is that, if you link the Harbour libraries with other * files to produce an executable, this does not by itself cause the * resulting executable to be covered by the GNU General Public License. * Your use of that executable is in no way restricted on account of * linking the Harbour library code into it. * * This exception does not however invalidate any other reasons why * the executable file might be covered by the GNU General Public License. * * This exception applies only to the code released by the Harbour * Project under the name Harbour. If you copy code from other * Harbour Project or Free Software Foundation releases into a copy of * Harbour, as the General Public License permits, the exception does * not apply to the code that you add in this way. To avoid misleading * anyone as to the status of such modified files, you must delete * this exception notice from them. * * If you write modifications of your own for Harbour, it is your choice * whether to permit this exception to apply to your modifications. * If you do not wish that, delete this exception notice. * */ #if !defined( _LARGEFILE64_SOURCE ) # define _LARGEFILE64_SOURCE #endif #define INCL_DOSFILEMGR #define INCL_DOSERRORS #define HB_OS_WIN_USED #include "hbapi.h" #include "hbapifs.h" /* File Find API functions */ extern HB_EXPORT PHB_FFIND sfn_hb_fsFindFirst( const char * pszFileName, ULONG ulAttrMask, BOOL * bUse ); extern HB_EXPORT BOOL sfn_hb_fsFindNext( PHB_FFIND ffind, BOOL * bUse ); extern HB_EXPORT void sfn_hb_fsFindClose( PHB_FFIND ffind ); #include "hbvm.h" #include "hbdate.h" #include "hb_io.h" /* ------------------------------------------------------------- */ #if defined( HB_OS_DOS ) #if defined( __DJGPP__ ) || defined( __RSX32__ ) #include <sys/param.h> #endif #if defined( __DJGPP__ ) || defined( __RSX32__ ) || defined( __BORLANDC__ ) #include <sys/stat.h> #endif #include <dos.h> #if !defined( __WATCOMC__ ) #include <dir.h> #endif #include <time.h> #if defined( __WATCOMC__ ) typedef struct { struct find_t entry; } HB_FFIND_INFO, * PHB_FFIND_INFO; #define FA_ARCH _A_ARCH #define FA_DIREC _A_SUBDIR #define FA_HIDDEN _A_HIDDEN #define FA_RDONLY _A_RDONLY #define FA_LABEL _A_VOLID #define FA_SYSTEM _A_SYSTEM #define ff_name name #define ff_fsize size #define ff_attrib attrib #else typedef struct { struct ffblk entry; } HB_FFIND_INFO, * PHB_FFIND_INFO; #endif #elif defined( HB_OS_OS2 ) #include <sys/types.h> #include <sys/stat.h> #include <time.h> typedef struct { HDIR hFindFile; FILEFINDBUF3 entry; ULONG fileTypes; ULONG findSize; ULONG findCount; } HB_FFIND_INFO, * PHB_FFIND_INFO; #elif defined( HB_OS_WIN ) typedef struct { HANDLE hFindFile; WIN32_FIND_DATA pFindFileData; DWORD dwAttr; } HB_FFIND_INFO, * PHB_FFIND_INFO; #define HB_WIN_MATCH() \ ( \ ( ( info->pFindFileData.dwFileAttributes & ( FILE_ATTRIBUTE_DIRECTORY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM ) ) == 0 ) || \ ( ( info->dwAttr & info->pFindFileData.dwFileAttributes & ( FILE_ATTRIBUTE_DIRECTORY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM ) ) != 0 ) \ ) #elif defined( HB_OS_UNIX ) #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <dirent.h> #include <time.h> typedef struct { DIR * dir; struct dirent * entry; char pattern[ HB_PATH_MAX ]; char path[ HB_PATH_MAX ]; } HB_FFIND_INFO, * PHB_FFIND_INFO; #else typedef void HB_FFIND_INFO, * PHB_FFIND_INFO; #endif #if !defined( HB_USE_LARGEFILE64 ) && defined( HB_OS_UNIX ) #if defined( __USE_LARGEFILE64 ) /* * The macro: __USE_LARGEFILE64 is set when _LARGEFILE64_SOURCE is * define and efectively enables lseek64/flock64/ftruncate64 functions * on 32bit machines. */ #define HB_USE_LARGEFILE64 #elif defined( HB_OS_UNIX ) && defined( O_LARGEFILE ) #define HB_USE_LARGEFILE64 #endif #endif /* ------------------------------------------------------------- */ #if defined( THIS_WONT_BE_DEFINED ) ULONG hb_fsAttrFromRaw( ULONG raw_attr ) { ULONG ulAttr; HB_TRACE(HB_TR_DEBUG, ("hb_fsAttrFromRaw(%lu)", raw_attr)); #if defined( HB_OS_DOS ) ulAttr = 0; if( raw_attr & FA_ARCH ) ulAttr |= HB_FA_ARCHIVE; if( raw_attr & FA_DIREC ) ulAttr |= HB_FA_DIRECTORY; if( raw_attr & FA_HIDDEN ) ulAttr |= HB_FA_HIDDEN; if( raw_attr & FA_RDONLY ) ulAttr |= HB_FA_READONLY; if( raw_attr & FA_LABEL ) ulAttr |= HB_FA_LABEL; if( raw_attr & FA_SYSTEM ) ulAttr |= HB_FA_SYSTEM; #elif defined( HB_OS_OS2 ) ulAttr = 0; if( raw_attr & FILE_ARCHIVED ) ulAttr |= HB_FA_ARCHIVE; if( raw_attr & FILE_DIRECTORY ) ulAttr |= HB_FA_DIRECTORY; if( raw_attr & FILE_HIDDEN ) ulAttr |= HB_FA_HIDDEN; if( raw_attr & FILE_READONLY ) ulAttr |= HB_FA_READONLY; if( raw_attr & FILE_SYSTEM ) ulAttr |= HB_FA_SYSTEM; #elif defined( HB_OS_WIN ) ulAttr = 0; if( raw_attr & FILE_ATTRIBUTE_ARCHIVE ) ulAttr |= HB_FA_ARCHIVE; if( raw_attr & FILE_ATTRIBUTE_DIRECTORY ) ulAttr |= HB_FA_DIRECTORY; if( raw_attr & FILE_ATTRIBUTE_HIDDEN ) ulAttr |= HB_FA_HIDDEN; if( raw_attr & FILE_ATTRIBUTE_READONLY ) ulAttr |= HB_FA_READONLY; if( raw_attr & FILE_ATTRIBUTE_SYSTEM ) ulAttr |= HB_FA_SYSTEM; if( raw_attr & FILE_ATTRIBUTE_NORMAL ) ulAttr |= HB_FA_NORMAL; /* Note that FILE_ATTRIBUTE_NORMAL is not needed HB_FA_DEVICE not supported HB_FA_VOLCOMP needs to be checked */ if( raw_attr & FILE_ATTRIBUTE_ENCRYPTED ) ulAttr |= HB_FA_ENCRYPTED; if( raw_attr & FILE_ATTRIBUTE_TEMPORARY ) ulAttr |= HB_FA_TEMPORARY; if( raw_attr & FILE_ATTRIBUTE_SPARSE_FILE ) ulAttr |= HB_FA_SPARSE; if( raw_attr & FILE_ATTRIBUTE_REPARSE_POINT ) ulAttr |= HB_FA_REPARSE; if( raw_attr & FILE_ATTRIBUTE_COMPRESSED ) ulAttr |= HB_FA_COMPRESSED; if( raw_attr & FILE_ATTRIBUTE_OFFLINE ) ulAttr |= HB_FA_OFFLINE; /* FILE_ATTRIBUTE_NOT_CONTENT_INDEXED */ /* not defined in some older winnt.h */ if( raw_attr & 0x00002000 ) ulAttr |= HB_FA_NOTINDEXED; if( raw_attr & 0x00008000 ) ulAttr |= HB_FA_VOLCOMP; #elif defined( HB_OS_UNIX ) ulAttr = ( ( raw_attr & S_IXOTH ) ? HB_FA_XOTH : 0 ) | ( ( raw_attr & S_IWOTH ) ? HB_FA_WOTH : 0 ) | ( ( raw_attr & S_IROTH ) ? HB_FA_ROTH : 0 ) | ( ( raw_attr & S_IXGRP ) ? HB_FA_XGRP : 0 ) | ( ( raw_attr & S_IWGRP ) ? HB_FA_WGRP : 0 ) | ( ( raw_attr & S_IRGRP ) ? HB_FA_RGRP : 0 ) | ( ( raw_attr & S_IXUSR ) ? HB_FA_XUSR : 0 ) | ( ( raw_attr & S_IWUSR ) ? HB_FA_WUSR : 0 ) | ( ( raw_attr & S_IRUSR ) ? HB_FA_RUSR : 0 ) | ( ( raw_attr & S_ISVTX ) ? HB_FA_SVTX : 0 ) | ( ( raw_attr & S_ISGID ) ? HB_FA_SGID : 0 ) | ( ( raw_attr & S_ISUID ) ? HB_FA_SUID : 0 ); if( S_ISREG( raw_attr ) ) ulAttr |= HB_FA_FILE; if( S_ISDIR( raw_attr ) ) ulAttr |= HB_FA_DIRECTORY; if( S_ISLNK( raw_attr ) ) ulAttr |= HB_FA_LINK; if( S_ISCHR( raw_attr ) ) ulAttr |= HB_FA_CHRDEVICE; if( S_ISBLK( raw_attr ) ) ulAttr |= HB_FA_BLKDEVICE; if( S_ISFIFO( raw_attr ) ) ulAttr |= HB_FA_FIFO; if( S_ISSOCK( raw_attr ) ) ulAttr |= HB_FA_SOCKET; #else HB_SYMBOL_UNUSED( raw_attr ); ulAttr = 0; #endif return ulAttr; } ULONG hb_fsAttrToRaw( ULONG ulAttr ) { ULONG raw_attr; HB_TRACE(HB_TR_DEBUG, ("hb_fsAttrToRaw(%lu)", ulAttr)); #if defined( HB_OS_DOS ) raw_attr = 0; if( ulAttr & HB_FA_ARCHIVE ) raw_attr |= FA_ARCH; if( ulAttr & HB_FA_DIRECTORY ) raw_attr |= FA_DIREC; if( ulAttr & HB_FA_HIDDEN ) raw_attr |= FA_HIDDEN; if( ulAttr & HB_FA_READONLY ) raw_attr |= FA_RDONLY; if( ulAttr & HB_FA_LABEL ) raw_attr |= FA_LABEL; if( ulAttr & HB_FA_SYSTEM ) raw_attr |= FA_SYSTEM; #elif defined( HB_OS_OS2 ) raw_attr = 0; if( ulAttr & HB_FA_ARCHIVE ) raw_attr |= FILE_ARCHIVED; if( ulAttr & HB_FA_DIRECTORY ) raw_attr |= FILE_DIRECTORY; if( ulAttr & HB_FA_HIDDEN ) raw_attr |= FILE_HIDDEN; if( ulAttr & HB_FA_READONLY ) raw_attr |= FILE_READONLY; if( ulAttr & HB_FA_SYSTEM ) raw_attr |= FILE_SYSTEM; #elif defined( HB_OS_WIN ) raw_attr = 0; if( ulAttr & HB_FA_ARCHIVE ) raw_attr |= FILE_ATTRIBUTE_ARCHIVE; if( ulAttr & HB_FA_DIRECTORY ) raw_attr |= FILE_ATTRIBUTE_DIRECTORY; if( ulAttr & HB_FA_HIDDEN ) raw_attr |= FILE_ATTRIBUTE_HIDDEN; if( ulAttr & HB_FA_READONLY ) raw_attr |= FILE_ATTRIBUTE_READONLY; if( ulAttr & HB_FA_SYSTEM ) raw_attr |= FILE_ATTRIBUTE_SYSTEM; if( ulAttr & HB_FA_NORMAL ) raw_attr |= FILE_ATTRIBUTE_NORMAL; /* Note that FILE_ATTRIBUTE_NORMAL is not needed HB_FA_DEVICE not supported HB_FA_VOLCOMP needs to be checked */ if( ulAttr & HB_FA_ENCRYPTED ) raw_attr |= FILE_ATTRIBUTE_ENCRYPTED; if( ulAttr & HB_FA_TEMPORARY ) raw_attr |= FILE_ATTRIBUTE_TEMPORARY; if( ulAttr & HB_FA_SPARSE ) raw_attr |= FILE_ATTRIBUTE_SPARSE_FILE; if( ulAttr & HB_FA_REPARSE ) raw_attr |= FILE_ATTRIBUTE_REPARSE_POINT; if( ulAttr & HB_FA_COMPRESSED ) raw_attr |= FILE_ATTRIBUTE_COMPRESSED; if( ulAttr & HB_FA_OFFLINE ) raw_attr |= FILE_ATTRIBUTE_OFFLINE; if( ulAttr & HB_FA_NOTINDEXED ) raw_attr |= 0x00002000; /* FILE_ATTRIBUTE_NOT_CONTENT_INDEXED not defined in some older winnt.h */ if( ulAttr & HB_FA_VOLCOMP ) raw_attr |= 0x00008000; #elif defined( HB_OS_UNIX ) raw_attr = HB_FA_POSIX_ATTR( ulAttr ); if( ulAttr & HB_FA_FILE ) raw_attr |= S_IFREG; if( ulAttr & HB_FA_DIRECTORY ) raw_attr |= S_IFDIR; if( ulAttr & HB_FA_LINK ) raw_attr |= S_IFLNK; if( ulAttr & HB_FA_CHRDEVICE ) raw_attr |= S_IFCHR; if( ulAttr & HB_FA_BLKDEVICE ) raw_attr |= S_IFBLK; if( ulAttr & HB_FA_FIFO ) raw_attr |= S_IFIFO; if( ulAttr & HB_FA_SOCKET ) raw_attr |= S_IFSOCK; #else HB_SYMBOL_UNUSED( ulAttr ); raw_attr = 0; #endif return raw_attr; } /* Converts a CA-Cl*pper compatible file attribute string to the internal reprensentation. */ ULONG hb_fsAttrEncode( const char * szAttr ) { const char * pos = szAttr; char ch; ULONG ulAttr = 0; HB_TRACE(HB_TR_DEBUG, ("hb_fsAttrEncode(%p)", szAttr)); while( ( ch = ( char ) HB_TOUPPER( *pos ) ) != '\0' ) { switch( ch ) { case 'R': ulAttr |= HB_FA_READONLY; break; case 'H': ulAttr |= HB_FA_HIDDEN; break; case 'S': ulAttr |= HB_FA_SYSTEM; break; case 'V': ulAttr |= HB_FA_LABEL; break; case 'D': ulAttr |= HB_FA_DIRECTORY; break; case 'A': ulAttr |= HB_FA_ARCHIVE; break; } pos++; } return ulAttr; } /* Converts a file attribute (ffind->attr) to the CA-Cl*pper compatible file attribute string format. */ /* NOTE: szAttr buffer must be at least 16 chars long */ char * hb_fsAttrDecode( ULONG ulAttr, char * szAttr ) { char * ptr = szAttr; HB_TRACE(HB_TR_DEBUG, ("hb_fsAttrDecode(%lu, %p)", ulAttr, szAttr)); /* Using the same order as CA-Cl*pper did: RHSVDA. */ if( ulAttr & HB_FA_READONLY ) *ptr++ = 'R'; if( ulAttr & HB_FA_HIDDEN ) *ptr++ = 'H'; if( ulAttr & HB_FA_SYSTEM ) *ptr++ = 'S'; if( ulAttr & HB_FA_LABEL ) *ptr++ = 'V'; if( ulAttr & HB_FA_DIRECTORY ) *ptr++ = 'D'; if( ulAttr & HB_FA_ARCHIVE ) *ptr++ = 'A'; *ptr = '\0'; return szAttr; } #endif /* Finds the first then the next matching file on each call. Does low-level (platform dependent filtering if needed. */ static BOOL sfn_hb_fsFindNextLow( PHB_FFIND ffind, BOOL * bUse ) { BOOL bFound; int iYear = 0; int iMonth = 0; int iDay = 0; int iHour = 0; int iMin = 0; int iSec = 0; ULONG raw_attr = 0; /* Set the default values in case some platforms don't support some of these, or they may fail on them. */ ffind->szName[ 0 ] = '\0'; ffind->size = 0; /* Do platform dependant first/next search */ hb_vmUnlock(); #if defined( HB_OS_DOS ) { PHB_FFIND_INFO info = ( PHB_FFIND_INFO ) ffind->info; /* Handling HB_FA_LABEL doesn't need any special tricks under the DOS platform. */ if( ffind->bFirst ) { ffind->bFirst = FALSE; /* tzset(); */ #if defined( __WATCOMC__ ) bFound = ( _dos_findfirst( ffind->pszFileMask, ( USHORT ) hb_fsAttrToRaw( ffind->attrmask ), &info->entry ) == 0 ); #else bFound = ( findfirst( ffind->pszFileMask, &info->entry, ( USHORT ) hb_fsAttrToRaw( ffind->attrmask ) ) == 0 ); #endif } else { #if defined( __WATCOMC__ ) bFound = ( _dos_findnext( &info->entry ) == 0 ); #else bFound = ( findnext( &info->entry ) == 0 ); #endif } /* Fill Harbour found file info */ if( bFound ) { hb_strncpy( ffind->szName, info->entry.ff_name, sizeof( ffind->szName ) - 1 ); ffind->size = info->entry.ff_fsize; raw_attr = info->entry.ff_attrib; { time_t ftime; struct tm * ft; struct stat sStat; stat( info->entry.ff_name, &sStat ); ftime = sStat.st_mtime; ft = localtime( &ftime ); iYear = ft->tm_year + 1900; iMonth = ft->tm_mon + 1; iDay = ft->tm_mday; iHour = ft->tm_hour; iMin = ft->tm_min; iSec = ft->tm_sec; } } hb_fsSetIOError( bFound, 0 ); } #elif defined( HB_OS_OS2 ) { PHB_FFIND_INFO info = ( PHB_FFIND_INFO ) ffind->info; /* TODO: HB_FA_LABEL handling */ if( ffind->bFirst ) { ffind->bFirst = FALSE; /* tzset(); */ info->hFindFile = HDIR_CREATE; info->findCount = 1; bFound = DosFindFirst( ( PCSZ ) ffind->pszFileMask, &info->hFindFile, ( LONG ) hb_fsAttrToRaw( ffind->attrmask ), &info->entry, sizeof( info->entry ), &info->findCount, FIL_STANDARD ) == NO_ERROR && info->findCount > 0; } else bFound = DosFindNext( info->hFindFile, &info->entry, sizeof( info->entry ), &info->findCount ) == NO_ERROR && info->findCount > 0; /* Fill Harbour found file info */ if( bFound ) { hb_strncpy( ffind->szName, info->entry.achName, sizeof( ffind->szName ) - 1 ); ffind->size = info->entry.cbFile; raw_attr = info->entry.attrFile; iYear = info->entry.fdateLastWrite.year + 1980; iMonth = info->entry.fdateLastWrite.month; iDay = info->entry.fdateLastWrite.day; iHour = info->entry.ftimeLastWrite.hours; iMin = info->entry.ftimeLastWrite.minutes; iSec = info->entry.ftimeLastWrite.twosecs; } hb_fsSetIOError( bFound, 0 ); } #elif defined( HB_OS_WIN ) { PHB_FFIND_INFO info = ( PHB_FFIND_INFO ) ffind->info; bFound = FALSE; if( ffind->attrmask & HB_FA_LABEL ) { #if !defined( HB_OS_WIN_CE ) if( ffind->bFirst ) { LPTSTR lpFileMask = HB_TCHAR_CONVTO( ffind->pszFileMask ); TCHAR szName[ HB_PATH_MAX ]; ffind->bFirst = FALSE; ffind->szName[ 0 ] = '\0'; bFound = GetVolumeInformation( lpFileMask, szName, sizeof( szName ), NULL, NULL, NULL, NULL, 0 ); HB_TCHAR_FREE( lpFileMask ); HB_TCHAR_GETFROM( ffind->szName, szName, sizeof( ffind->szName ) ); ffind->szName[ sizeof( ffind->szName ) - 1 ] = '\0'; } #endif } else { if( ffind->bFirst ) { LPTSTR lpFileMask = HB_TCHAR_CONVTO( ffind->pszFileMask ); ffind->bFirst = FALSE; info->hFindFile = FindFirstFile( lpFileMask, &info->pFindFileData ); info->dwAttr = ( DWORD ) hb_fsAttrToRaw( ffind->attrmask ); if( ( info->hFindFile != INVALID_HANDLE_VALUE ) && HB_WIN_MATCH() ) bFound = TRUE; HB_TCHAR_FREE( lpFileMask ); } if( ! bFound && info->hFindFile != INVALID_HANDLE_VALUE ) { while( FindNextFile( info->hFindFile, &info->pFindFileData ) ) { if( HB_WIN_MATCH() ) { bFound = TRUE; break; } } } /* Fill Harbour found file info */ if( bFound ) { //HB_TCHAR_GETFROM( ffind->szName, info->pFindFileData.cFileName, sizeof( ffind->szName ) - 1 ); if ( info->pFindFileData.cAlternateFileName[ 0 ] != '\0' ) { HB_TCHAR_GETFROM( ffind->szName, info->pFindFileData.cAlternateFileName, sizeof( ffind->szName ) - 1 ); *bUse = TRUE; } if( info->pFindFileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY ) ffind->size = 0; else { #if defined( __XCC__ ) || ( defined( __POCC__ ) && __POCC__ >= 500 ) /* NOTE: PellesC 5.00.1 will go into an infinite loop if we don't split this into two operations. [vszakats] */ ffind->size = ( HB_FOFFSET ) info->pFindFileData.nFileSizeLow; ffind->size += ( HB_FOFFSET ) info->pFindFileData.nFileSizeHigh << 32; #else ffind->size = ( HB_FOFFSET ) info->pFindFileData.nFileSizeLow + ( ( HB_FOFFSET ) info->pFindFileData.nFileSizeHigh << 32 ); #endif } raw_attr = ( ULONG ) info->pFindFileData.dwFileAttributes; /* NOTE: One of these may fail when searching on an UNC path, I don't know yet what's the reason. [vszakats] */ { FILETIME ft; SYSTEMTIME time; if( FileTimeToLocalFileTime( &info->pFindFileData.ftLastWriteTime, &ft ) && FileTimeToSystemTime( &ft, &time ) ) { iYear = time.wYear; iMonth = time.wMonth; iDay = time.wDay; iHour = time.wHour; iMin = time.wMinute; iSec = time.wSecond; } } } } hb_fsSetIOError( bFound, 0 ); } #elif defined( HB_OS_UNIX ) { PHB_FFIND_INFO info = ( PHB_FFIND_INFO ) ffind->info; char dirname[ HB_PATH_MAX ]; bFound = FALSE; /* TODO: HB_FA_LABEL handling */ if( ffind->bFirst ) { char * pos; ffind->bFirst = FALSE; hb_strncpy( dirname, ffind->pszFileMask, sizeof( dirname ) - 1 ); pos = strrchr( dirname, HB_OS_PATH_DELIM_CHR ); if( pos ) { hb_strncpy( info->pattern, pos + 1, sizeof( info->pattern ) - 1 ); *( pos + 1 ) = '\0'; } else { hb_strncpy( info->pattern, dirname, sizeof( info->pattern ) - 1 ); dirname[ 0 ] = '.'; dirname[ 1 ] = HB_OS_PATH_DELIM_CHR; dirname[ 2 ] = '\0'; } /* tzset(); */ info->dir = opendir( dirname ); hb_strncpy( info->path, dirname, sizeof( info->path ) - 1 ); } if( info->dir && info->pattern[ 0 ] != '\0' ) { while( ( info->entry = readdir( info->dir ) ) != NULL ) { if( hb_strMatchFile( info->entry->d_name, info->pattern ) ) { bFound = TRUE; break; } } } /* Fill Harbour found file info */ if( bFound ) { hb_strncpy( dirname, info->path, sizeof( dirname ) - 1 ); hb_strncat( dirname, info->entry->d_name, sizeof( dirname ) - 1 ); { time_t ftime; struct tm lt; #if defined( HB_USE_LARGEFILE64 ) struct stat64 sStat; if( stat64( dirname, &sStat ) == 0 ) #else struct stat sStat; if( stat( dirname, &sStat ) == 0 ) #endif { hb_strncpy( ffind->szName, info->entry->d_name, sizeof( ffind->szName ) - 1 ); ffind->size = sStat.st_size; raw_attr = sStat.st_mode; ftime = sStat.st_mtime; # if defined( HB_HAS_LOCALTIME_R ) localtime_r( &ftime, < ); # else lt = *localtime( &ftime ); # endif iYear = lt.tm_year + 1900; iMonth = lt.tm_mon + 1; iDay = lt.tm_mday; iHour = lt.tm_hour; iMin = lt.tm_min; iSec = lt.tm_sec; } else bFound = FALSE; } } hb_fsSetIOError( bFound, 0 ); } #else { int TODO; /* TODO: for given platform */ /* HB_SYMBOL_UNUSED( ffind ); */ HB_SYMBOL_UNUSED( iYear ); HB_SYMBOL_UNUSED( iMonth ); HB_SYMBOL_UNUSED( iDay ); HB_SYMBOL_UNUSED( iHour ); HB_SYMBOL_UNUSED( iMin ); HB_SYMBOL_UNUSED( iSec ); HB_SYMBOL_UNUSED( raw_attr ); bFound = FALSE; hb_fsSetError( ( USHORT ) FS_ERROR ); } #endif /* Fill common Harbour found file info */ if( bFound ) { /* Do the conversions common for all platforms */ ffind->szName[ sizeof( ffind->szName ) - 1 ] = '\0'; /* Convert from OS codepage */ { char * pszFree = NULL; ULONG ulSize = sizeof( ffind->szName ); const char * pszResult = hb_osDecodeCP( ffind->szName, &pszFree, &ulSize ); if( pszFree ) { hb_strncpy( ffind->szName, pszResult, sizeof( ffind->szName ) - 1 ); hb_xfree( pszFree ); } } ffind->attr = hb_fsAttrFromRaw( raw_attr ); ffind->lDate = hb_dateEncode( iYear, iMonth, iDay ); hb_dateStrPut( ffind->szDate, iYear, iMonth, iDay ); ffind->szDate[ 8 ] = '\0'; hb_snprintf( ffind->szTime, sizeof( ffind->szTime ), "%02d:%02d:%02d", iHour, iMin, iSec ); } hb_vmLock(); return bFound; } PHB_FFIND sfn_hb_fsFindFirst( const char * pszFileMask, ULONG attrmask, BOOL * bUse ) { PHB_FFIND ffind; ffind = ( PHB_FFIND ) hb_xgrab( sizeof( HB_FFIND ) ); memset( ffind, 0, sizeof( HB_FFIND ) ); /* Allocate platform dependent file find info storage */ ffind->info = ( void * ) hb_xgrab( sizeof( HB_FFIND_INFO ) ); memset( ffind->info, 0, sizeof( HB_FFIND_INFO ) ); /* Store search parameters */ ffind->pszFileMask = pszFileMask; ffind->attrmask = attrmask; ffind->bFirst = TRUE; /* Find first/next matching file */ if( sfn_hb_fsFindNext( ffind, bUse ) ) return ffind; /* If no file found at all, free stuff allocated so far and return NULL. */ sfn_hb_fsFindClose( ffind ); return NULL; } /* Finds next matching file, and applies a filter which makes searching CA-Cl*pper/MS-DOS compatible. */ BOOL sfn_hb_fsFindNext( PHB_FFIND ffind, BOOL * bUse ) { while( sfn_hb_fsFindNextLow( ffind, bUse ) ) { /* Filter the result to stay MS-DOS and CA-Cl*pper compatible. */ if( !( ( ( ffind->attrmask & HB_FA_HIDDEN ) == 0 && ( ffind->attr & HB_FA_HIDDEN ) != 0 ) || ( ( ffind->attrmask & HB_FA_SYSTEM ) == 0 && ( ffind->attr & HB_FA_SYSTEM ) != 0 ) || ( ( ffind->attrmask & HB_FA_LABEL ) == 0 && ( ffind->attr & HB_FA_LABEL ) != 0 ) || ( ( ffind->attrmask & HB_FA_DIRECTORY ) == 0 && ( ffind->attr & HB_FA_DIRECTORY ) != 0 ) ) ) { return TRUE; } } return FALSE; } void sfn_hb_fsFindClose( PHB_FFIND ffind ) { if( ffind ) { /* Do platform dependant cleanup */ if( ffind->info ) { PHB_FFIND_INFO info = ( PHB_FFIND_INFO ) ffind->info; hb_vmUnlock(); #if defined( HB_OS_DOS ) #if defined( __DJGPP__ ) || defined( __BORLANDC__ ) { HB_SYMBOL_UNUSED( info ); } #else { #if defined( __WATCOMC__ ) _dos_findclose( &info->entry ); #else findclose( &info->entry ); #endif } #endif #elif defined( HB_OS_OS2 ) { DosFindClose( info->hFindFile ); } #elif defined( HB_OS_WIN ) if( info->hFindFile != INVALID_HANDLE_VALUE ) { FindClose( info->hFindFile ); } #elif defined( HB_OS_UNIX ) if( info->dir ) { closedir( info->dir ); } #else { /* Intentionally do nothing */ int TODO; /* TODO: for given platform */ HB_SYMBOL_UNUSED( info ); } #endif hb_vmLock(); hb_xfree( ( void * ) ffind->info ); } hb_xfree( ( void * ) ffind ); } }
_______________________________________________ Harbour mailing list (attachment size limit: 40KB) Harbour@harbour-project.org http://lists.harbour-project.org/mailman/listinfo/harbour