xref: /petsc/src/sys/fileio/ftest.c (revision 5b6bfdb9644f185dbf5e5a09b808ec241507e1e7)
1 
2 #include <petscsys.h>
3 #include <errno.h>
4 #if defined(PETSC_HAVE_PWD_H)
5 #include <pwd.h>
6 #endif
7 #include <ctype.h>
8 #include <sys/stat.h>
9 #if defined(PETSC_HAVE_UNISTD_H)
10 #include <unistd.h>
11 #endif
12 #if defined(PETSC_HAVE_SYS_UTSNAME_H)
13 #include <sys/utsname.h>
14 #endif
15 #if defined(PETSC_HAVE_IO_H)
16 #include <io.h>
17 #endif
18 #if defined(PETSC_HAVE_SYS_SYSTEMINFO_H)
19 #include <sys/systeminfo.h>
20 #endif
21 
22 #if defined(PETSC_HAVE__ACCESS) || defined(PETSC_HAVE_ACCESS)
23 
24 static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool  *flg)
25 {
26   int            m = R_OK;
27   PetscErrorCode ierr;
28 
29   PetscFunctionBegin;
30   if (mode == 'r') m = R_OK;
31   else if (mode == 'w') m = W_OK;
32   else if (mode == 'x') m = X_OK;
33   else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONG, "Mode must be one of r, w, or x");
34 #if defined(PETSC_HAVE_ACCESS)
35   if (!access(fname, m)) {
36     ierr = PetscInfo1(NULL,"System call access() succeeded on file %s\n",fname);CHKERRQ(ierr);
37     *flg = PETSC_TRUE;
38   } else {
39     ierr = PetscInfo1(NULL,"System call access() failed on file %s\n",fname);CHKERRQ(ierr);
40     *flg = PETSC_FALSE;
41   }
42 #else
43   if (m == X_OK) SETERRQ1(PETSC_COMM_SELF,PETSC_ERR_SUP, "Unable to check execute permission for file %s", fname);
44   if (!_access(fname, m)) *flg = PETSC_TRUE;
45 #endif
46   PetscFunctionReturn(0);
47 }
48 
49 #else  /* PETSC_HAVE_ACCESS or PETSC_HAVE__ACCESS */
50 
51 static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool  *flg)
52 {
53   uid_t          uid;
54   gid_t          *gid = NULL;
55   int            numGroups;
56   int            rbit = S_IROTH;
57   int            wbit = S_IWOTH;
58   int            ebit = S_IXOTH;
59   PetscErrorCode ierr;
60 
61   PetscFunctionBegin;
62   /* Get the number of supplementary group IDs */
63 #if !defined(PETSC_MISSING_GETGROUPS)
64   numGroups = getgroups(0, gid); if (numGroups < 0) SETERRQ(PETSC_COMM_SELF,numGroups, "Unable to count supplementary group IDs");
65   ierr = PetscMalloc1(numGroups+1, &gid);CHKERRQ(ierr);
66 #else
67   numGroups = 0;
68 #endif
69 
70   /* Get the (effective) user and group of the caller */
71   uid    = geteuid();
72   gid[0] = getegid();
73 
74   /* Get supplementary group IDs */
75 #if !defined(PETSC_MISSING_GETGROUPS)
76   ierr = getgroups(numGroups, gid+1); if (ierr < 0) SETERRQ(PETSC_COMM_SELF,ierr, "Unable to obtain supplementary group IDs");
77 #endif
78 
79   /* Test for accessibility */
80   if (fuid == uid) {
81     rbit = S_IRUSR;
82     wbit = S_IWUSR;
83     ebit = S_IXUSR;
84   } else {
85     int g;
86 
87     for (g = 0; g <= numGroups; g++) {
88       if (fgid == gid[g]) {
89         rbit = S_IRGRP;
90         wbit = S_IWGRP;
91         ebit = S_IXGRP;
92         break;
93       }
94     }
95   }
96   ierr = PetscFree(gid);CHKERRQ(ierr);
97 
98   if (mode == 'r') {
99     if (fmode & rbit) *flg = PETSC_TRUE;
100   } else if (mode == 'w') {
101     if (fmode & wbit) *flg = PETSC_TRUE;
102   } else if (mode == 'x') {
103     if (fmode & ebit) *flg = PETSC_TRUE;
104   }
105   PetscFunctionReturn(0);
106 }
107 
108 #endif /* PETSC_HAVE_ACCESS */
109 
110 static PetscErrorCode PetscGetFileStat(const char fname[], uid_t *fileUid, gid_t *fileGid, int *fileMode,PetscBool  *exists)
111 {
112   struct stat    statbuf;
113   PetscErrorCode ierr;
114 
115   PetscFunctionBegin;
116   *fileMode = 0;
117   *exists = PETSC_FALSE;
118 #if defined(PETSC_HAVE_STAT_NO_CONST)
119   ierr = stat((char*) fname, &statbuf);
120 #else
121   ierr = stat(fname, &statbuf);
122 #endif
123   if (ierr) {
124 #if defined(EOVERFLOW)
125     if (errno == EOVERFLOW) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SYS,"EOVERFLOW in stat(), configure PETSc --with-large-file-io=1 to support files larger than 2GiB");
126 #endif
127     ierr    = PetscInfo1(NULL,"System call stat() failed on file %s\n",fname);CHKERRQ(ierr);
128     *exists = PETSC_FALSE;
129   } else {
130     ierr      = PetscInfo1(NULL,"System call stat() succeeded on file %s\n",fname);CHKERRQ(ierr);
131     *exists   = PETSC_TRUE;
132     *fileUid  = statbuf.st_uid;
133     *fileGid  = statbuf.st_gid;
134     *fileMode = statbuf.st_mode;
135   }
136   PetscFunctionReturn(0);
137 }
138 
139 /*@C
140    PetscTestFile - checks for the existence of a file
141 
142    Not Collective
143 
144    Input Parameter:
145 +  fname - the filename
146 -  mode - either 'r', 'w', or 'x'
147 
148    Output Parameter:
149 .  flg - the file exists and satisfies the mode
150 
151    Level: intermediate
152 
153 .seealso: PetscTestDirectory(), PetscLs()
154 @*/
155 PetscErrorCode  PetscTestFile(const char fname[], char mode, PetscBool  *flg)
156 {
157   uid_t          fuid;
158   gid_t          fgid;
159   int            fmode;
160   PetscErrorCode ierr;
161   PetscBool      exists;
162 
163   PetscFunctionBegin;
164   *flg = PETSC_FALSE;
165   if (!fname) PetscFunctionReturn(0);
166 
167   ierr = PetscGetFileStat(fname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
168   if (!exists) PetscFunctionReturn(0);
169   /* Except for systems that have this broken stat macros (rare), this is the correct way to check for a regular file */
170   if (!S_ISREG(fmode)) PetscFunctionReturn(0);
171 
172   ierr = PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
173   PetscFunctionReturn(0);
174 }
175 
176 /*@C
177    PetscTestDirectory - checks for the existence of a directory
178 
179    Not Collective
180 
181    Input Parameter:
182 +  dirname - the directory name
183 -  mode - either 'r', 'w', or 'x'
184 
185    Output Parameter:
186 .  flg - the directory exists and satisfies the mode
187 
188    Level: intermediate
189 
190 .seealso: PetscTestFile(), PetscLs()
191 @*/
192 PetscErrorCode  PetscTestDirectory(const char dirname[],char mode,PetscBool  *flg)
193 {
194   uid_t          fuid;
195   gid_t          fgid;
196   int            fmode;
197   PetscErrorCode ierr;
198   PetscBool      exists;
199 
200   PetscFunctionBegin;
201   *flg = PETSC_FALSE;
202   if (!dirname) PetscFunctionReturn(0);
203 
204   ierr = PetscGetFileStat(dirname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
205   if (!exists) PetscFunctionReturn(0);
206   /* Except for systems that have this broken stat macros (rare), this
207      is the correct way to check for a directory */
208   if (!S_ISDIR(fmode)) PetscFunctionReturn(0);
209 
210   ierr = PetscTestOwnership(dirname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
211   PetscFunctionReturn(0);
212 }
213 
214 /*@C
215    PetscLs - produce a listing of the files in a directory
216 
217    Collective on MPI_Comm
218 
219    Input Parameter:
220 +  comm - the MPI communicator
221 .  dirname - the directory name
222 -  tlen - the length of the buffer found[]
223 
224    Output Parameter:
225 +  found - listing of files
226 -  flg - the directory exists
227 
228    Level: intermediate
229 
230 .seealso: PetscTestFile(), PetscLs()
231 @*/
232 PetscErrorCode  PetscLs(MPI_Comm comm,const char dirname[],char found[],size_t tlen,PetscBool  *flg)
233 {
234   PetscErrorCode ierr;
235   size_t         len;
236   char           *f,program[PETSC_MAX_PATH_LEN];
237   FILE           *fp;
238 
239   PetscFunctionBegin;
240   ierr = PetscStrcpy(program,"ls ");CHKERRQ(ierr);
241   ierr = PetscStrcat(program,dirname);CHKERRQ(ierr);
242 #if defined(PETSC_HAVE_POPEN)
243   ierr = PetscPOpen(comm,NULL,program,"r",&fp);CHKERRQ(ierr);
244 #else
245   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
246 #endif
247   f = fgets(found,tlen,fp);
248   if (f) *flg = PETSC_TRUE;
249   else *flg = PETSC_FALSE;
250   while (f) {
251     ierr = PetscStrlen(found,&len);CHKERRQ(ierr);
252     f    = fgets(found+len,tlen-len,fp);
253   }
254   if (*flg) {ierr = PetscInfo2(0,"ls on %s gives \n%s\n",dirname,found);CHKERRQ(ierr);}
255 #if defined(PETSC_HAVE_POPEN)
256   ierr = PetscPClose(comm,fp);CHKERRQ(ierr);
257 #else
258   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
259 #endif
260   PetscFunctionReturn(0);
261 }
262