PostgreSQL Source Code  git master
dirent.c
Go to the documentation of this file.
1 /*-------------------------------------------------------------------------
2  *
3  * dirent.c
4  * opendir/readdir/closedir for win32/msvc
5  *
6  * Portions Copyright (c) 1996-2017, PostgreSQL Global Development Group
7  * Portions Copyright (c) 1994, Regents of the University of California
8  *
9  *
10  * IDENTIFICATION
11  * src/port/dirent.c
12  *
13  *-------------------------------------------------------------------------
14  */
15 
16 #ifndef FRONTEND
17 #include "postgres.h"
18 #else
19 #include "postgres_fe.h"
20 #endif
21 
22 #include <dirent.h>
23 
24 
25 struct DIR
26 {
27  char *dirname;
28  struct dirent ret; /* Used to return to caller */
29  HANDLE handle;
30 };
31 
32 DIR *
33 opendir(const char *dirname)
34 {
35  DWORD attr;
36  DIR *d;
37 
38  /* Make sure it is a directory */
39  attr = GetFileAttributes(dirname);
40  if (attr == INVALID_FILE_ATTRIBUTES)
41  {
42  errno = ENOENT;
43  return NULL;
44  }
45  if ((attr & FILE_ATTRIBUTE_DIRECTORY) != FILE_ATTRIBUTE_DIRECTORY)
46  {
47  errno = ENOTDIR;
48  return NULL;
49  }
50 
51  d = malloc(sizeof(DIR));
52  if (!d)
53  {
54  errno = ENOMEM;
55  return NULL;
56  }
57  d->dirname = malloc(strlen(dirname) + 4);
58  if (!d->dirname)
59  {
60  errno = ENOMEM;
61  free(d);
62  return NULL;
63  }
64  strcpy(d->dirname, dirname);
65  if (d->dirname[strlen(d->dirname) - 1] != '/' &&
66  d->dirname[strlen(d->dirname) - 1] != '\\')
67  strcat(d->dirname, "\\"); /* Append backslash if not already there */
68  strcat(d->dirname, "*"); /* Search for entries named anything */
69  d->handle = INVALID_HANDLE_VALUE;
70  d->ret.d_ino = 0; /* no inodes on win32 */
71  d->ret.d_reclen = 0; /* not used on win32 */
72 
73  return d;
74 }
75 
76 struct dirent *
78 {
79  WIN32_FIND_DATA fd;
80 
81  if (d->handle == INVALID_HANDLE_VALUE)
82  {
83  d->handle = FindFirstFile(d->dirname, &fd);
84  if (d->handle == INVALID_HANDLE_VALUE)
85  {
86  errno = ENOENT;
87  return NULL;
88  }
89  }
90  else
91  {
92  if (!FindNextFile(d->handle, &fd))
93  {
94  if (GetLastError() == ERROR_NO_MORE_FILES)
95  {
96  /* No more files, force errno=0 (unlike mingw) */
97  errno = 0;
98  return NULL;
99  }
100  _dosmaperr(GetLastError());
101  return NULL;
102  }
103  }
104  strcpy(d->ret.d_name, fd.cFileName); /* Both strings are MAX_PATH long */
105  d->ret.d_namlen = strlen(d->ret.d_name);
106 
107  return &d->ret;
108 }
109 
110 int
112 {
113  int ret = 0;
114 
115  if (d->handle != INVALID_HANDLE_VALUE)
116  ret = !FindClose(d->handle);
117  free(d->dirname);
118  free(d);
119 
120  return ret;
121 }
HANDLE handle
Definition: dirent.c:29
unsigned short d_reclen
Definition: dirent.h:12
void _dosmaperr(unsigned long)
Definition: win32error.c:171
Definition: dirent.h:9
static int fd(const char *x, int i)
Definition: preproc-init.c:105
#define malloc(a)
Definition: header.h:50
struct dirent * readdir(DIR *d)
Definition: dirent.c:77
unsigned short d_namlen
Definition: dirent.h:13
Definition: dirent.c:25
char * dirname
Definition: dirent.c:27
long d_ino
Definition: dirent.h:11
struct dirent ret
Definition: dirent.c:28
#define free(a)
Definition: header.h:65
DIR * opendir(const char *dirname)
Definition: dirent.c:33
int closedir(DIR *d)
Definition: dirent.c:111
char d_name[MAX_PATH]
Definition: dirent.h:14