/* | |
* dirent.c | |
* This file has no copyright assigned and is placed in the Public Domain. | |
* This file is a part of the mingw-runtime package. | |
* No warranty is given; refer to the file DISCLAIMER within the package. | |
* | |
* Derived from DIRLIB.C by Matt J. Weinstein | |
* This note appears in the DIRLIB.H | |
* DIRLIB.H by M. J. Weinstein Released to public domain 1-Jan-89 | |
* | |
* Updated by Jeremy Bettis <jeremy@hksys.com> | |
* Significantly revised and rewinddir, seekdir and telldir added by Colin | |
* Peters <colin@fu.is.saga-u.ac.jp> | |
* | |
*/ | |
#include <stdlib.h> | |
#include <errno.h> | |
#include <string.h> | |
#include <io.h> | |
#include <direct.h> | |
#include "dirent.h" | |
#define WIN32_LEAN_AND_MEAN | |
#include <windows.h> /* for GetFileAttributes */ | |
#include <tchar.h> | |
#ifdef _UNICODE | |
#define _tdirent _wdirent | |
#define _TDIR _WDIR | |
#define _topendir _wopendir | |
#define _tclosedir _wclosedir | |
#define _treaddir _wreaddir | |
#define _trewinddir _wrewinddir | |
#define _ttelldir _wtelldir | |
#define _tseekdir _wseekdir | |
#else | |
#define _tdirent dirent | |
#define _TDIR DIR | |
#define _topendir opendir | |
#define _tclosedir closedir | |
#define _treaddir readdir | |
#define _trewinddir rewinddir | |
#define _ttelldir telldir | |
#define _tseekdir seekdir | |
#endif | |
#define SUFFIX _T("*") | |
#define SLASH _T("\\") | |
/* | |
* opendir | |
* | |
* Returns a pointer to a DIR structure appropriately filled in to begin | |
* searching a directory. | |
*/ | |
_TDIR * | |
_topendir (const _TCHAR *szPath) | |
{ | |
_TDIR *nd; | |
unsigned int rc; | |
_TCHAR szFullPath[MAX_PATH]; | |
errno = 0; | |
if (!szPath) | |
{ | |
errno = EFAULT; | |
return (_TDIR *) 0; | |
} | |
if (szPath[0] == _T('\0')) | |
{ | |
errno = ENOTDIR; | |
return (_TDIR *) 0; | |
} | |
/* Attempt to determine if the given path really is a directory. */ | |
rc = GetFileAttributes (szPath); | |
if (rc == (unsigned int)-1) | |
{ | |
/* call GetLastError for more error info */ | |
errno = ENOENT; | |
return (_TDIR *) 0; | |
} | |
if (!(rc & FILE_ATTRIBUTE_DIRECTORY)) | |
{ | |
/* Error, entry exists but not a directory. */ | |
errno = ENOTDIR; | |
return (_TDIR *) 0; | |
} | |
/* Make an absolute pathname. */ | |
_tfullpath (szFullPath, szPath, MAX_PATH); | |
/* Allocate enough space to store DIR structure and the complete | |
* directory path given. */ | |
nd = (_TDIR *) malloc (sizeof (_TDIR) + (_tcslen(szFullPath) + _tcslen (SLASH) + | |
_tcslen(SUFFIX) + 1) * sizeof(_TCHAR)); | |
if (!nd) | |
{ | |
/* Error, out of memory. */ | |
errno = ENOMEM; | |
return (_TDIR *) 0; | |
} | |
/* Create the search expression. */ | |
_tcscpy (nd->dd_name, szFullPath); | |
/* Add on a slash if the path does not end with one. */ | |
if (nd->dd_name[0] != _T('\0') && | |
nd->dd_name[_tcslen (nd->dd_name) - 1] != _T('/') && | |
nd->dd_name[_tcslen (nd->dd_name) - 1] != _T('\\')) | |
{ | |
_tcscat (nd->dd_name, SLASH); | |
} | |
/* Add on the search pattern */ | |
_tcscat (nd->dd_name, SUFFIX); | |
/* Initialize handle to -1 so that a premature closedir doesn't try | |
* to call _findclose on it. */ | |
nd->dd_handle = -1; | |
/* Initialize the status. */ | |
nd->dd_stat = 0; | |
/* Initialize the dirent structure. ino and reclen are invalid under | |
* Win32, and name simply points at the appropriate part of the | |
* findfirst_t structure. */ | |
nd->dd_dir.d_ino = 0; | |
nd->dd_dir.d_reclen = 0; | |
nd->dd_dir.d_namlen = 0; | |
memset (nd->dd_dir.d_name, 0, sizeof (nd->dd_dir.d_name)); | |
return nd; | |
} | |
/* | |
* readdir | |
* | |
* Return a pointer to a dirent structure filled with the information on the | |
* next entry in the directory. | |
*/ | |
struct _tdirent * | |
_treaddir (_TDIR * dirp) | |
{ | |
errno = 0; | |
/* Check for valid DIR struct. */ | |
if (!dirp) | |
{ | |
errno = EFAULT; | |
return (struct _tdirent *) 0; | |
} | |
if (dirp->dd_stat < 0) | |
{ | |
/* We have already returned all files in the directory | |
* (or the structure has an invalid dd_stat). */ | |
return (struct _tdirent *) 0; | |
} | |
else if (dirp->dd_stat == 0) | |
{ | |
/* We haven't started the search yet. */ | |
/* Start the search */ | |
dirp->dd_handle = _tfindfirst (dirp->dd_name, &(dirp->dd_dta)); | |
if (dirp->dd_handle == -1) | |
{ | |
/* Whoops! Seems there are no files in that | |
* directory. */ | |
dirp->dd_stat = -1; | |
} | |
else | |
{ | |
dirp->dd_stat = 1; | |
} | |
} | |
else | |
{ | |
/* Get the next search entry. */ | |
if (_tfindnext (dirp->dd_handle, &(dirp->dd_dta))) | |
{ | |
/* We are off the end or otherwise error. | |
_findnext sets errno to ENOENT if no more file | |
Undo this. */ | |
DWORD winerr = GetLastError(); | |
if (winerr == ERROR_NO_MORE_FILES) | |
errno = 0; | |
_findclose (dirp->dd_handle); | |
dirp->dd_handle = -1; | |
dirp->dd_stat = -1; | |
} | |
else | |
{ | |
/* Update the status to indicate the correct | |
* number. */ | |
dirp->dd_stat++; | |
} | |
} | |
if (dirp->dd_stat > 0) | |
{ | |
/* Successfully got an entry. Everything about the file is | |
* already appropriately filled in except the length of the | |
* file name. */ | |
dirp->dd_dir.d_namlen = _tcslen (dirp->dd_dta.name); | |
_tcscpy (dirp->dd_dir.d_name, dirp->dd_dta.name); | |
return &dirp->dd_dir; | |
} | |
return (struct _tdirent *) 0; | |
} | |
/* | |
* closedir | |
* | |
* Frees up resources allocated by opendir. | |
*/ | |
int | |
_tclosedir (_TDIR * dirp) | |
{ | |
int rc; | |
errno = 0; | |
rc = 0; | |
if (!dirp) | |
{ | |
errno = EFAULT; | |
return -1; | |
} | |
if (dirp->dd_handle != -1) | |
{ | |
rc = _findclose (dirp->dd_handle); | |
} | |
/* Delete the dir structure. */ | |
free (dirp); | |
return rc; | |
} | |
/* | |
* rewinddir | |
* | |
* Return to the beginning of the directory "stream". We simply call findclose | |
* and then reset things like an opendir. | |
*/ | |
void | |
_trewinddir (_TDIR * dirp) | |
{ | |
errno = 0; | |
if (!dirp) | |
{ | |
errno = EFAULT; | |
return; | |
} | |
if (dirp->dd_handle != -1) | |
{ | |
_findclose (dirp->dd_handle); | |
} | |
dirp->dd_handle = -1; | |
dirp->dd_stat = 0; | |
} | |
/* | |
* telldir | |
* | |
* Returns the "position" in the "directory stream" which can be used with | |
* seekdir to go back to an old entry. We simply return the value in stat. | |
*/ | |
long | |
_ttelldir (_TDIR * dirp) | |
{ | |
errno = 0; | |
if (!dirp) | |
{ | |
errno = EFAULT; | |
return -1; | |
} | |
return dirp->dd_stat; | |
} | |
/* | |
* seekdir | |
* | |
* Seek to an entry previously returned by telldir. We rewind the directory | |
* and call readdir repeatedly until either dd_stat is the position number | |
* or -1 (off the end). This is not perfect, in that the directory may | |
* have changed while we weren't looking. But that is probably the case with | |
* any such system. | |
*/ | |
void | |
_tseekdir (_TDIR * dirp, long lPos) | |
{ | |
errno = 0; | |
if (!dirp) | |
{ | |
errno = EFAULT; | |
return; | |
} | |
if (lPos < -1) | |
{ | |
/* Seeking to an invalid position. */ | |
errno = EINVAL; | |
return; | |
} | |
else if (lPos == -1) | |
{ | |
/* Seek past end. */ | |
if (dirp->dd_handle != -1) | |
{ | |
_findclose (dirp->dd_handle); | |
} | |
dirp->dd_handle = -1; | |
dirp->dd_stat = -1; | |
} | |
else | |
{ | |
/* Rewind and read forward to the appropriate index. */ | |
_trewinddir (dirp); | |
while ((dirp->dd_stat < lPos) && _treaddir (dirp)) | |
; | |
} | |
} |