| /* |
| * 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)) |
| ; |
| } |
| } |